mirror of
https://github.com/qemu/qemu.git
synced 2026-04-05 21:46:25 +00:00
tcg: Pass host-endian values to plugin_gen_mem_callbacks_*
If the host does not support swapped-endian loads and stores, then we emulate those within the tcg expanders with explicit bswap operations. However, we were passing values to the plugin interface in the middle of those bswap operations, which meant that we would pass values of the wrong endianness to plugins when running on hosts without swapped-endian loads and stores. Resolves: https://gitlab.com/qemu-project/qemu/-/work_items/3351 Signed-off-by: Richard Henderson <richard.henderson@linaro.org> Reviewed-by: Pierrick Bouvier <pierrick.bouvier@linaro.org> Tested-by: Pierrick Bouvier <pierrick.bouvier@linaro.org> Link: https://lore.kernel.org/qemu-devel/20260325024252.3369186-2-pierrick.bouvier@linaro.org Signed-off-by: Pierrick Bouvier <pierrick.bouvier@linaro.org>
This commit is contained in:
committed by
Pierrick Bouvier
parent
007b29752e
commit
539421a428
@@ -262,9 +262,6 @@ static void tcg_gen_qemu_ld_i32_int(TCGv_i32 val, TCGTemp *addr,
|
|||||||
addr_new = tci_extend_addr(addr);
|
addr_new = tci_extend_addr(addr);
|
||||||
copy_addr = plugin_maybe_preserve_addr(addr);
|
copy_addr = plugin_maybe_preserve_addr(addr);
|
||||||
gen_ldst1(INDEX_op_qemu_ld, TCG_TYPE_I32, tcgv_i32_temp(val), addr_new, oi);
|
gen_ldst1(INDEX_op_qemu_ld, TCG_TYPE_I32, tcgv_i32_temp(val), addr_new, oi);
|
||||||
plugin_gen_mem_callbacks_i32(val, copy_addr, addr, orig_oi,
|
|
||||||
QEMU_PLUGIN_MEM_R);
|
|
||||||
maybe_free_addr(addr, addr_new);
|
|
||||||
|
|
||||||
if ((orig_memop ^ memop) & MO_BSWAP) {
|
if ((orig_memop ^ memop) & MO_BSWAP) {
|
||||||
switch (orig_memop & MO_SIZE) {
|
switch (orig_memop & MO_SIZE) {
|
||||||
@@ -280,6 +277,10 @@ static void tcg_gen_qemu_ld_i32_int(TCGv_i32 val, TCGTemp *addr,
|
|||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
plugin_gen_mem_callbacks_i32(val, copy_addr, addr, orig_oi,
|
||||||
|
QEMU_PLUGIN_MEM_R);
|
||||||
|
maybe_free_addr(addr, addr_new);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_gen_qemu_ld_i32_chk(TCGv_i32 val, TCGTemp *addr, TCGArg idx,
|
void tcg_gen_qemu_ld_i32_chk(TCGv_i32 val, TCGTemp *addr, TCGArg idx,
|
||||||
@@ -290,10 +291,10 @@ void tcg_gen_qemu_ld_i32_chk(TCGv_i32 val, TCGTemp *addr, TCGArg idx,
|
|||||||
tcg_gen_qemu_ld_i32_int(val, addr, idx, memop);
|
tcg_gen_qemu_ld_i32_int(val, addr, idx, memop);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_gen_qemu_st_i32_int(TCGv_i32 val, TCGTemp *addr,
|
static void tcg_gen_qemu_st_i32_int(TCGv_i32 orig_val, TCGTemp *addr,
|
||||||
TCGArg idx, MemOp memop)
|
TCGArg idx, MemOp memop)
|
||||||
{
|
{
|
||||||
TCGv_i32 swap = NULL;
|
TCGv_i32 val = orig_val;
|
||||||
MemOpIdx orig_oi, oi;
|
MemOpIdx orig_oi, oi;
|
||||||
TCGTemp *addr_new;
|
TCGTemp *addr_new;
|
||||||
|
|
||||||
@@ -302,29 +303,29 @@ static void tcg_gen_qemu_st_i32_int(TCGv_i32 val, TCGTemp *addr,
|
|||||||
orig_oi = oi = make_memop_idx(memop, idx);
|
orig_oi = oi = make_memop_idx(memop, idx);
|
||||||
|
|
||||||
if ((memop & MO_BSWAP) && !tcg_target_has_memory_bswap(memop)) {
|
if ((memop & MO_BSWAP) && !tcg_target_has_memory_bswap(memop)) {
|
||||||
swap = tcg_temp_ebb_new_i32();
|
val = tcg_temp_ebb_new_i32();
|
||||||
switch (memop & MO_SIZE) {
|
switch (memop & MO_SIZE) {
|
||||||
case MO_16:
|
case MO_16:
|
||||||
tcg_gen_bswap16_i32(swap, val, 0);
|
tcg_gen_bswap16_i32(val, orig_val, 0);
|
||||||
break;
|
break;
|
||||||
case MO_32:
|
case MO_32:
|
||||||
tcg_gen_bswap32_i32(swap, val);
|
tcg_gen_bswap32_i32(val, orig_val);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
val = swap;
|
|
||||||
memop &= ~MO_BSWAP;
|
memop &= ~MO_BSWAP;
|
||||||
oi = make_memop_idx(memop, idx);
|
oi = make_memop_idx(memop, idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
addr_new = tci_extend_addr(addr);
|
addr_new = tci_extend_addr(addr);
|
||||||
gen_ldst1(INDEX_op_qemu_st, TCG_TYPE_I32, tcgv_i32_temp(val), addr_new, oi);
|
gen_ldst1(INDEX_op_qemu_st, TCG_TYPE_I32, tcgv_i32_temp(val), addr_new, oi);
|
||||||
plugin_gen_mem_callbacks_i32(val, NULL, addr, orig_oi, QEMU_PLUGIN_MEM_W);
|
plugin_gen_mem_callbacks_i32(orig_val, NULL, addr, orig_oi,
|
||||||
|
QEMU_PLUGIN_MEM_W);
|
||||||
maybe_free_addr(addr, addr_new);
|
maybe_free_addr(addr, addr_new);
|
||||||
|
|
||||||
if (swap) {
|
if (val != orig_val) {
|
||||||
tcg_temp_free_i32(swap);
|
tcg_temp_free_i32(val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -360,9 +361,6 @@ static void tcg_gen_qemu_ld_i64_int(TCGv_i64 val, TCGTemp *addr,
|
|||||||
addr_new = tci_extend_addr(addr);
|
addr_new = tci_extend_addr(addr);
|
||||||
copy_addr = plugin_maybe_preserve_addr(addr);
|
copy_addr = plugin_maybe_preserve_addr(addr);
|
||||||
gen_ld_i64(val, addr_new, oi);
|
gen_ld_i64(val, addr_new, oi);
|
||||||
plugin_gen_mem_callbacks_i64(val, copy_addr, addr, orig_oi,
|
|
||||||
QEMU_PLUGIN_MEM_R);
|
|
||||||
maybe_free_addr(addr, addr_new);
|
|
||||||
|
|
||||||
if ((orig_memop ^ memop) & MO_BSWAP) {
|
if ((orig_memop ^ memop) & MO_BSWAP) {
|
||||||
int flags = (orig_memop & MO_SIGN
|
int flags = (orig_memop & MO_SIGN
|
||||||
@@ -382,6 +380,10 @@ static void tcg_gen_qemu_ld_i64_int(TCGv_i64 val, TCGTemp *addr,
|
|||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
plugin_gen_mem_callbacks_i64(val, copy_addr, addr, orig_oi,
|
||||||
|
QEMU_PLUGIN_MEM_R);
|
||||||
|
maybe_free_addr(addr, addr_new);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tcg_gen_qemu_ld_i64_chk(TCGv_i64 val, TCGTemp *addr, TCGArg idx,
|
void tcg_gen_qemu_ld_i64_chk(TCGv_i64 val, TCGTemp *addr, TCGArg idx,
|
||||||
@@ -392,10 +394,10 @@ void tcg_gen_qemu_ld_i64_chk(TCGv_i64 val, TCGTemp *addr, TCGArg idx,
|
|||||||
tcg_gen_qemu_ld_i64_int(val, addr, idx, memop);
|
tcg_gen_qemu_ld_i64_int(val, addr, idx, memop);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tcg_gen_qemu_st_i64_int(TCGv_i64 val, TCGTemp *addr,
|
static void tcg_gen_qemu_st_i64_int(TCGv_i64 orig_val, TCGTemp *addr,
|
||||||
TCGArg idx, MemOp memop)
|
TCGArg idx, MemOp memop)
|
||||||
{
|
{
|
||||||
TCGv_i64 swap = NULL;
|
TCGv_i64 val = orig_val;
|
||||||
MemOpIdx orig_oi, oi;
|
MemOpIdx orig_oi, oi;
|
||||||
TCGTemp *addr_new;
|
TCGTemp *addr_new;
|
||||||
|
|
||||||
@@ -404,32 +406,32 @@ static void tcg_gen_qemu_st_i64_int(TCGv_i64 val, TCGTemp *addr,
|
|||||||
orig_oi = oi = make_memop_idx(memop, idx);
|
orig_oi = oi = make_memop_idx(memop, idx);
|
||||||
|
|
||||||
if ((memop & MO_BSWAP) && !tcg_target_has_memory_bswap(memop)) {
|
if ((memop & MO_BSWAP) && !tcg_target_has_memory_bswap(memop)) {
|
||||||
swap = tcg_temp_ebb_new_i64();
|
val = tcg_temp_ebb_new_i64();
|
||||||
switch (memop & MO_SIZE) {
|
switch (memop & MO_SIZE) {
|
||||||
case MO_16:
|
case MO_16:
|
||||||
tcg_gen_bswap16_i64(swap, val, 0);
|
tcg_gen_bswap16_i64(val, orig_val, 0);
|
||||||
break;
|
break;
|
||||||
case MO_32:
|
case MO_32:
|
||||||
tcg_gen_bswap32_i64(swap, val, 0);
|
tcg_gen_bswap32_i64(val, orig_val, 0);
|
||||||
break;
|
break;
|
||||||
case MO_64:
|
case MO_64:
|
||||||
tcg_gen_bswap64_i64(swap, val);
|
tcg_gen_bswap64_i64(val, orig_val);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
}
|
}
|
||||||
val = swap;
|
|
||||||
memop &= ~MO_BSWAP;
|
memop &= ~MO_BSWAP;
|
||||||
oi = make_memop_idx(memop, idx);
|
oi = make_memop_idx(memop, idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
addr_new = tci_extend_addr(addr);
|
addr_new = tci_extend_addr(addr);
|
||||||
gen_st_i64(val, addr_new, oi);
|
gen_st_i64(val, addr_new, oi);
|
||||||
plugin_gen_mem_callbacks_i64(val, NULL, addr, orig_oi, QEMU_PLUGIN_MEM_W);
|
plugin_gen_mem_callbacks_i64(orig_val, NULL, addr, orig_oi,
|
||||||
|
QEMU_PLUGIN_MEM_W);
|
||||||
maybe_free_addr(addr, addr_new);
|
maybe_free_addr(addr, addr_new);
|
||||||
|
|
||||||
if (swap) {
|
if (val != orig_val) {
|
||||||
tcg_temp_free_i64(swap);
|
tcg_temp_free_i64(val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user