From nobody Fri Apr 4 04:51:00 2025 Delivered-To: importer@patchew.org Authentication-Results: mx.zohomail.com; spf=pass (zohomail.com: domain of gnu.org designates 209.51.188.17 as permitted sender) smtp.mailfrom=qemu-devel-bounces+importer=patchew.org@nongnu.org Return-Path: Received: from lists.gnu.org (lists.gnu.org [209.51.188.17]) by mx.zohomail.com with SMTPS id 1741620043096525.5172929935591; Mon, 10 Mar 2025 08:20:43 -0700 (PDT) Received: from localhost ([::1] helo=lists1p.gnu.org) by lists.gnu.org with esmtp (Exim 4.90_1) (envelope-from ) id 1tren7-00029E-6A; Mon, 10 Mar 2025 11:11:41 -0400 Received: from eggs.gnu.org ([2001:470:142:3::10]) by lists.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.90_1) (envelope-from ) id 1tren2-00024I-7r for qemu-devel@nongnu.org; Mon, 10 Mar 2025 11:11:36 -0400 Received: from smtp1.lauterbach.com ([62.154.241.196]) by eggs.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.90_1) (envelope-from ) id 1tremw-0007gY-Me for qemu-devel@nongnu.org; Mon, 10 Mar 2025 11:11:35 -0400 Received: (qmail 30771 invoked by uid 484); 10 Mar 2025 15:11:11 -0000 Received: from unknown (HELO mflpc1.LTB.LAN) (Authenticated_SSL:mfleischmann@[10.2.13.100]) (envelope-sender ) by smtp1.lauterbach.com (qmail-ldap-1.03) with TLS_AES_256_GCM_SHA384 encrypted SMTP for ; 10 Mar 2025 15:11:11 -0000 X-Qmail-Scanner-Diagnostics: from 10.2.13.100 by smtp1.lauterbach.com (envelope-from , uid 484) with qmail-scanner-2.11 (mhr: 1.0. clamdscan: 0.99/21437. spamassassin: 3.4.0. Clear:RC:1(10.2.13.100):. Processed in 0.013161 secs); 10 Mar 2025 15:11:11 -0000 From: Mario Fleischmann To: qemu-devel@nongnu.org Cc: alex.bennee@linaro.org, philmd@linaro.org, armbru@redhat.com, christian.boenig@lauterbach.com, Mario Fleischmann , Eric Blake , Fabiano Rosas , Laurent Vivier , Paolo Bonzini Subject: [PATCH 07/16] mcd: Implement memory space query Date: Mon, 10 Mar 2025 16:05:01 +0100 Message-Id: <20250310150510.200607-8-mario.fleischmann@lauterbach.com> X-Mailer: git-send-email 2.34.1 In-Reply-To: <20250310150510.200607-1-mario.fleischmann@lauterbach.com> References: <20250310150510.200607-1-mario.fleischmann@lauterbach.com> MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable X-Qmail-Scanner-2.11: added fake Content-Type header Received-SPF: pass (zohomail.com: domain of gnu.org designates 209.51.188.17 as permitted sender) client-ip=209.51.188.17; envelope-from=qemu-devel-bounces+importer=patchew.org@nongnu.org; helo=lists.gnu.org; Received-SPF: pass client-ip=62.154.241.196; envelope-from=mario.fleischmann@lauterbach.com; helo=smtp1.lauterbach.com X-Spam_score_int: -18 X-Spam_score: -1.9 X-Spam_bar: - X-Spam_report: (-1.9 / 5.0 requ) BAYES_00=-1.9, RCVD_IN_VALIDITY_RPBL_BLOCKED=0.001, RCVD_IN_VALIDITY_SAFE_BLOCKED=0.001, SPF_HELO_PASS=-0.001, SPF_PASS=-0.001 autolearn=ham autolearn_force=no X-Spam_action: no action X-BeenThere: qemu-devel@nongnu.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: qemu-devel-bounces+importer=patchew.org@nongnu.org Sender: qemu-devel-bounces+importer=patchew.org@nongnu.org X-ZM-MESSAGEID: 1741620045887019100 Content-Type: text/plain; charset="utf-8" Support three main memory space types: * Physical memory * Logical memory (MMU) * GDB Registers Use custom memory type to mark memory spaces as secure V=3D1 QTEST_QEMU_BINARY=3D"./qemu-system-arm -M virt,secure=3Don -cpu corte= x-a15" tests/qtest/mcd-test Signed-off-by: Mario Fleischmann --- mcd/libmcd_qapi.c | 22 +++++ mcd/libmcd_qapi.h | 2 + mcd/mcdserver.c | 199 +++++++++++++++++++++++++++++++------- mcd/mcdstub_qapi.c | 44 +++++++++ qapi/mcd.json | 199 ++++++++++++++++++++++++++++++++++++++ tests/qtest/libmcd-test.c | 20 ++++ tests/qtest/libmcd-test.h | 3 + tests/qtest/mcd-test.c | 79 +++++++++++++++ 8 files changed, 534 insertions(+), 34 deletions(-) diff --git a/mcd/libmcd_qapi.c b/mcd/libmcd_qapi.c index cb65643110..0421152705 100644 --- a/mcd/libmcd_qapi.c +++ b/mcd/libmcd_qapi.c @@ -125,3 +125,25 @@ mcd_core_con_info_st unmarshal_mcd_core_con_info(MCDCo= reConInfo *con_info) =20 return unmarshal; } + +MCDMemspace *marshal_mcd_memspace(const mcd_memspace_st *mem_space) +{ + MCDMemspace *marshal =3D g_malloc0(sizeof(*marshal)); + + *marshal =3D (MCDMemspace) { + .mem_space_id =3D mem_space->mem_space_id, + .mem_space_name =3D g_strdup(mem_space->mem_space_name), + .mem_type =3D mem_space->mem_type, + .bits_per_mau =3D mem_space->bits_per_mau, + .invariance =3D mem_space->invariance, + .endian =3D mem_space->endian, + .min_addr =3D mem_space->min_addr, + .max_addr =3D mem_space->max_addr, + .num_mem_blocks =3D mem_space->num_mem_blocks, + .supported_access_options =3D mem_space->supported_access_options, + .core_mode_mask_read =3D mem_space->core_mode_mask_read, + .core_mode_mask_write =3D mem_space->core_mode_mask_write, + }; + + return marshal; +} diff --git a/mcd/libmcd_qapi.h b/mcd/libmcd_qapi.h index ceb4b438bb..7e874dec25 100644 --- a/mcd/libmcd_qapi.h +++ b/mcd/libmcd_qapi.h @@ -25,6 +25,8 @@ MCDServerInfo *marshal_mcd_server_info(const mcd_server_i= nfo_st *server_info); =20 MCDCoreConInfo *marshal_mcd_core_con_info(const mcd_core_con_info_st *con_= info); =20 +MCDMemspace *marshal_mcd_memspace(const mcd_memspace_st *mem_space); + mcd_api_version_st unmarshal_mcd_api_version(MCDAPIVersion *api_version); =20 mcd_core_con_info_st unmarshal_mcd_core_con_info(MCDCoreConInfo *con_info); diff --git a/mcd/mcdserver.c b/mcd/mcdserver.c index 83ffa4f097..f924bf3799 100644 --- a/mcd/mcdserver.c +++ b/mcd/mcdserver.c @@ -12,6 +12,10 @@ #include "qemu/cutils.h" #include "mcd_api.h" #include "hw/boards.h" +#include "exec/tswap.h" + +/* Custom memory space type */ +static const mcd_mem_type_et MCD_MEM_SPACE_IS_SECURE =3D 0x00010000; =20 static const mcd_error_info_st MCD_ERROR_NOT_IMPLEMENTED =3D { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, @@ -48,37 +52,45 @@ static const mcd_error_info_st MCD_ERROR_NONE =3D { .error_str =3D "", }; =20 -/* reserves memory for custom errors */ -static mcd_error_info_st custom_mcd_error; - /** * struct mcdcore_state - State of a core. * - * @last_error: Error info of most recent executed function. - * @info: Core connection information. - * @open_core: Open core instance as allocated in mcd_open_core_f(). + * @last_error: Error info of most recent executed core-related functio= n. + * @custom_error: Reserves memory for custom MCD errors. + * @info: Core connection information. + * @open_core: Open core instance as allocated in mcd_open_core_f(). + * @cpu: QEMU's internal CPU handle. + * @memory_spaces: Memory spaces as queried by mcd_qry_mem_spaces_f(). * * MCD is mainly being used on the core level: * After the initial query functions, a core connection is opened in * mcd_open_core_f(). The allocated mcd_core_st instance is then the basis * of subsequent operations. + * + * @cpu is the internal CPU handle through which core specific debug + * functions are implemented. */ typedef struct mcdcore_state { const mcd_error_info_st *last_error; + mcd_error_info_st custom_error; mcd_core_con_info_st info; mcd_core_st *open_core; + CPUState *cpu; + GArray *memory_spaces; } mcdcore_state; =20 /** * struct mcdserver_state - State of the MCD server * - * @last_error: Error info of most recent executed function. - * @open_server: Open server instance as allocated in mcd_open_server_f(). - * @system_key: System key as provided in mcd_open_server_f() - * @cores: Internal core information database. + * @last_error: Error info of most recent executed function. + * @custom_error: Reserves memory for custom MCD errors. + * @open_server: Open server instance as allocated in mcd_open_server_f(). + * @system_key: System key as provided in mcd_open_server_f() + * @cores: Internal core information database. */ typedef struct mcdserver_state { const mcd_error_info_st *last_error; + mcd_error_info_st custom_error; mcd_server_st *open_server; char system_key[MCD_KEY_LEN]; GArray *cores; @@ -134,13 +146,13 @@ mcd_return_et mcd_initialize_f(const mcd_api_version_= st *version_req, version_req->v_api_minor <=3D MCD_API_VER_MINOR) { g_server_state.last_error =3D &MCD_ERROR_NONE; } else { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_GENERAL, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "incompatible versions", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; } =20 return g_server_state.last_error->return_status; @@ -160,13 +172,13 @@ mcd_return_et mcd_qry_servers_f(const char *host, boo= l running, mcd_server_info_st *server_info) { if (start_index >=3D 1) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_PARAM, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "QEMU only has one MCD server", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -214,13 +226,13 @@ mcd_return_et mcd_open_server_f(const char *system_ke= y, CPUState *cpu; =20 if (g_server_state.open_server) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_CONNECTION, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "server already open", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -253,6 +265,8 @@ mcd_return_et mcd_open_server_f(const char *system_key, }, .last_error =3D &MCD_ERROR_NONE, .open_core =3D NULL, + .cpu =3D cpu, + .memory_spaces =3D g_array_new(false, true, sizeof(mcd_memspac= e_st)), }; pstrcpy(c.info.core, MCD_UNIQUE_NAME_LEN, cpu_model); g_array_append_val(g_server_state.cores, c); @@ -265,24 +279,24 @@ mcd_return_et mcd_open_server_f(const char *system_ke= y, mcd_return_et mcd_close_server_f(const mcd_server_st *server) { if (!g_server_state.open_server) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_CONNECTION, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "server not open", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 if (server !=3D g_server_state.open_server) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_CONNECTION, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "unknown server", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -333,13 +347,13 @@ mcd_return_et mcd_qry_systems_f(uint32_t start_index,= uint32_t *num_systems, } =20 if (start_index >=3D 1) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_PARAM, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "QEMU only emulates one system", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -381,13 +395,13 @@ mcd_return_et mcd_qry_devices_f(const mcd_core_con_in= fo_st *system_con_info, } =20 if (start_index >=3D 1) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_PARAM, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "QEMU only emulates one machine", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -431,13 +445,13 @@ mcd_return_et mcd_qry_cores_f(const mcd_core_con_info= _st *connection_info, } =20 if (start_index >=3D g_server_state.cores->len) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_PARAM, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "start_index exceeds the number of cores", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 @@ -471,6 +485,59 @@ mcd_return_et mcd_qry_core_modes_f(const mcd_core_st *= core, return g_server_state.last_error->return_status; } =20 +static mcd_return_et query_memspaces(mcdcore_state *core_state) +{ + g_array_set_size(core_state->memory_spaces, 0); + + mcd_endian_et endian =3D target_words_bigendian() + ? MCD_ENDIAN_BIG : MCD_ENDIAN_LITTLE; + + for (uint32_t i =3D 0; i < core_state->cpu->num_ases; i++) { + AddressSpace *as =3D cpu_get_address_space(core_state->cpu, i); + + int secure_flag =3D 0; + if (core_state->cpu->num_ases > 1) { + int sid =3D cpu_asidx_from_attrs(core_state->cpu, + (MemTxAttrs) { .secure =3D 1 }); + if (i =3D=3D sid) { + secure_flag =3D MCD_MEM_SPACE_IS_SECURE; + } + } + + const char *as_name =3D as->name; + const char *mr_name =3D as->root->name; + + mcd_memspace_st physical =3D { + /* mem space ID 0 is reserved */ + .mem_space_id =3D core_state->memory_spaces->len + 1, + .mem_type =3D MCD_MEM_SPACE_IS_PHYSICAL | secure_flag, + .endian =3D endian, + }; + strncpy(physical.mem_space_name, mr_name, MCD_MEM_SPACE_NAME_LEN -= 1); + + g_array_append_val(core_state->memory_spaces, physical); + + mcd_memspace_st logical =3D { + .mem_space_id =3D core_state->memory_spaces->len + 1, + .mem_type =3D MCD_MEM_SPACE_IS_LOGICAL | secure_flag, + .endian =3D endian, + }; + strncpy(logical.mem_space_name, as_name, MCD_MEM_SPACE_NAME_LEN - = 1); + + g_array_append_val(core_state->memory_spaces, logical); + } + + mcd_memspace_st gdb_registers =3D { + .mem_space_id =3D core_state->memory_spaces->len + 1, + .mem_space_name =3D "GDB Registers", + .mem_type =3D MCD_MEM_SPACE_IS_REGISTERS, + .endian =3D endian, + }; + g_array_append_val(core_state->memory_spaces, gdb_registers); + + return MCD_RET_ACT_NONE; +} + mcd_return_et mcd_open_core_f(const mcd_core_con_info_st *core_con_info, mcd_core_st **core) { @@ -490,25 +557,29 @@ mcd_return_et mcd_open_core_f(const mcd_core_con_info= _st *core_con_info, =20 core_id =3D core_con_info->core_id; if (core_id > g_server_state.cores->len) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_PARAM, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "specified core index exceeds the number of cor= es", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 core_state =3D &g_array_index(g_server_state.cores, mcdcore_state, cor= e_id); if (core_state->open_core) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_CONNECTION, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "core already open", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; + return g_server_state.last_error->return_status; + } + + if (query_memspaces(core_state) !=3D MCD_RET_ACT_NONE) { return g_server_state.last_error->return_status; } =20 @@ -540,19 +611,21 @@ mcd_return_et mcd_close_core_f(const mcd_core_st *cor= e) } =20 if (core_state->open_core !=3D core) { - custom_mcd_error =3D (mcd_error_info_st) { + g_server_state.custom_error =3D (mcd_error_info_st) { .return_status =3D MCD_RET_ACT_HANDLE_ERROR, .error_code =3D MCD_ERR_CONNECTION, .error_events =3D MCD_ERR_EVT_NONE, .error_str =3D "core not open", }; - g_server_state.last_error =3D &custom_mcd_error; + g_server_state.last_error =3D &g_server_state.custom_error; return g_server_state.last_error->return_status; } =20 g_free((void *)core->core_con_info); g_free((void *)core); core_state->open_core =3D NULL; + core_state->cpu =3D NULL; + g_array_set_size(core_state->memory_spaces, 0); =20 g_server_state.last_error =3D &MCD_ERROR_NONE; return g_server_state.last_error->return_status; @@ -613,8 +686,66 @@ mcd_return_et mcd_qry_mem_spaces_f(const mcd_core_st *= core, uint32_t *num_mem_spaces, mcd_memspace_st *mem_spaces) { - g_server_state.last_error =3D &MCD_ERROR_NOT_IMPLEMENTED; - return g_server_state.last_error->return_status; + uint32_t i; + mcdcore_state *core_state; + + if (!core || !num_mem_spaces) { + g_server_state.last_error =3D &MCD_ERROR_INVALID_NULL_PARAM; + return g_server_state.last_error->return_status; + } + + core_state =3D find_core(core->core_con_info); + if (!core_state || core_state->open_core !=3D core) { + g_server_state.last_error =3D &MCD_ERROR_UNKNOWN_CORE; + return g_server_state.last_error->return_status; + } + + g_assert(core_state->memory_spaces); + + if (core_state->memory_spaces->len =3D=3D 0) { + core_state->custom_error =3D (mcd_error_info_st) { + .return_status =3D MCD_RET_ACT_HANDLE_ERROR, + .error_code =3D MCD_ERR_NO_MEM_SPACES, + .error_events =3D MCD_ERR_EVT_NONE, + .error_str =3D "", + }; + core_state->last_error =3D &core_state->custom_error; + return core_state->last_error->return_status; + } + + if (*num_mem_spaces =3D=3D 0) { + *num_mem_spaces =3D core_state->memory_spaces->len; + core_state->last_error =3D &MCD_ERROR_NONE; + return core_state->last_error->return_status; + } + + if (start_index >=3D core_state->memory_spaces->len) { + core_state->custom_error =3D (mcd_error_info_st) { + .return_status =3D MCD_RET_ACT_HANDLE_ERROR, + .error_code =3D MCD_ERR_PARAM, + .error_events =3D MCD_ERR_EVT_NONE, + .error_str =3D "start_index exceeds the number of memory space= s", + }; + core_state->last_error =3D &core_state->custom_error; + return core_state->last_error->return_status; + } + + if (!mem_spaces) { + core_state->last_error =3D &MCD_ERROR_INVALID_NULL_PARAM; + return core_state->last_error->return_status; + } + + for (i =3D 0; i < *num_mem_spaces && + start_index + i < core_state->memory_spaces->len; i++) { + + mem_spaces[i] =3D g_array_index(core_state->memory_spaces, + mcd_memspace_st, start_index + i); + } + + *num_mem_spaces =3D i; + + core_state->last_error =3D &MCD_ERROR_NONE; + return core_state->last_error->return_status; } =20 mcd_return_et mcd_qry_mem_blocks_f(const mcd_core_st *core, diff --git a/mcd/mcdstub_qapi.c b/mcd/mcdstub_qapi.c index 51292d239d..437d2c2e3e 100644 --- a/mcd/mcdstub_qapi.c +++ b/mcd/mcdstub_qapi.c @@ -368,3 +368,47 @@ MCDErrorInfo *qmp_mcd_qry_error_info(uint32_t core_uid= , Error **errp) result =3D marshal_mcd_error_info(&error_info); return result; } + +MCDQryMemSpacesResult *qmp_mcd_qry_mem_spaces(uint32_t core_uid, + uint32_t start_index, + uint32_t num_mem_spaces, + Error **errp) +{ + MCDMemspaceList **tailp; + MCDMemspace *ms; + mcd_memspace_st *memspaces =3D NULL; + bool query_num_only =3D num_mem_spaces =3D=3D 0; + MCDQryMemSpacesResult *result =3D g_malloc0(sizeof(*result)); + mcd_core_st *core =3D NULL; + + if (retrieve_open_core(core_uid, &core) !=3D MCD_RET_ACT_NONE) { + g_stub_state.on_error_ask_server =3D false; + } + + if (!query_num_only) { + memspaces =3D g_malloc0(num_mem_spaces * sizeof(*memspaces)); + } + + result->return_status =3D mcd_qry_mem_spaces_f(core, start_index, + &num_mem_spaces, memspace= s); + + if (result->return_status =3D=3D MCD_RET_ACT_NONE) { + result->has_num_mem_spaces =3D true; + result->num_mem_spaces =3D num_mem_spaces; + if (!query_num_only) { + result->has_mem_spaces =3D true; + tailp =3D &(result->mem_spaces); + for (uint32_t i =3D 0; i < num_mem_spaces; i++) { + ms =3D marshal_mcd_memspace(memspaces + i); + QAPI_LIST_APPEND(tailp, ms); + } + } + } + + if (!query_num_only) { + g_free(memspaces); + } + + g_stub_state.on_error_ask_server =3D true; + return result; +} diff --git a/qapi/mcd.json b/qapi/mcd.json index 7219056464..214933e279 100644 --- a/qapi/mcd.json +++ b/qapi/mcd.json @@ -147,6 +147,74 @@ 'core-id' : 'uint32' } } =20 =20 +## +# @MCDMemspace: +# +# Structure type containing information about a memory space. +# +# @mem-space-id: ID of this memory space, ID 0 is reserved. +# @mem-space-name: Unique name of the memory space. +# @mem-type: Type of the memory space. +# @bits-per-mau: Bits per minimum addressable unit (MAU). The +# minimum addressable unit of a memory is defin= ed as +# the size in bits of its basic block that may = have +# a unique address. For example for a byte +# addressable memory this value would be set to= '8' +# according to the 8 bits of a byte block. +# @invariance: The total number of bytes in a memory word, w= hich +# is @bits-per-mau divided by 8, consists of gr= oups +# of "invariant" bytes. These groups can be arr= anged +# in Big Endian or Little Endian order. +# For example an @invariance of '2' and '64' +# @bits-per-mau, a Little Endian word are +# represented as b0 b1 b2 b3 b4 b5 b6 b7. +# In contrast to this, a Big Endian word is +# represented as b6 b7 b4 b5 b2 b3 b0 b1. +# @endian: Endianness of this memory space. Can be overr= iden +# by @endian of a MCDMemblock. +# @min-addr: Minimum address of this memory space. +# @max-addr: Maximum address of this memory space. +# @num-mem-blocks: Number of memory blocks in this memory space.= Each +# memory space may have a certain number of mem= ory +# blocks. Memory blocks contain additional +# information pertaining to the intended purpos= e of +# the memory. This information may be used as a= hint +# for memory data representation within a tool's +# memory view. This field specifies the number = of +# memory blocks present in this memory space. +# @supported-access-options: Supported memory access options (OR'ed bitmas= k). +# Can be overriden by @supported-access-options= of a +# MCDMemblock. +# @core-mode-mask-read: Mask of core modes for which read accesses are +# impossible. A set bit indicates that read acc= esses +# are denied in this mode. Bit 0 represents core +# mode '1', bit 31 represents core mode '32'. C= an be +# overriden by @core-mode-mask-read of a MCDMem= block. +# @core-mode-mask-write: Mask of core modes for which write accesses a= re +# impossible; a set bit indicates that write +# accesses are denied in this mode. Bit 0 repre= sents +# core mode '1', bit 31 represents core mode '3= 2'. +# Can be overriden by +# @core-mode-mask-write of a MCDMemblock. +# +# Since: 9.1 +## +{ 'struct': 'MCDMemspace', + 'data': { + 'mem-space-id' : 'uint32', + 'mem-space-name' : 'str', + 'mem-type' : 'uint32', + 'bits-per-mau' : 'uint32', + 'invariance' : 'uint8', + 'endian' : 'uint32', + 'min-addr' : 'uint64', + 'max-addr' : 'uint64', + 'num-mem-blocks' : 'uint32', + 'supported-access-options': 'uint32', + 'core-mode-mask-read' : 'uint32', + 'core-mode-mask-write' : 'uint32' } } + + ## # =3D=3D Target Initialization API ## @@ -878,3 +946,134 @@ { 'command': 'mcd-qry-error-info', 'data': { 'core-uid': 'uint32' }, 'returns': 'MCDErrorInfo' } + + +## +# @MCDQryMemSpacesResult: +# +# Return value of @mcd-qry-mem-spaces. +# +# @return-status: Return code. +# @num-mem-spaces: The number of returned memory spaces. In case the input= value +# of @num-mem-spaces is '0', this is the number of all +# available memory spaces for the selected core. +# @mem-spaces: Memory space information. +# +# Since: 9.1 +## +{ 'struct': 'MCDQryMemSpacesResult', + 'data': { + 'return-status' : 'uint32', + '*num-mem-spaces': 'uint32', + '*mem-spaces' : [ 'MCDMemspace' ] }} + +## +# @mcd-qry-mem-spaces: +# +# Function querying the available memory spaces for a particular component. +# +# @core-uid: Unique identifier of the open core as returned by +# @mcd-open-core. +# @start-index: Start index of the requested memory spaces. This refers= to +# an internal list of the target side implementation. +# @num-mem-spaces: Number of memory spaces, information is requested of. I= f it +# is set to '0', no memory space information is returned = but +# the number of all available memory spaces for the selec= ted +# core. +# +# Returns: @MCDQryMemSpacesResult +# +# Since: 9.1 +# +# .. qmp-example:: +# :title: Arm TrustZone +# +# -> { "execute": "mcd-qry-mem-spaces", +# "arguments": { "core-uid": 1, +# "start-index": 0, +# "num-mem-spaces": 20 } } +# <- { +# "return": { +# "mem-spaces": [ +# { +# "mem-space-id": 1, +# "bits-per-mau": 0, +# "mem-space-name": "system", +# "endian": 0, +# "max-addr": 0, +# "mem-type": 16, +# "core-mode-mask-write": 0, +# "core-mode-mask-read": 0, +# "supported-access-options": 0, +# "invariance": 0, +# "num-mem-blocks": 0, +# "min-addr": 0 +# }, +# { +# "mem-space-id": 2, +# "bits-per-mau": 0, +# "mem-space-name": "cpu-memory-0", +# "endian": 0, +# "max-addr": 0, +# "mem-type": 32, +# "core-mode-mask-write": 0, +# "core-mode-mask-read": 0, +# "supported-access-options": 0, +# "invariance": 0, +# "num-mem-blocks": 0, +# "min-addr": 0 +# }, +# { +# "mem-space-id": 3, +# "bits-per-mau": 0, +# "mem-space-name": "secure-memory", +# "endian": 0, +# "max-addr": 0, +# "mem-type": 65552, +# "core-mode-mask-write": 0, +# "core-mode-mask-read": 0, +# "supported-access-options": 0, +# "invariance": 0, +# "num-mem-blocks": 0, +# "min-addr": 0 +# }, +# { +# "mem-space-id": 4, +# "bits-per-mau": 0, +# "mem-space-name": "cpu-secure-memory-0", +# "endian": 0, +# "max-addr": 0, +# "mem-type": 65568, +# "core-mode-mask-write": 0, +# "core-mode-mask-read": 0, +# "supported-access-options": 0, +# "invariance": 0, +# "num-mem-blocks": 0, +# "min-addr": 0 +# }, +# { +# "mem-space-id": 5, +# "bits-per-mau": 0, +# "mem-space-name": "GDB Registers", +# "endian": 0, +# "max-addr": 0, +# "mem-type": 1, +# "core-mode-mask-write": 0, +# "core-mode-mask-read": 0, +# "supported-access-options": 0, +# "invariance": 0, +# "num-mem-blocks": 0, +# "min-addr": 0 +# } +# ], +# "return-status": 0, +# "num-mem-spaces": 5 +# } +# } +## +{ 'command': 'mcd-qry-mem-spaces', + 'data': { + 'core-uid' : 'uint32', + 'start-index' : 'uint32', + 'num-mem-spaces': 'uint32' }, + 'returns': 'MCDQryMemSpacesResult' } diff --git a/tests/qtest/libmcd-test.c b/tests/qtest/libmcd-test.c index c976eb1bed..9e41a0218f 100644 --- a/tests/qtest/libmcd-test.c +++ b/tests/qtest/libmcd-test.c @@ -258,3 +258,23 @@ MCDCloseCoreResult *qtest_mcd_close_core(QTestState *q= ts, =20 return unmarshal; } + +MCDQryMemSpacesResult *qtest_mcd_qry_mem_spaces( + QTestState *qts, q_obj_mcd_qry_mem_spaces_arg *args) +{ + Visitor *v; + QObject *marshal; + QDict *arg, *resp; + QObject *ret; + bool ok; + MCDQryMemSpacesResult *unmarshal; + + MARSHAL_ARGS(q_obj_mcd_qry_mem_spaces_arg); + + resp =3D qtest_qmp(qts, "{'execute': 'mcd-qry-mem-spaces'," + "'arguments': %p}", arg); + + UNMARSHAL_RESULT(MCDQryMemSpacesResult); + + return unmarshal; +} diff --git a/tests/qtest/libmcd-test.h b/tests/qtest/libmcd-test.h index 323458785e..26f35b8931 100644 --- a/tests/qtest/libmcd-test.h +++ b/tests/qtest/libmcd-test.h @@ -47,4 +47,7 @@ MCDOpenCoreResult *qtest_mcd_open_core(QTestState *qts, MCDCloseCoreResult *qtest_mcd_close_core(QTestState *qts, q_obj_mcd_close_core_arg *args); =20 +MCDQryMemSpacesResult *qtest_mcd_qry_mem_spaces(QTestState *qts, + q_obj_mcd_qry_mem_spaces_arg *args); + #endif /* LIBMCD_TEST_H */ diff --git a/tests/qtest/mcd-test.c b/tests/qtest/mcd-test.c index 8a7e04c5cf..7deca39f93 100644 --- a/tests/qtest/mcd-test.c +++ b/tests/qtest/mcd-test.c @@ -347,6 +347,84 @@ static void test_open_core(void) qtest_quit(qts); } =20 +static void test_qry_core_info(void) +{ + QTestState *qts =3D qtest_init(QEMU_EXTRA_ARGS); + MCDQryCoresResult *cores_query =3D open_server_query_cores(qts); + + MCDCoreConInfoList *core_head =3D cores_query->core_con_info; + for (uint32_t c =3D 0; c < cores_query->num_cores; c++) { + q_obj_mcd_qry_mem_spaces_arg qry_mem_spaces_args; + q_obj_mcd_close_core_arg close_core_args; + MCDQryMemSpacesResult *qry_mem_spaces_result; + MCDCloseCoreResult *close_core_result; + + MCDCoreConInfo *core_con_info =3D core_head->value; + q_obj_mcd_open_core_arg open_core_args =3D { + .core_con_info =3D core_con_info, + }; + MCDOpenCoreResult *open_core_result =3D + qtest_mcd_open_core(qts, &open_core_args); + g_assert(open_core_result->return_status =3D=3D MCD_RET_ACT_NONE); + g_assert(open_core_result->has_core_uid); + + if (verbose) { + fprintf(stderr, "[INFO]\tCore %s #%d\n", + core_con_info->core, + core_con_info->core_id); + } + + qry_mem_spaces_args =3D (q_obj_mcd_qry_mem_spaces_arg) { + .core_uid =3D open_core_result->core_uid, + .start_index =3D 0, + .num_mem_spaces =3D 0, + }; + + qry_mem_spaces_result =3D qtest_mcd_qry_mem_spaces(qts, + &qry_mem_spaces_a= rgs); + g_assert(qry_mem_spaces_result->return_status =3D=3D MCD_RET_ACT_N= ONE); + g_assert(qry_mem_spaces_result->has_num_mem_spaces); + g_assert(qry_mem_spaces_result->num_mem_spaces > 0); + + qry_mem_spaces_args.num_mem_spaces =3D + qry_mem_spaces_result->num_mem_spaces; + qapi_free_MCDQryMemSpacesResult(qry_mem_spaces_result); + qry_mem_spaces_result =3D qtest_mcd_qry_mem_spaces(qts, + &qry_mem_spaces_a= rgs); + g_assert(qry_mem_spaces_result->return_status =3D=3D MCD_RET_ACT_N= ONE); + g_assert(qry_mem_spaces_result->has_num_mem_spaces); + + if (verbose) { + MCDMemspaceList *ms_head =3D qry_mem_spaces_result->mem_spaces; + for (uint32_t i =3D 0; + i < qry_mem_spaces_result->num_mem_spaces; i++) { + MCDMemspace *ms =3D ms_head->value; + if (verbose) { + fprintf(stderr, "\tMemory Space: %s (#%d)\n" + "\t Type: 0x%x\n", + ms->mem_space_name, + ms->mem_space_id, + ms->mem_type); + } + ms_head =3D ms_head->next; + } + } + + qapi_free_MCDQryMemSpacesResult(qry_mem_spaces_result); + close_core_args.core_uid =3D open_core_result->core_uid; + close_core_result =3D qtest_mcd_close_core(qts, &close_core_args); + g_assert(close_core_result->return_status =3D=3D MCD_RET_ACT_NONE); + + qapi_free_MCDCloseCoreResult(close_core_result); + qapi_free_MCDOpenCoreResult(open_core_result); + core_head =3D core_head->next; + } + + qapi_free_MCDQryCoresResult(cores_query); + qtest_mcd_exit(qts); + qtest_quit(qts); +} + int main(int argc, char *argv[]) { char *v_env =3D getenv("V"); @@ -358,5 +436,6 @@ int main(int argc, char *argv[]) qtest_add_func("mcd/open-server", test_open_server); qtest_add_func("mcd/qry-cores", test_qry_cores); qtest_add_func("mcd/open-core", test_open_core); + qtest_add_func("mcd/qry-core-info", test_qry_core_info); return g_test_run(); } --=20 2.34.1