xref: /openbmc/qemu/plugins/api-system.c (revision 6d1829fce4ea50d343f2df63eeff96685a359bf5)
1903e870fSAlex Bennée /*
2903e870fSAlex Bennée  * QEMU Plugin API - System specific implementations
3903e870fSAlex Bennée  *
4903e870fSAlex Bennée  * This provides the APIs that have a specific system implementation
5903e870fSAlex Bennée  * or are only relevant to system-mode.
6903e870fSAlex Bennée  *
7903e870fSAlex Bennée  * Copyright (C) 2017, Emilio G. Cota <cota@braap.org>
8903e870fSAlex Bennée  * Copyright (C) 2019-2025, Linaro
9903e870fSAlex Bennée  *
10903e870fSAlex Bennée  * SPDX-License-Identifier: GPL-2.0-or-later
11903e870fSAlex Bennée  */
12903e870fSAlex Bennée 
13903e870fSAlex Bennée #include "qemu/osdep.h"
14903e870fSAlex Bennée #include "qemu/main-loop.h"
15455a2d26SAlex Bennée #include "qapi/error.h"
16455a2d26SAlex Bennée #include "migration/blocker.h"
17455a2d26SAlex Bennée #include "hw/boards.h"
18455a2d26SAlex Bennée #include "qemu/plugin-memory.h"
19903e870fSAlex Bennée #include "qemu/plugin.h"
20903e870fSAlex Bennée 
21903e870fSAlex Bennée /*
22903e870fSAlex Bennée  * In system mode we cannot trace the binary being executed so the
23903e870fSAlex Bennée  * helpers all return NULL/0.
24903e870fSAlex Bennée  */
qemu_plugin_path_to_binary(void)25903e870fSAlex Bennée const char *qemu_plugin_path_to_binary(void)
26903e870fSAlex Bennée {
27903e870fSAlex Bennée     return NULL;
28903e870fSAlex Bennée }
29903e870fSAlex Bennée 
qemu_plugin_start_code(void)30903e870fSAlex Bennée uint64_t qemu_plugin_start_code(void)
31903e870fSAlex Bennée {
32903e870fSAlex Bennée     return 0;
33903e870fSAlex Bennée }
34903e870fSAlex Bennée 
qemu_plugin_end_code(void)35903e870fSAlex Bennée uint64_t qemu_plugin_end_code(void)
36903e870fSAlex Bennée {
37903e870fSAlex Bennée     return 0;
38903e870fSAlex Bennée }
39903e870fSAlex Bennée 
qemu_plugin_entry_code(void)40903e870fSAlex Bennée uint64_t qemu_plugin_entry_code(void)
41903e870fSAlex Bennée {
42903e870fSAlex Bennée     return 0;
43903e870fSAlex Bennée }
44455a2d26SAlex Bennée 
45455a2d26SAlex Bennée /*
46455a2d26SAlex Bennée  * Virtual Memory queries
47455a2d26SAlex Bennée  */
48455a2d26SAlex Bennée 
49455a2d26SAlex Bennée static __thread struct qemu_plugin_hwaddr hwaddr_info;
50455a2d26SAlex Bennée 
qemu_plugin_get_hwaddr(qemu_plugin_meminfo_t info,uint64_t vaddr)51455a2d26SAlex Bennée struct qemu_plugin_hwaddr *qemu_plugin_get_hwaddr(qemu_plugin_meminfo_t info,
52455a2d26SAlex Bennée                                                   uint64_t vaddr)
53455a2d26SAlex Bennée {
54455a2d26SAlex Bennée     CPUState *cpu = current_cpu;
55455a2d26SAlex Bennée     unsigned int mmu_idx = get_mmuidx(info);
56455a2d26SAlex Bennée     enum qemu_plugin_mem_rw rw = get_plugin_meminfo_rw(info);
57455a2d26SAlex Bennée     hwaddr_info.is_store = (rw & QEMU_PLUGIN_MEM_W) != 0;
58455a2d26SAlex Bennée 
59455a2d26SAlex Bennée     assert(mmu_idx < NB_MMU_MODES);
60455a2d26SAlex Bennée 
61455a2d26SAlex Bennée     if (!tlb_plugin_lookup(cpu, vaddr, mmu_idx,
62455a2d26SAlex Bennée                            hwaddr_info.is_store, &hwaddr_info)) {
63455a2d26SAlex Bennée         error_report("invalid use of qemu_plugin_get_hwaddr");
64455a2d26SAlex Bennée         return NULL;
65455a2d26SAlex Bennée     }
66455a2d26SAlex Bennée 
67455a2d26SAlex Bennée     return &hwaddr_info;
68455a2d26SAlex Bennée }
69455a2d26SAlex Bennée 
qemu_plugin_hwaddr_is_io(const struct qemu_plugin_hwaddr * haddr)70455a2d26SAlex Bennée bool qemu_plugin_hwaddr_is_io(const struct qemu_plugin_hwaddr *haddr)
71455a2d26SAlex Bennée {
72455a2d26SAlex Bennée     return haddr->is_io;
73455a2d26SAlex Bennée }
74455a2d26SAlex Bennée 
qemu_plugin_hwaddr_phys_addr(const struct qemu_plugin_hwaddr * haddr)75455a2d26SAlex Bennée uint64_t qemu_plugin_hwaddr_phys_addr(const struct qemu_plugin_hwaddr *haddr)
76455a2d26SAlex Bennée {
77455a2d26SAlex Bennée     if (haddr) {
78455a2d26SAlex Bennée         return haddr->phys_addr;
79455a2d26SAlex Bennée     }
80455a2d26SAlex Bennée     return 0;
81455a2d26SAlex Bennée }
82455a2d26SAlex Bennée 
qemu_plugin_hwaddr_device_name(const struct qemu_plugin_hwaddr * h)83455a2d26SAlex Bennée const char *qemu_plugin_hwaddr_device_name(const struct qemu_plugin_hwaddr *h)
84455a2d26SAlex Bennée {
85455a2d26SAlex Bennée     if (h && h->is_io) {
86455a2d26SAlex Bennée         MemoryRegion *mr = h->mr;
87455a2d26SAlex Bennée         if (!mr->name) {
88455a2d26SAlex Bennée             unsigned maddr = (uintptr_t)mr;
89455a2d26SAlex Bennée             g_autofree char *temp = g_strdup_printf("anon%08x", maddr);
90455a2d26SAlex Bennée             return g_intern_string(temp);
91455a2d26SAlex Bennée         } else {
92455a2d26SAlex Bennée             return g_intern_string(mr->name);
93455a2d26SAlex Bennée         }
94455a2d26SAlex Bennée     } else {
95455a2d26SAlex Bennée         return g_intern_static_string("RAM");
96455a2d26SAlex Bennée     }
97455a2d26SAlex Bennée }
98*1d3e745fSAlex Bennée 
99*1d3e745fSAlex Bennée /*
100*1d3e745fSAlex Bennée  * Time control
101*1d3e745fSAlex Bennée  */
102*1d3e745fSAlex Bennée static bool has_control;
103*1d3e745fSAlex Bennée static Error *migration_blocker;
104*1d3e745fSAlex Bennée 
qemu_plugin_request_time_control(void)105*1d3e745fSAlex Bennée const void *qemu_plugin_request_time_control(void)
106*1d3e745fSAlex Bennée {
107*1d3e745fSAlex Bennée     if (!has_control) {
108*1d3e745fSAlex Bennée         has_control = true;
109*1d3e745fSAlex Bennée         error_setg(&migration_blocker,
110*1d3e745fSAlex Bennée                    "TCG plugin time control does not support migration");
111*1d3e745fSAlex Bennée         migrate_add_blocker(&migration_blocker, NULL);
112*1d3e745fSAlex Bennée         return &has_control;
113*1d3e745fSAlex Bennée     }
114*1d3e745fSAlex Bennée     return NULL;
115*1d3e745fSAlex Bennée }
116*1d3e745fSAlex Bennée 
advance_virtual_time__async(CPUState * cpu,run_on_cpu_data data)117*1d3e745fSAlex Bennée static void advance_virtual_time__async(CPUState *cpu, run_on_cpu_data data)
118*1d3e745fSAlex Bennée {
119*1d3e745fSAlex Bennée     int64_t new_time = data.host_ulong;
120*1d3e745fSAlex Bennée     qemu_clock_advance_virtual_time(new_time);
121*1d3e745fSAlex Bennée }
122*1d3e745fSAlex Bennée 
qemu_plugin_update_ns(const void * handle,int64_t new_time)123*1d3e745fSAlex Bennée void qemu_plugin_update_ns(const void *handle, int64_t new_time)
124*1d3e745fSAlex Bennée {
125*1d3e745fSAlex Bennée     if (handle == &has_control) {
126*1d3e745fSAlex Bennée         /* Need to execute out of cpu_exec, so bql can be locked. */
127*1d3e745fSAlex Bennée         async_run_on_cpu(current_cpu,
128*1d3e745fSAlex Bennée                          advance_virtual_time__async,
129*1d3e745fSAlex Bennée                          RUN_ON_CPU_HOST_ULONG(new_time));
130*1d3e745fSAlex Bennée     }
131*1d3e745fSAlex Bennée }
132