1
Most of this is the Neon decodetree patches, followed by Edgar's versal cleanups.
1
Hi; here's a target-arm pullreq. Mostly this is some decodetree
2
conversion patches from me, plus a scattering of other bug fixes.
2
3
3
thanks
4
thanks
4
-- PMM
5
-- PMM
5
6
7
The following changes since commit e3660cc1e3cb136af50c0eaaeac27943c2438d1d:
6
8
7
The following changes since commit 2ef486e76d64436be90f7359a3071fb2a56ce835:
9
Merge tag 'pull-loongarch-20230616' of https://gitlab.com/gaosong/qemu into staging (2023-06-16 12:30:16 +0200)
8
9
Merge remote-tracking branch 'remotes/marcel/tags/rdma-pull-request' into staging (2020-05-03 14:12:56 +0100)
10
10
11
are available in the Git repository at:
11
are available in the Git repository at:
12
12
13
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20200504
13
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20230619
14
14
15
for you to fetch changes up to 9aefc6cf9b73f66062d2f914a0136756e7a28211:
15
for you to fetch changes up to 074259c0f2ac40042dce766d870318cc22f388eb:
16
16
17
target/arm: Move gen_ function typedefs to translate.h (2020-05-04 12:59:26 +0100)
17
hw/misc/bcm2835_property: Handle CORE_CLK_ID firmware property (2023-06-19 15:27:21 +0100)
18
18
19
----------------------------------------------------------------
19
----------------------------------------------------------------
20
target-arm queue:
20
target-arm queue:
21
* Start of conversion of Neon insns to decodetree
21
* Fix return value from LDSMIN/LDSMAX 8/16 bit atomics
22
* versal board: support SD and RTC
22
* Return correct result for LDG when ATA=0
23
* Implement ARMv8.2-TTS2UXN
23
* Conversion of system insns, loads and stores to decodetree
24
* Make VQDMULL undefined when U=1
24
* hw/intc/allwinner-a10-pic: Handle IRQ levels other than 0 or 1
25
* Some minor code cleanups
25
* hw/sd/allwinner-sdhost: Don't send non-boolean IRQ line levels
26
* hw/timer/nrf51_timer: Don't lose time when timer is queried in tight loop
27
* hw/arm/Kconfig: sbsa-ref uses Bochs display
28
* imx_serial: set wake bit when we receive a data byte
29
* docs: sbsa: document board to firmware interface
30
* hw/misc/bcm2835_property: avoid hard-coded constants
26
31
27
----------------------------------------------------------------
32
----------------------------------------------------------------
28
Edgar E. Iglesias (11):
33
Marcin Juszkiewicz (2):
29
hw/arm: versal: Remove inclusion of arm_gicv3_common.h
34
hw/arm/Kconfig: sbsa-ref uses Bochs display
30
hw/arm: versal: Move misplaced comment
35
docs: sbsa: document board to firmware interface
31
hw/arm: versal-virt: Fix typo xlnx-ve -> xlnx-versal
32
hw/arm: versal: Embed the UARTs into the SoC type
33
hw/arm: versal: Embed the GEMs into the SoC type
34
hw/arm: versal: Embed the ADMAs into the SoC type
35
hw/arm: versal: Embed the APUs into the SoC type
36
hw/arm: versal: Add support for SD
37
hw/arm: versal: Add support for the RTC
38
hw/arm: versal-virt: Add support for SD
39
hw/arm: versal-virt: Add support for the RTC
40
36
41
Fredrik Strupe (1):
37
Martin Kaiser (1):
42
target/arm: Make VQDMULL undefined when U=1
38
imx_serial: set wake bit when we receive a data byte
43
39
44
Peter Maydell (25):
40
Peter Maydell (26):
45
target/arm: Don't use a TLB for ARMMMUIdx_Stage2
41
target/arm: Fix return value from LDSMIN/LDSMAX 8/16 bit atomics
46
target/arm: Use enum constant in get_phys_addr_lpae() call
42
target/arm: Return correct result for LDG when ATA=0
47
target/arm: Add new 's1_is_el0' argument to get_phys_addr_lpae()
43
target/arm: Pass memop to gen_mte_check1_mmuidx() in reg_imm9 decode
48
target/arm: Implement ARMv8.2-TTS2UXN
44
target/arm: Consistently use finalize_memop_asimd() for ASIMD loads/stores
49
target/arm: Use correct variable for setting 'max' cpu's ID_AA64DFR0
45
target/arm: Convert hint instruction space to decodetree
50
target/arm/translate-vfp.inc.c: Remove duplicate simd_r32 check
46
target/arm: Convert barrier insns to decodetree
51
target/arm: Don't allow Thumb Neon insns without FEATURE_NEON
47
target/arm: Convert CFINV, XAFLAG and AXFLAG to decodetree
52
target/arm: Add stubs for AArch32 Neon decodetree
48
target/arm: Convert MSR (immediate) to decodetree
53
target/arm: Convert VCMLA (vector) to decodetree
49
target/arm: Convert MSR (reg), MRS, SYS, SYSL to decodetree
54
target/arm: Convert VCADD (vector) to decodetree
50
target/arm: Convert exception generation instructions to decodetree
55
target/arm: Convert V[US]DOT (vector) to decodetree
51
target/arm: Convert load/store exclusive and ordered to decodetree
56
target/arm: Convert VFM[AS]L (vector) to decodetree
52
target/arm: Convert LDXP, STXP, CASP, CAS to decodetree
57
target/arm: Convert VCMLA (scalar) to decodetree
53
target/arm: Convert load reg (literal) group to decodetree
58
target/arm: Convert V[US]DOT (scalar) to decodetree
54
target/arm: Convert load/store-pair to decodetree
59
target/arm: Convert VFM[AS]L (scalar) to decodetree
55
target/arm: Convert ld/st reg+imm9 insns to decodetree
60
target/arm: Convert Neon load/store multiple structures to decodetree
56
target/arm: Convert LDR/STR with 12-bit immediate to decodetree
61
target/arm: Convert Neon 'load single structure to all lanes' to decodetree
57
target/arm: Convert LDR/STR reg+reg to decodetree
62
target/arm: Convert Neon 'load/store single structure' to decodetree
58
target/arm: Convert atomic memory ops to decodetree
63
target/arm: Convert Neon 3-reg-same VADD/VSUB to decodetree
59
target/arm: Convert load (pointer auth) insns to decodetree
64
target/arm: Convert Neon 3-reg-same logic ops to decodetree
60
target/arm: Convert LDAPR/STLR (imm) to decodetree
65
target/arm: Convert Neon 3-reg-same VMAX/VMIN to decodetree
61
target/arm: Convert load/store (multiple structures) to decodetree
66
target/arm: Convert Neon 3-reg-same comparisons to decodetree
62
target/arm: Convert load/store single structure to decodetree
67
target/arm: Convert Neon 3-reg-same VQADD/VQSUB to decodetree
63
target/arm: Convert load/store tags insns to decodetree
68
target/arm: Convert Neon 3-reg-same VMUL, VMLA, VMLS, VSHL to decodetree
64
hw/intc/allwinner-a10-pic: Handle IRQ levels other than 0 or 1
69
target/arm: Move gen_ function typedefs to translate.h
65
hw/sd/allwinner-sdhost: Don't send non-boolean IRQ line levels
66
hw/timer/nrf51_timer: Don't lose time when timer is queried in tight loop
70
67
71
Philippe Mathieu-Daudé (2):
68
Sergey Kambalin (4):
72
hw/arm/mps2-tz: Use TYPE_IOTKIT instead of hardcoded string
69
hw/arm/raspi: Import Linux raspi definitions as 'raspberrypi-fw-defs.h'
73
target/arm: Use uint64_t for midr field in CPU state struct
70
hw/misc/bcm2835_property: Use 'raspberrypi-fw-defs.h' definitions
71
hw/misc/bcm2835_property: Replace magic frequency values by definitions
72
hw/misc/bcm2835_property: Handle CORE_CLK_ID firmware property
74
73
75
include/hw/arm/xlnx-versal.h | 31 +-
74
docs/system/arm/sbsa.rst | 38 +-
76
target/arm/cpu-param.h | 2 +-
75
include/hw/arm/raspi_platform.h | 10 +
77
target/arm/cpu.h | 38 ++-
76
include/hw/char/imx_serial.h | 1 +
78
target/arm/translate-a64.h | 9 -
77
include/hw/misc/raspberrypi-fw-defs.h | 163 ++
79
target/arm/translate.h | 26 ++
78
target/arm/tcg/a64.decode | 403 ++++
80
target/arm/neon-dp.decode | 86 +++++
79
hw/char/imx_serial.c | 5 +-
81
target/arm/neon-ls.decode | 52 +++
80
hw/intc/allwinner-a10-pic.c | 2 +-
82
target/arm/neon-shared.decode | 66 ++++
81
hw/misc/bcm2835_property.c | 112 +-
83
hw/arm/mps2-tz.c | 2 +-
82
hw/sd/allwinner-sdhost.c | 2 +-
84
hw/arm/xlnx-versal-virt.c | 74 ++++-
83
hw/timer/nrf51_timer.c | 7 +-
85
hw/arm/xlnx-versal.c | 115 +++++--
84
target/arm/tcg/translate-a64.c | 3319 +++++++++++++++------------------
86
target/arm/cpu.c | 3 +-
85
hw/arm/Kconfig | 1 +
87
target/arm/cpu64.c | 8 +-
86
12 files changed, 2157 insertions(+), 1906 deletions(-)
88
target/arm/helper.c | 183 ++++------
87
create mode 100644 include/hw/misc/raspberrypi-fw-defs.h
89
target/arm/translate-a64.c | 17 -
90
target/arm/translate-neon.inc.c | 714 +++++++++++++++++++++++++++++++++++++++
91
target/arm/translate-vfp.inc.c | 6 -
92
target/arm/translate.c | 716 +++-------------------------------------
93
target/arm/Makefile.objs | 18 +
94
19 files changed, 1302 insertions(+), 864 deletions(-)
95
create mode 100644 target/arm/neon-dp.decode
96
create mode 100644 target/arm/neon-ls.decode
97
create mode 100644 target/arm/neon-shared.decode
98
create mode 100644 target/arm/translate-neon.inc.c
99
diff view generated by jsdifflib
Deleted patch
1
From: Fredrik Strupe <fredrik@strupe.net>
2
1
3
According to Arm ARM, VQDMULL is only valid when U=0, while having
4
U=1 is unallocated.
5
6
Signed-off-by: Fredrik Strupe <fredrik@strupe.net>
7
Fixes: 695272dcb976 ("target-arm: Handle UNDEF cases for Neon 3-regs-different-widths")
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
11
target/arm/translate.c | 2 +-
12
1 file changed, 1 insertion(+), 1 deletion(-)
13
14
diff --git a/target/arm/translate.c b/target/arm/translate.c
15
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/translate.c
17
+++ b/target/arm/translate.c
18
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
19
{0, 0, 0, 0}, /* VMLSL */
20
{0, 0, 0, 9}, /* VQDMLSL */
21
{0, 0, 0, 0}, /* Integer VMULL */
22
- {0, 0, 0, 1}, /* VQDMULL */
23
+ {0, 0, 0, 9}, /* VQDMULL */
24
{0, 0, 0, 0xa}, /* Polynomial VMULL */
25
{0, 0, 0, 7}, /* Reserved: always UNDEF */
26
};
27
--
28
2.20.1
29
30
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
The atomic memory operations are supposed to return the old memory
2
data value in the destination register. This value is not
3
sign-extended, even if the operation is the signed minimum or
4
maximum. (In the pseudocode for the instructions the returned data
5
value is passed to ZeroExtend() to create the value in the register.)
2
6
3
Add support for SD.
7
We got this wrong because we were doing a 32-to-64 zero extend on the
8
result for 8 and 16 bit data values, rather than the correct amount
9
of zero extension.
4
10
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
11
Fix the bug by using ext8u and ext16u for the MO_8 and MO_16 data
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
12
sizes rather than ext32u.
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
13
8
Message-id: 20200427181649.26851-11-edgar.iglesias@gmail.com
14
Cc: qemu-stable@nongnu.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
17
Message-id: 20230602155223.2040685-2-peter.maydell@linaro.org
10
---
18
---
11
hw/arm/xlnx-versal-virt.c | 46 +++++++++++++++++++++++++++++++++++++++
19
target/arm/tcg/translate-a64.c | 18 ++++++++++++++++--
12
1 file changed, 46 insertions(+)
20
1 file changed, 16 insertions(+), 2 deletions(-)
13
21
14
diff --git a/hw/arm/xlnx-versal-virt.c b/hw/arm/xlnx-versal-virt.c
22
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
15
index XXXXXXX..XXXXXXX 100644
23
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/arm/xlnx-versal-virt.c
24
--- a/target/arm/tcg/translate-a64.c
17
+++ b/hw/arm/xlnx-versal-virt.c
25
+++ b/target/arm/tcg/translate-a64.c
18
@@ -XXX,XX +XXX,XX @@
26
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_atomic(DisasContext *s, uint32_t insn,
19
#include "hw/arm/sysbus-fdt.h"
27
*/
20
#include "hw/arm/fdt.h"
28
fn(tcg_rt, clean_addr, tcg_rs, get_mem_index(s), mop);
21
#include "cpu.h"
29
22
+#include "hw/qdev-properties.h"
30
- if ((mop & MO_SIGN) && size != MO_64) {
23
#include "hw/arm/xlnx-versal.h"
31
- tcg_gen_ext32u_i64(tcg_rt, tcg_rt);
24
32
+ if (mop & MO_SIGN) {
25
#define TYPE_XLNX_VERSAL_VIRT_MACHINE MACHINE_TYPE_NAME("xlnx-versal-virt")
33
+ switch (size) {
26
@@ -XXX,XX +XXX,XX @@ static void fdt_add_zdma_nodes(VersalVirt *s)
34
+ case MO_8:
35
+ tcg_gen_ext8u_i64(tcg_rt, tcg_rt);
36
+ break;
37
+ case MO_16:
38
+ tcg_gen_ext16u_i64(tcg_rt, tcg_rt);
39
+ break;
40
+ case MO_32:
41
+ tcg_gen_ext32u_i64(tcg_rt, tcg_rt);
42
+ break;
43
+ case MO_64:
44
+ break;
45
+ default:
46
+ g_assert_not_reached();
47
+ }
27
}
48
}
28
}
49
}
29
50
30
+static void fdt_add_sd_nodes(VersalVirt *s)
31
+{
32
+ const char clocknames[] = "clk_xin\0clk_ahb";
33
+ const char compat[] = "arasan,sdhci-8.9a";
34
+ int i;
35
+
36
+ for (i = ARRAY_SIZE(s->soc.pmc.iou.sd) - 1; i >= 0; i--) {
37
+ uint64_t addr = MM_PMC_SD0 + MM_PMC_SD0_SIZE * i;
38
+ char *name = g_strdup_printf("/sdhci@%" PRIx64, addr);
39
+
40
+ qemu_fdt_add_subnode(s->fdt, name);
41
+
42
+ qemu_fdt_setprop_cells(s->fdt, name, "clocks",
43
+ s->phandle.clk_25Mhz, s->phandle.clk_25Mhz);
44
+ qemu_fdt_setprop(s->fdt, name, "clock-names",
45
+ clocknames, sizeof(clocknames));
46
+ qemu_fdt_setprop_cells(s->fdt, name, "interrupts",
47
+ GIC_FDT_IRQ_TYPE_SPI, VERSAL_SD0_IRQ_0 + i * 2,
48
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI);
49
+ qemu_fdt_setprop_sized_cells(s->fdt, name, "reg",
50
+ 2, addr, 2, MM_PMC_SD0_SIZE);
51
+ qemu_fdt_setprop(s->fdt, name, "compatible", compat, sizeof(compat));
52
+ g_free(name);
53
+ }
54
+}
55
+
56
static void fdt_nop_memory_nodes(void *fdt, Error **errp)
57
{
58
Error *err = NULL;
59
@@ -XXX,XX +XXX,XX @@ static void create_virtio_regions(VersalVirt *s)
60
}
61
}
62
63
+static void sd_plugin_card(SDHCIState *sd, DriveInfo *di)
64
+{
65
+ BlockBackend *blk = di ? blk_by_legacy_dinfo(di) : NULL;
66
+ DeviceState *card;
67
+
68
+ card = qdev_create(qdev_get_child_bus(DEVICE(sd), "sd-bus"), TYPE_SD_CARD);
69
+ object_property_add_child(OBJECT(sd), "card[*]", OBJECT(card),
70
+ &error_fatal);
71
+ qdev_prop_set_drive(card, "drive", blk, &error_fatal);
72
+ object_property_set_bool(OBJECT(card), true, "realized", &error_fatal);
73
+}
74
+
75
static void versal_virt_init(MachineState *machine)
76
{
77
VersalVirt *s = XLNX_VERSAL_VIRT_MACHINE(machine);
78
int psci_conduit = QEMU_PSCI_CONDUIT_DISABLED;
79
+ int i;
80
81
/*
82
* If the user provides an Operating System to be loaded, we expect them
83
@@ -XXX,XX +XXX,XX @@ static void versal_virt_init(MachineState *machine)
84
fdt_add_gic_nodes(s);
85
fdt_add_timer_nodes(s);
86
fdt_add_zdma_nodes(s);
87
+ fdt_add_sd_nodes(s);
88
fdt_add_cpu_nodes(s, psci_conduit);
89
fdt_add_clk_node(s, "/clk125", 125000000, s->phandle.clk_125Mhz);
90
fdt_add_clk_node(s, "/clk25", 25000000, s->phandle.clk_25Mhz);
91
@@ -XXX,XX +XXX,XX @@ static void versal_virt_init(MachineState *machine)
92
memory_region_add_subregion_overlap(get_system_memory(),
93
0, &s->soc.fpd.apu.mr, 0);
94
95
+ /* Plugin SD cards. */
96
+ for (i = 0; i < ARRAY_SIZE(s->soc.pmc.iou.sd); i++) {
97
+ sd_plugin_card(&s->soc.pmc.iou.sd[i], drive_get_next(IF_SD));
98
+ }
99
+
100
s->binfo.ram_size = machine->ram_size;
101
s->binfo.loader_start = 0x0;
102
s->binfo.get_dtb = versal_virt_get_dtb;
103
--
51
--
104
2.20.1
52
2.34.1
105
106
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
The LDG instruction loads the tag from a memory address (identified
2
by [Xn + offset]), and then merges that tag into the destination
3
register Xt. We implemented this correctly for the case when
4
allocation tags are enabled, but didn't get it right when ATA=0:
5
instead of merging the tag bits into Xt, we merged them into the
6
memory address [Xn + offset] and then set Xt to that.
2
7
3
Add support for the RTC.
8
Merge the tag bits into the old Xt value, as they should be.
4
9
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
10
Cc: qemu-stable@nongnu.org
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
11
Fixes: c15294c1e36a7dd9b25 ("target/arm: Implement LDG, STG, ST2G instructions")
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
12
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200427181649.26851-12-edgar.iglesias@gmail.com
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
14
---
11
hw/arm/xlnx-versal-virt.c | 22 ++++++++++++++++++++++
15
target/arm/tcg/translate-a64.c | 6 +++++-
12
1 file changed, 22 insertions(+)
16
1 file changed, 5 insertions(+), 1 deletion(-)
13
17
14
diff --git a/hw/arm/xlnx-versal-virt.c b/hw/arm/xlnx-versal-virt.c
18
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
15
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/arm/xlnx-versal-virt.c
20
--- a/target/arm/tcg/translate-a64.c
17
+++ b/hw/arm/xlnx-versal-virt.c
21
+++ b/target/arm/tcg/translate-a64.c
18
@@ -XXX,XX +XXX,XX @@ static void fdt_add_sd_nodes(VersalVirt *s)
22
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
19
}
23
if (s->ata) {
20
}
24
gen_helper_ldg(tcg_rt, cpu_env, addr, tcg_rt);
21
25
} else {
22
+static void fdt_add_rtc_node(VersalVirt *s)
26
+ /*
23
+{
27
+ * Tag access disabled: we must check for aborts on the load
24
+ const char compat[] = "xlnx,zynqmp-rtc";
28
+ * load from [rn+offset], and then insert a 0 tag into rt.
25
+ const char interrupt_names[] = "alarm\0sec";
29
+ */
26
+ char *name = g_strdup_printf("/rtc@%x", MM_PMC_RTC);
30
clean_addr = clean_data_tbi(s, addr);
27
+
31
gen_probe_access(s, clean_addr, MMU_DATA_LOAD, MO_8);
28
+ qemu_fdt_add_subnode(s->fdt, name);
32
- gen_address_with_allocation_tag0(tcg_rt, addr);
29
+
33
+ gen_address_with_allocation_tag0(tcg_rt, tcg_rt);
30
+ qemu_fdt_setprop_cells(s->fdt, name, "interrupts",
34
}
31
+ GIC_FDT_IRQ_TYPE_SPI, VERSAL_RTC_ALARM_IRQ,
35
} else {
32
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI,
36
tcg_rt = cpu_reg_sp(s, rt);
33
+ GIC_FDT_IRQ_TYPE_SPI, VERSAL_RTC_SECONDS_IRQ,
34
+ GIC_FDT_IRQ_FLAGS_LEVEL_HI);
35
+ qemu_fdt_setprop(s->fdt, name, "interrupt-names",
36
+ interrupt_names, sizeof(interrupt_names));
37
+ qemu_fdt_setprop_sized_cells(s->fdt, name, "reg",
38
+ 2, MM_PMC_RTC, 2, MM_PMC_RTC_SIZE);
39
+ qemu_fdt_setprop(s->fdt, name, "compatible", compat, sizeof(compat));
40
+ g_free(name);
41
+}
42
+
43
static void fdt_nop_memory_nodes(void *fdt, Error **errp)
44
{
45
Error *err = NULL;
46
@@ -XXX,XX +XXX,XX @@ static void versal_virt_init(MachineState *machine)
47
fdt_add_timer_nodes(s);
48
fdt_add_zdma_nodes(s);
49
fdt_add_sd_nodes(s);
50
+ fdt_add_rtc_node(s);
51
fdt_add_cpu_nodes(s, psci_conduit);
52
fdt_add_clk_node(s, "/clk125", 125000000, s->phandle.clk_125Mhz);
53
fdt_add_clk_node(s, "/clk25", 25000000, s->phandle.clk_25Mhz);
54
--
37
--
55
2.20.1
38
2.34.1
56
57
diff view generated by jsdifflib
1
We're going to want at least some of the NeonGen* typedefs
1
In disas_ldst_reg_imm9() we missed one place where a call to
2
for the refactored 32-bit Neon decoder, so move them all
2
a gen_mte_check* function should now be passed the memop we
3
to translate.h since it makes more sense to keep them in
3
have created rather than just being passed the size. Fix this.
4
one group.
5
4
5
Fixes: 0a9091424d ("target/arm: Pass memop to gen_mte_check1*")
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200430181003.21682-23-peter.maydell@linaro.org
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
9
---
9
---
10
target/arm/translate.h | 17 +++++++++++++++++
10
target/arm/tcg/translate-a64.c | 2 +-
11
target/arm/translate-a64.c | 17 -----------------
11
1 file changed, 1 insertion(+), 1 deletion(-)
12
2 files changed, 17 insertions(+), 17 deletions(-)
13
12
14
diff --git a/target/arm/translate.h b/target/arm/translate.h
13
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
15
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/translate.h
15
--- a/target/arm/tcg/translate-a64.c
17
+++ b/target/arm/translate.h
16
+++ b/target/arm/tcg/translate-a64.c
18
@@ -XXX,XX +XXX,XX @@ typedef void GVecGen3Fn(unsigned, uint32_t, uint32_t,
17
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_imm9(DisasContext *s, uint32_t insn,
19
typedef void GVecGen4Fn(unsigned, uint32_t, uint32_t, uint32_t,
18
20
uint32_t, uint32_t, uint32_t);
19
clean_addr = gen_mte_check1_mmuidx(s, dirty_addr, is_store,
21
20
writeback || rn != 31,
22
+/* Function prototype for gen_ functions for calling Neon helpers */
21
- size, is_unpriv, memidx);
23
+typedef void NeonGenOneOpEnvFn(TCGv_i32, TCGv_ptr, TCGv_i32);
22
+ memop, is_unpriv, memidx);
24
+typedef void NeonGenTwoOpFn(TCGv_i32, TCGv_i32, TCGv_i32);
23
25
+typedef void NeonGenTwoOpEnvFn(TCGv_i32, TCGv_ptr, TCGv_i32, TCGv_i32);
24
if (is_vector) {
26
+typedef void NeonGenTwo64OpFn(TCGv_i64, TCGv_i64, TCGv_i64);
25
if (is_store) {
27
+typedef void NeonGenTwo64OpEnvFn(TCGv_i64, TCGv_ptr, TCGv_i64, TCGv_i64);
28
+typedef void NeonGenNarrowFn(TCGv_i32, TCGv_i64);
29
+typedef void NeonGenNarrowEnvFn(TCGv_i32, TCGv_ptr, TCGv_i64);
30
+typedef void NeonGenWidenFn(TCGv_i64, TCGv_i32);
31
+typedef void NeonGenTwoSingleOPFn(TCGv_i32, TCGv_i32, TCGv_i32, TCGv_ptr);
32
+typedef void NeonGenTwoDoubleOPFn(TCGv_i64, TCGv_i64, TCGv_i64, TCGv_ptr);
33
+typedef void NeonGenOneOpFn(TCGv_i64, TCGv_i64);
34
+typedef void CryptoTwoOpFn(TCGv_ptr, TCGv_ptr);
35
+typedef void CryptoThreeOpIntFn(TCGv_ptr, TCGv_ptr, TCGv_i32);
36
+typedef void CryptoThreeOpFn(TCGv_ptr, TCGv_ptr, TCGv_ptr);
37
+typedef void AtomicThreeOpFn(TCGv_i64, TCGv_i64, TCGv_i64, TCGArg, MemOp);
38
+
39
#endif /* TARGET_ARM_TRANSLATE_H */
40
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
41
index XXXXXXX..XXXXXXX 100644
42
--- a/target/arm/translate-a64.c
43
+++ b/target/arm/translate-a64.c
44
@@ -XXX,XX +XXX,XX @@ typedef struct AArch64DecodeTable {
45
AArch64DecodeFn *disas_fn;
46
} AArch64DecodeTable;
47
48
-/* Function prototype for gen_ functions for calling Neon helpers */
49
-typedef void NeonGenOneOpEnvFn(TCGv_i32, TCGv_ptr, TCGv_i32);
50
-typedef void NeonGenTwoOpFn(TCGv_i32, TCGv_i32, TCGv_i32);
51
-typedef void NeonGenTwoOpEnvFn(TCGv_i32, TCGv_ptr, TCGv_i32, TCGv_i32);
52
-typedef void NeonGenTwo64OpFn(TCGv_i64, TCGv_i64, TCGv_i64);
53
-typedef void NeonGenTwo64OpEnvFn(TCGv_i64, TCGv_ptr, TCGv_i64, TCGv_i64);
54
-typedef void NeonGenNarrowFn(TCGv_i32, TCGv_i64);
55
-typedef void NeonGenNarrowEnvFn(TCGv_i32, TCGv_ptr, TCGv_i64);
56
-typedef void NeonGenWidenFn(TCGv_i64, TCGv_i32);
57
-typedef void NeonGenTwoSingleOPFn(TCGv_i32, TCGv_i32, TCGv_i32, TCGv_ptr);
58
-typedef void NeonGenTwoDoubleOPFn(TCGv_i64, TCGv_i64, TCGv_i64, TCGv_ptr);
59
-typedef void NeonGenOneOpFn(TCGv_i64, TCGv_i64);
60
-typedef void CryptoTwoOpFn(TCGv_ptr, TCGv_ptr);
61
-typedef void CryptoThreeOpIntFn(TCGv_ptr, TCGv_ptr, TCGv_i32);
62
-typedef void CryptoThreeOpFn(TCGv_ptr, TCGv_ptr, TCGv_ptr);
63
-typedef void AtomicThreeOpFn(TCGv_i64, TCGv_i64, TCGv_i64, TCGArg, MemOp);
64
-
65
/* initialize TCG globals. */
66
void a64_translate_init(void)
67
{
68
--
26
--
69
2.20.1
27
2.34.1
70
28
71
29
diff view generated by jsdifflib
1
Somewhere along theline we accidentally added a duplicate
1
In the recent refactoring we missed a few places which should be
2
"using D16-D31 when they don't exist" check to do_vfm_dp()
2
calling finalize_memop_asimd() for ASIMD loads and stores but
3
(probably an artifact of a patchseries rebase). Remove it.
3
instead are just calling finalize_memop(); fix these.
4
5
For the disas_ldst_single_struct() and disas_ldst_multiple_struct()
6
cases, this is not a behaviour change because there the size
7
is never MO_128 and the two finalize functions do the same thing.
4
8
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Message-id: 20200430181003.21682-2-peter.maydell@linaro.org
9
---
11
---
10
target/arm/translate-vfp.inc.c | 6 ------
12
target/arm/tcg/translate-a64.c | 10 ++++++----
11
1 file changed, 6 deletions(-)
13
1 file changed, 6 insertions(+), 4 deletions(-)
12
14
13
diff --git a/target/arm/translate-vfp.inc.c b/target/arm/translate-vfp.inc.c
15
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
14
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/translate-vfp.inc.c
17
--- a/target/arm/tcg/translate-a64.c
16
+++ b/target/arm/translate-vfp.inc.c
18
+++ b/target/arm/tcg/translate-a64.c
17
@@ -XXX,XX +XXX,XX @@ static bool do_vfm_dp(DisasContext *s, arg_VFMA_dp *a, bool neg_n, bool neg_d)
19
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
18
return false;
20
if (!fp_access_check(s)) {
21
return;
22
}
23
+ memop = finalize_memop_asimd(s, size);
24
} else {
25
if (size == 3 && opc == 2) {
26
/* PRFM - prefetch */
27
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
28
is_store = (opc == 0);
29
is_signed = !is_store && extract32(opc, 1, 1);
30
is_extended = (size < 3) && extract32(opc, 0, 1);
31
+ memop = finalize_memop(s, size + is_signed * MO_SIGN);
19
}
32
}
20
33
21
- /* UNDEF accesses to D16-D31 if they don't exist. */
34
if (rn == 31) {
22
- if (!dc_isar_feature(aa32_simd_r32, s) &&
35
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
23
- ((a->vd | a->vn | a->vm) & 0x10)) {
36
24
- return false;
37
tcg_gen_add_i64(dirty_addr, dirty_addr, tcg_rm);
25
- }
38
26
-
39
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
27
if (!vfp_access_check(s)) {
40
clean_addr = gen_mte_check1(s, dirty_addr, is_store, true, memop);
28
return true;
41
42
if (is_vector) {
43
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_unsigned_imm(DisasContext *s, uint32_t insn,
44
if (!fp_access_check(s)) {
45
return;
46
}
47
+ memop = finalize_memop_asimd(s, size);
48
} else {
49
if (size == 3 && opc == 2) {
50
/* PRFM - prefetch */
51
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_unsigned_imm(DisasContext *s, uint32_t insn,
52
is_store = (opc == 0);
53
is_signed = !is_store && extract32(opc, 1, 1);
54
is_extended = (size < 3) && extract32(opc, 0, 1);
55
+ memop = finalize_memop(s, size + is_signed * MO_SIGN);
29
}
56
}
57
58
if (rn == 31) {
59
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_unsigned_imm(DisasContext *s, uint32_t insn,
60
offset = imm12 << size;
61
tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
62
63
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
64
clean_addr = gen_mte_check1(s, dirty_addr, is_store, rn != 31, memop);
65
66
if (is_vector) {
67
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
68
* promote consecutive little-endian elements below.
69
*/
70
clean_addr = gen_mte_checkN(s, tcg_rn, is_store, is_postidx || rn != 31,
71
- total, finalize_memop(s, size));
72
+ total, finalize_memop_asimd(s, size));
73
74
/*
75
* Consecutive little-endian elements from a single register
76
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
77
total = selem << scale;
78
tcg_rn = cpu_reg_sp(s, rn);
79
80
- mop = finalize_memop(s, scale);
81
+ mop = finalize_memop_asimd(s, scale);
82
83
clean_addr = gen_mte_checkN(s, tcg_rn, !is_load, is_postidx || rn != 31,
84
total, mop);
30
--
85
--
31
2.20.1
86
2.34.1
32
33
diff view generated by jsdifflib
1
Convert the Neon VQADD/VQSUB insns in the 3-reg-same grouping
1
Convert the various instructions in the hint instruction space
2
to decodetree.
2
to decodetree.
3
3
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-19-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-3-peter.maydell@linaro.org
7
---
7
---
8
target/arm/neon-dp.decode | 6 ++++++
8
target/arm/tcg/a64.decode | 31 ++++
9
target/arm/translate-neon.inc.c | 15 +++++++++++++++
9
target/arm/tcg/translate-a64.c | 277 ++++++++++++++++++---------------
10
target/arm/translate.c | 14 ++------------
10
2 files changed, 185 insertions(+), 123 deletions(-)
11
3 files changed, 23 insertions(+), 12 deletions(-)
12
11
13
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
14
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-dp.decode
14
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-dp.decode
15
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ ERETA 1101011 0100 11111 00001 m:1 11111 11111 &reta # ERETAA, ERETAB
18
@3same .... ... . . . size:2 .... .... .... . q:1 . . .... \
17
# the processor is in halting debug state (which we don't implement).
19
&3same vm=%vm_dp vn=%vn_dp vd=%vd_dp
18
# The pattern is listed here as documentation.
20
19
# DRPS 1101011 0101 11111 000000 11111 00000
21
+VQADD_S_3s 1111 001 0 0 . .. .... .... 0000 . . . 1 .... @3same
20
+
22
+VQADD_U_3s 1111 001 1 0 . .. .... .... 0000 . . . 1 .... @3same
21
+# Hint instruction group
23
+
22
+{
24
@3same_logic .... ... . . . .. .... .... .... . q:1 .. .... \
23
+ [
25
&3same vm=%vm_dp vn=%vn_dp vd=%vd_dp size=0
24
+ YIELD 1101 0101 0000 0011 0010 0000 001 11111
26
25
+ WFE 1101 0101 0000 0011 0010 0000 010 11111
27
@@ -XXX,XX +XXX,XX @@ VBSL_3s 1111 001 1 0 . 01 .... .... 0001 ... 1 .... @3same_logic
26
+ WFI 1101 0101 0000 0011 0010 0000 011 11111
28
VBIT_3s 1111 001 1 0 . 10 .... .... 0001 ... 1 .... @3same_logic
27
+ # We implement WFE to never block, so our SEV/SEVL are NOPs
29
VBIF_3s 1111 001 1 0 . 11 .... .... 0001 ... 1 .... @3same_logic
28
+ # SEV 1101 0101 0000 0011 0010 0000 100 11111
30
29
+ # SEVL 1101 0101 0000 0011 0010 0000 101 11111
31
+VQSUB_S_3s 1111 001 0 0 . .. .... .... 0010 . . . 1 .... @3same
30
+ # Our DGL is a NOP because we don't merge memory accesses anyway.
32
+VQSUB_U_3s 1111 001 1 0 . .. .... .... 0010 . . . 1 .... @3same
31
+ # DGL 1101 0101 0000 0011 0010 0000 110 11111
33
+
32
+ XPACLRI 1101 0101 0000 0011 0010 0000 111 11111
34
VCGT_S_3s 1111 001 0 0 . .. .... .... 0011 . . . 0 .... @3same
33
+ PACIA1716 1101 0101 0000 0011 0010 0001 000 11111
35
VCGT_U_3s 1111 001 1 0 . .. .... .... 0011 . . . 0 .... @3same
34
+ PACIB1716 1101 0101 0000 0011 0010 0001 010 11111
36
VCGE_S_3s 1111 001 0 0 . .. .... .... 0011 . . . 1 .... @3same
35
+ AUTIA1716 1101 0101 0000 0011 0010 0001 100 11111
37
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
36
+ AUTIB1716 1101 0101 0000 0011 0010 0001 110 11111
37
+ ESB 1101 0101 0000 0011 0010 0010 000 11111
38
+ PACIAZ 1101 0101 0000 0011 0010 0011 000 11111
39
+ PACIASP 1101 0101 0000 0011 0010 0011 001 11111
40
+ PACIBZ 1101 0101 0000 0011 0010 0011 010 11111
41
+ PACIBSP 1101 0101 0000 0011 0010 0011 011 11111
42
+ AUTIAZ 1101 0101 0000 0011 0010 0011 100 11111
43
+ AUTIASP 1101 0101 0000 0011 0010 0011 101 11111
44
+ AUTIBZ 1101 0101 0000 0011 0010 0011 110 11111
45
+ AUTIBSP 1101 0101 0000 0011 0010 0011 111 11111
46
+ ]
47
+ # The canonical NOP has CRm == op2 == 0, but all of the space
48
+ # that isn't specifically allocated to an instruction must NOP
49
+ NOP 1101 0101 0000 0011 0010 ---- --- 11111
50
+}
51
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
38
index XXXXXXX..XXXXXXX 100644
52
index XXXXXXX..XXXXXXX 100644
39
--- a/target/arm/translate-neon.inc.c
53
--- a/target/arm/tcg/translate-a64.c
40
+++ b/target/arm/translate-neon.inc.c
54
+++ b/target/arm/tcg/translate-a64.c
41
@@ -XXX,XX +XXX,XX @@ static void gen_VTST_3s(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
55
@@ -XXX,XX +XXX,XX @@ static bool trans_ERETA(DisasContext *s, arg_reta *a)
42
tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, oprsz, maxsz, &cmtst_op[vece]);
56
return true;
43
}
57
}
44
DO_3SAME_NO_SZ_3(VTST, gen_VTST_3s)
58
45
+
59
-/* HINT instruction group, including various allocated HINTs */
46
+#define DO_3SAME_GVEC4(INSN, OPARRAY) \
60
-static void handle_hint(DisasContext *s, uint32_t insn,
47
+ static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
61
- unsigned int op1, unsigned int op2, unsigned int crm)
48
+ uint32_t rn_ofs, uint32_t rm_ofs, \
62
+static bool trans_NOP(DisasContext *s, arg_NOP *a)
49
+ uint32_t oprsz, uint32_t maxsz) \
63
{
50
+ { \
64
- unsigned int selector = crm << 3 | op2;
51
+ tcg_gen_gvec_4(rd_ofs, offsetof(CPUARMState, vfp.qc), \
65
+ return true;
52
+ rn_ofs, rm_ofs, oprsz, maxsz, &OPARRAY[vece]); \
66
+}
53
+ } \
67
54
+ DO_3SAME(INSN, gen_##INSN##_3s)
68
- if (op1 != 3) {
55
+
69
- unallocated_encoding(s);
56
+DO_3SAME_GVEC4(VQADD_S, sqadd_op)
70
- return;
57
+DO_3SAME_GVEC4(VQADD_U, uqadd_op)
71
+static bool trans_YIELD(DisasContext *s, arg_YIELD *a)
58
+DO_3SAME_GVEC4(VQSUB_S, sqsub_op)
72
+{
59
+DO_3SAME_GVEC4(VQSUB_U, uqsub_op)
73
+ /*
60
diff --git a/target/arm/translate.c b/target/arm/translate.c
74
+ * When running in MTTCG we don't generate jumps to the yield and
61
index XXXXXXX..XXXXXXX 100644
75
+ * WFE helpers as it won't affect the scheduling of other vCPUs.
62
--- a/target/arm/translate.c
76
+ * If we wanted to more completely model WFE/SEV so we don't busy
63
+++ b/target/arm/translate.c
77
+ * spin unnecessarily we would need to do something more involved.
64
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
78
+ */
65
}
79
+ if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
66
return 1;
80
+ s->base.is_jmp = DISAS_YIELD;
67
81
}
68
- case NEON_3R_VQADD:
82
+ return true;
69
- tcg_gen_gvec_4(rd_ofs, offsetof(CPUARMState, vfp.qc),
83
+}
70
- rn_ofs, rm_ofs, vec_size, vec_size,
84
71
- (u ? uqadd_op : sqadd_op) + size);
85
- switch (selector) {
72
- return 0;
86
- case 0b00000: /* NOP */
73
-
87
- break;
74
- case NEON_3R_VQSUB:
88
- case 0b00011: /* WFI */
75
- tcg_gen_gvec_4(rd_ofs, offsetof(CPUARMState, vfp.qc),
89
- s->base.is_jmp = DISAS_WFI;
76
- rn_ofs, rm_ofs, vec_size, vec_size,
90
- break;
77
- (u ? uqsub_op : sqsub_op) + size);
91
- case 0b00001: /* YIELD */
78
- return 0;
92
- /* When running in MTTCG we don't generate jumps to the yield and
79
-
93
- * WFE helpers as it won't affect the scheduling of other vCPUs.
80
case NEON_3R_VMUL: /* VMUL */
94
- * If we wanted to more completely model WFE/SEV so we don't busy
81
if (u) {
95
- * spin unnecessarily we would need to do something more involved.
82
/* Polynomial case allows only P8. */
96
+static bool trans_WFI(DisasContext *s, arg_WFI *a)
83
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
97
+{
84
case NEON_3R_VTST_VCEQ:
98
+ s->base.is_jmp = DISAS_WFI;
85
case NEON_3R_VCGT:
99
+ return true;
86
case NEON_3R_VCGE:
100
+}
87
+ case NEON_3R_VQADD:
101
+
88
+ case NEON_3R_VQSUB:
102
+static bool trans_WFE(DisasContext *s, arg_WFI *a)
89
/* Already handled by decodetree */
103
+{
90
return 1;
104
+ /*
105
+ * When running in MTTCG we don't generate jumps to the yield and
106
+ * WFE helpers as it won't affect the scheduling of other vCPUs.
107
+ * If we wanted to more completely model WFE/SEV so we don't busy
108
+ * spin unnecessarily we would need to do something more involved.
109
+ */
110
+ if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
111
+ s->base.is_jmp = DISAS_WFE;
112
+ }
113
+ return true;
114
+}
115
+
116
+static bool trans_XPACLRI(DisasContext *s, arg_XPACLRI *a)
117
+{
118
+ if (s->pauth_active) {
119
+ gen_helper_xpaci(cpu_X[30], cpu_env, cpu_X[30]);
120
+ }
121
+ return true;
122
+}
123
+
124
+static bool trans_PACIA1716(DisasContext *s, arg_PACIA1716 *a)
125
+{
126
+ if (s->pauth_active) {
127
+ gen_helper_pacia(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
128
+ }
129
+ return true;
130
+}
131
+
132
+static bool trans_PACIB1716(DisasContext *s, arg_PACIB1716 *a)
133
+{
134
+ if (s->pauth_active) {
135
+ gen_helper_pacib(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
136
+ }
137
+ return true;
138
+}
139
+
140
+static bool trans_AUTIA1716(DisasContext *s, arg_AUTIA1716 *a)
141
+{
142
+ if (s->pauth_active) {
143
+ gen_helper_autia(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
144
+ }
145
+ return true;
146
+}
147
+
148
+static bool trans_AUTIB1716(DisasContext *s, arg_AUTIB1716 *a)
149
+{
150
+ if (s->pauth_active) {
151
+ gen_helper_autib(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
152
+ }
153
+ return true;
154
+}
155
+
156
+static bool trans_ESB(DisasContext *s, arg_ESB *a)
157
+{
158
+ /* Without RAS, we must implement this as NOP. */
159
+ if (dc_isar_feature(aa64_ras, s)) {
160
+ /*
161
+ * QEMU does not have a source of physical SErrors,
162
+ * so we are only concerned with virtual SErrors.
163
+ * The pseudocode in the ARM for this case is
164
+ * if PSTATE.EL IN {EL0, EL1} && EL2Enabled() then
165
+ * AArch64.vESBOperation();
166
+ * Most of the condition can be evaluated at translation time.
167
+ * Test for EL2 present, and defer test for SEL2 to runtime.
168
*/
169
- if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
170
- s->base.is_jmp = DISAS_YIELD;
171
+ if (s->current_el <= 1 && arm_dc_feature(s, ARM_FEATURE_EL2)) {
172
+ gen_helper_vesb(cpu_env);
91
}
173
}
174
- break;
175
- case 0b00010: /* WFE */
176
- if (!(tb_cflags(s->base.tb) & CF_PARALLEL)) {
177
- s->base.is_jmp = DISAS_WFE;
178
- }
179
- break;
180
- case 0b00100: /* SEV */
181
- case 0b00101: /* SEVL */
182
- case 0b00110: /* DGH */
183
- /* we treat all as NOP at least for now */
184
- break;
185
- case 0b00111: /* XPACLRI */
186
- if (s->pauth_active) {
187
- gen_helper_xpaci(cpu_X[30], cpu_env, cpu_X[30]);
188
- }
189
- break;
190
- case 0b01000: /* PACIA1716 */
191
- if (s->pauth_active) {
192
- gen_helper_pacia(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
193
- }
194
- break;
195
- case 0b01010: /* PACIB1716 */
196
- if (s->pauth_active) {
197
- gen_helper_pacib(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
198
- }
199
- break;
200
- case 0b01100: /* AUTIA1716 */
201
- if (s->pauth_active) {
202
- gen_helper_autia(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
203
- }
204
- break;
205
- case 0b01110: /* AUTIB1716 */
206
- if (s->pauth_active) {
207
- gen_helper_autib(cpu_X[17], cpu_env, cpu_X[17], cpu_X[16]);
208
- }
209
- break;
210
- case 0b10000: /* ESB */
211
- /* Without RAS, we must implement this as NOP. */
212
- if (dc_isar_feature(aa64_ras, s)) {
213
- /*
214
- * QEMU does not have a source of physical SErrors,
215
- * so we are only concerned with virtual SErrors.
216
- * The pseudocode in the ARM for this case is
217
- * if PSTATE.EL IN {EL0, EL1} && EL2Enabled() then
218
- * AArch64.vESBOperation();
219
- * Most of the condition can be evaluated at translation time.
220
- * Test for EL2 present, and defer test for SEL2 to runtime.
221
- */
222
- if (s->current_el <= 1 && arm_dc_feature(s, ARM_FEATURE_EL2)) {
223
- gen_helper_vesb(cpu_env);
224
- }
225
- }
226
- break;
227
- case 0b11000: /* PACIAZ */
228
- if (s->pauth_active) {
229
- gen_helper_pacia(cpu_X[30], cpu_env, cpu_X[30],
230
- tcg_constant_i64(0));
231
- }
232
- break;
233
- case 0b11001: /* PACIASP */
234
- if (s->pauth_active) {
235
- gen_helper_pacia(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
236
- }
237
- break;
238
- case 0b11010: /* PACIBZ */
239
- if (s->pauth_active) {
240
- gen_helper_pacib(cpu_X[30], cpu_env, cpu_X[30],
241
- tcg_constant_i64(0));
242
- }
243
- break;
244
- case 0b11011: /* PACIBSP */
245
- if (s->pauth_active) {
246
- gen_helper_pacib(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
247
- }
248
- break;
249
- case 0b11100: /* AUTIAZ */
250
- if (s->pauth_active) {
251
- gen_helper_autia(cpu_X[30], cpu_env, cpu_X[30],
252
- tcg_constant_i64(0));
253
- }
254
- break;
255
- case 0b11101: /* AUTIASP */
256
- if (s->pauth_active) {
257
- gen_helper_autia(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
258
- }
259
- break;
260
- case 0b11110: /* AUTIBZ */
261
- if (s->pauth_active) {
262
- gen_helper_autib(cpu_X[30], cpu_env, cpu_X[30],
263
- tcg_constant_i64(0));
264
- }
265
- break;
266
- case 0b11111: /* AUTIBSP */
267
- if (s->pauth_active) {
268
- gen_helper_autib(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
269
- }
270
- break;
271
- default:
272
- /* default specified as NOP equivalent */
273
- break;
274
}
275
+ return true;
276
+}
277
+
278
+static bool trans_PACIAZ(DisasContext *s, arg_PACIAZ *a)
279
+{
280
+ if (s->pauth_active) {
281
+ gen_helper_pacia(cpu_X[30], cpu_env, cpu_X[30], tcg_constant_i64(0));
282
+ }
283
+ return true;
284
+}
285
+
286
+static bool trans_PACIASP(DisasContext *s, arg_PACIASP *a)
287
+{
288
+ if (s->pauth_active) {
289
+ gen_helper_pacia(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
290
+ }
291
+ return true;
292
+}
293
+
294
+static bool trans_PACIBZ(DisasContext *s, arg_PACIBZ *a)
295
+{
296
+ if (s->pauth_active) {
297
+ gen_helper_pacib(cpu_X[30], cpu_env, cpu_X[30], tcg_constant_i64(0));
298
+ }
299
+ return true;
300
+}
301
+
302
+static bool trans_PACIBSP(DisasContext *s, arg_PACIBSP *a)
303
+{
304
+ if (s->pauth_active) {
305
+ gen_helper_pacib(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
306
+ }
307
+ return true;
308
+}
309
+
310
+static bool trans_AUTIAZ(DisasContext *s, arg_AUTIAZ *a)
311
+{
312
+ if (s->pauth_active) {
313
+ gen_helper_autia(cpu_X[30], cpu_env, cpu_X[30], tcg_constant_i64(0));
314
+ }
315
+ return true;
316
+}
317
+
318
+static bool trans_AUTIASP(DisasContext *s, arg_AUTIASP *a)
319
+{
320
+ if (s->pauth_active) {
321
+ gen_helper_autia(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
322
+ }
323
+ return true;
324
+}
325
+
326
+static bool trans_AUTIBZ(DisasContext *s, arg_AUTIBZ *a)
327
+{
328
+ if (s->pauth_active) {
329
+ gen_helper_autib(cpu_X[30], cpu_env, cpu_X[30], tcg_constant_i64(0));
330
+ }
331
+ return true;
332
+}
333
+
334
+static bool trans_AUTIBSP(DisasContext *s, arg_AUTIBSP *a)
335
+{
336
+ if (s->pauth_active) {
337
+ gen_helper_autib(cpu_X[30], cpu_env, cpu_X[30], cpu_X[31]);
338
+ }
339
+ return true;
340
}
341
342
static void gen_clrex(DisasContext *s, uint32_t insn)
343
@@ -XXX,XX +XXX,XX @@ static void disas_system(DisasContext *s, uint32_t insn)
344
return;
345
}
346
switch (crn) {
347
- case 2: /* HINT (including allocated hints like NOP, YIELD, etc) */
348
- handle_hint(s, insn, op1, op2, crm);
349
- break;
350
case 3: /* CLREX, DSB, DMB, ISB */
351
handle_sync(s, insn, op1, op2, crm);
352
break;
92
--
353
--
93
2.20.1
354
2.34.1
94
95
diff view generated by jsdifflib
1
Convert the V[US]DOT (scalar) insns in the 2reg-scalar-ext group
1
Convert the insns in the "Barriers" instruction class to
2
to decodetree.
2
decodetree: CLREX, DSB, DMB, ISB and SB.
3
3
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-10-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-4-peter.maydell@linaro.org
7
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
7
---
8
---
8
target/arm/neon-shared.decode | 3 +++
9
target/arm/tcg/a64.decode | 7 +++
9
target/arm/translate-neon.inc.c | 35 +++++++++++++++++++++++++++++++++
10
target/arm/tcg/translate-a64.c | 92 ++++++++++++++--------------------
10
target/arm/translate.c | 13 +-----------
11
2 files changed, 46 insertions(+), 53 deletions(-)
11
3 files changed, 39 insertions(+), 12 deletions(-)
12
12
13
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
14
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-shared.decode
15
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-shared.decode
16
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@ VCMLA_scalar 1111 1110 0 . rot:2 .... .... 1000 . q:1 index:1 0 vm:4 \
17
@@ -XXX,XX +XXX,XX @@ ERETA 1101011 0100 11111 00001 m:1 11111 11111 &reta # ERETAA, ERETAB
18
vn=%vn_dp vd=%vd_dp size=0
18
# that isn't specifically allocated to an instruction must NOP
19
VCMLA_scalar 1111 1110 1 . rot:2 .... .... 1000 . q:1 . 0 .... \
19
NOP 1101 0101 0000 0011 0010 ---- --- 11111
20
vm=%vm_dp vn=%vn_dp vd=%vd_dp size=1 index=0
20
}
21
+
21
+
22
+VDOT_scalar 1111 1110 0 . 10 .... .... 1101 . q:1 index:1 u:1 rm:4 \
22
+# Barriers
23
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp
23
+
24
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
24
+CLREX 1101 0101 0000 0011 0011 ---- 010 11111
25
+DSB_DMB 1101 0101 0000 0011 0011 domain:2 types:2 10- 11111
26
+ISB 1101 0101 0000 0011 0011 ---- 110 11111
27
+SB 1101 0101 0000 0011 0011 0000 111 11111
28
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
25
index XXXXXXX..XXXXXXX 100644
29
index XXXXXXX..XXXXXXX 100644
26
--- a/target/arm/translate-neon.inc.c
30
--- a/target/arm/tcg/translate-a64.c
27
+++ b/target/arm/translate-neon.inc.c
31
+++ b/target/arm/tcg/translate-a64.c
28
@@ -XXX,XX +XXX,XX @@ static bool trans_VCMLA_scalar(DisasContext *s, arg_VCMLA_scalar *a)
32
@@ -XXX,XX +XXX,XX @@ static bool trans_AUTIBSP(DisasContext *s, arg_AUTIBSP *a)
29
tcg_temp_free_ptr(fpst);
30
return true;
33
return true;
31
}
34
}
32
+
35
33
+static bool trans_VDOT_scalar(DisasContext *s, arg_VDOT_scalar *a)
36
-static void gen_clrex(DisasContext *s, uint32_t insn)
34
+{
37
+static bool trans_CLREX(DisasContext *s, arg_CLREX *a)
35
+ gen_helper_gvec_3 *fn_gvec;
38
{
36
+ int opr_sz;
39
tcg_gen_movi_i64(cpu_exclusive_addr, -1);
37
+ TCGv_ptr fpst;
40
+ return true;
38
+
41
}
39
+ if (!dc_isar_feature(aa32_dp, s)) {
42
40
+ return false;
43
-/* CLREX, DSB, DMB, ISB */
41
+ }
44
-static void handle_sync(DisasContext *s, uint32_t insn,
42
+
45
- unsigned int op1, unsigned int op2, unsigned int crm)
43
+ /* UNDEF accesses to D16-D31 if they don't exist. */
46
+static bool trans_DSB_DMB(DisasContext *s, arg_DSB_DMB *a)
44
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
47
{
45
+ ((a->vd | a->vn) & 0x10)) {
48
+ /* We handle DSB and DMB the same way */
46
+ return false;
49
TCGBar bar;
47
+ }
50
48
+
51
- if (op1 != 3) {
49
+ if ((a->vd | a->vn) & a->q) {
52
- unallocated_encoding(s);
50
+ return false;
53
- return;
51
+ }
54
+ switch (a->types) {
52
+
55
+ case 1: /* MBReqTypes_Reads */
53
+ if (!vfp_access_check(s)) {
56
+ bar = TCG_BAR_SC | TCG_MO_LD_LD | TCG_MO_LD_ST;
54
+ return true;
57
+ break;
55
+ }
58
+ case 2: /* MBReqTypes_Writes */
56
+
59
+ bar = TCG_BAR_SC | TCG_MO_ST_ST;
57
+ fn_gvec = a->u ? gen_helper_gvec_udot_idx_b : gen_helper_gvec_sdot_idx_b;
60
+ break;
58
+ opr_sz = (1 + a->q) * 8;
61
+ default: /* MBReqTypes_All */
59
+ fpst = get_fpstatus_ptr(1);
62
+ bar = TCG_BAR_SC | TCG_MO_ALL;
60
+ tcg_gen_gvec_3_ool(vfp_reg_offset(1, a->vd),
63
+ break;
61
+ vfp_reg_offset(1, a->vn),
64
}
62
+ vfp_reg_offset(1, a->rm),
65
+ tcg_gen_mb(bar);
63
+ opr_sz, opr_sz, a->index, fn_gvec);
64
+ tcg_temp_free_ptr(fpst);
65
+ return true;
66
+ return true;
66
+}
67
+}
67
diff --git a/target/arm/translate.c b/target/arm/translate.c
68
68
index XXXXXXX..XXXXXXX 100644
69
- switch (op2) {
69
--- a/target/arm/translate.c
70
- case 2: /* CLREX */
70
+++ b/target/arm/translate.c
71
- gen_clrex(s, insn);
71
@@ -XXX,XX +XXX,XX @@ static int disas_neon_insn_2reg_scalar_ext(DisasContext *s, uint32_t insn)
72
- return;
72
bool is_long = false, q = extract32(insn, 6, 1);
73
- case 4: /* DSB */
73
bool ptr_is_env = false;
74
- case 5: /* DMB */
74
75
- switch (crm & 3) {
75
- if ((insn & 0xffb00f00) == 0xfe200d00) {
76
- case 1: /* MBReqTypes_Reads */
76
- /* V[US]DOT -- 1111 1110 0.10 .... .... 1101 .Q.U .... */
77
- bar = TCG_BAR_SC | TCG_MO_LD_LD | TCG_MO_LD_ST;
77
- int u = extract32(insn, 4, 1);
78
- break;
79
- case 2: /* MBReqTypes_Writes */
80
- bar = TCG_BAR_SC | TCG_MO_ST_ST;
81
- break;
82
- default: /* MBReqTypes_All */
83
- bar = TCG_BAR_SC | TCG_MO_ALL;
84
- break;
85
- }
86
- tcg_gen_mb(bar);
87
- return;
88
- case 6: /* ISB */
89
- /* We need to break the TB after this insn to execute
90
- * a self-modified code correctly and also to take
91
- * any pending interrupts immediately.
92
- */
93
- reset_btype(s);
94
- gen_goto_tb(s, 0, 4);
95
- return;
96
+static bool trans_ISB(DisasContext *s, arg_ISB *a)
97
+{
98
+ /*
99
+ * We need to break the TB after this insn to execute
100
+ * self-modifying code correctly and also to take
101
+ * any pending interrupts immediately.
102
+ */
103
+ reset_btype(s);
104
+ gen_goto_tb(s, 0, 4);
105
+ return true;
106
+}
107
108
- case 7: /* SB */
109
- if (crm != 0 || !dc_isar_feature(aa64_sb, s)) {
110
- goto do_unallocated;
111
- }
112
- /*
113
- * TODO: There is no speculation barrier opcode for TCG;
114
- * MB and end the TB instead.
115
- */
116
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_SC);
117
- gen_goto_tb(s, 0, 4);
118
- return;
78
-
119
-
79
- if (!dc_isar_feature(aa32_dp, s)) {
120
- default:
80
- return 1;
121
- do_unallocated:
81
- }
122
- unallocated_encoding(s);
82
- fn_gvec = u ? gen_helper_gvec_udot_idx_b : gen_helper_gvec_sdot_idx_b;
123
- return;
83
- /* rm is just Vm, and index is M. */
124
+static bool trans_SB(DisasContext *s, arg_SB *a)
84
- data = extract32(insn, 5, 1); /* index */
125
+{
85
- rm = extract32(insn, 0, 4);
126
+ if (!dc_isar_feature(aa64_sb, s)) {
86
- } else if ((insn & 0xffa00f10) == 0xfe000810) {
127
+ return false;
87
+ if ((insn & 0xffa00f10) == 0xfe000810) {
128
}
88
/* VFM[AS]L -- 1111 1110 0.0S .... .... 1000 .Q.1 .... */
129
+ /*
89
int is_s = extract32(insn, 20, 1);
130
+ * TODO: There is no speculation barrier opcode for TCG;
90
int vm20 = extract32(insn, 0, 3);
131
+ * MB and end the TB instead.
132
+ */
133
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_SC);
134
+ gen_goto_tb(s, 0, 4);
135
+ return true;
136
}
137
138
static void gen_xaflag(void)
139
@@ -XXX,XX +XXX,XX @@ static void disas_system(DisasContext *s, uint32_t insn)
140
return;
141
}
142
switch (crn) {
143
- case 3: /* CLREX, DSB, DMB, ISB */
144
- handle_sync(s, insn, op1, op2, crm);
145
- break;
146
case 4: /* MSR (immediate) */
147
handle_msr_i(s, insn, op1, op2, crm);
148
break;
91
--
149
--
92
2.20.1
150
2.34.1
93
151
94
152
diff view generated by jsdifflib
1
Convert the VFM[AS]L (vector) insns to decodetree. This is the last
1
Convert the CFINV, XAFLAG and AXFLAG insns to decodetree.
2
insn in the legacy decoder for the 3same_ext group, so we can
2
The old decoder handles these in handle_msr_i(), but
3
delete the legacy decoder function for the group entirely.
3
the architecture defines them as separate instructions
4
4
from MSR (immediate).
5
Note that in disas_thumb2_insn() the parts of this encoding space
6
where the decodetree decoder returns false will correctly be directed
7
to illegal_op by the "(insn & (1 << 28))" check so they won't fall
8
into disas_coproc_insn() by mistake.
9
5
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
Message-id: 20200430181003.21682-8-peter.maydell@linaro.org
8
Message-id: 20230602155223.2040685-5-peter.maydell@linaro.org
13
---
9
---
14
target/arm/neon-shared.decode | 6 +++
10
target/arm/tcg/a64.decode | 6 ++++
15
target/arm/translate-neon.inc.c | 31 +++++++++++
11
target/arm/tcg/translate-a64.c | 53 +++++++++++++++++-----------------
16
target/arm/translate.c | 92 +--------------------------------
12
2 files changed, 32 insertions(+), 27 deletions(-)
17
3 files changed, 38 insertions(+), 91 deletions(-)
18
13
19
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
14
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
20
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
21
--- a/target/arm/neon-shared.decode
16
--- a/target/arm/tcg/a64.decode
22
+++ b/target/arm/neon-shared.decode
17
+++ b/target/arm/tcg/a64.decode
23
@@ -XXX,XX +XXX,XX @@ VCADD 1111 110 rot:1 1 . 0 size:1 .... .... 1000 . q:1 . 0 .... \
18
@@ -XXX,XX +XXX,XX @@ CLREX 1101 0101 0000 0011 0011 ---- 010 11111
24
# VUDOT and VSDOT
19
DSB_DMB 1101 0101 0000 0011 0011 domain:2 types:2 10- 11111
25
VDOT 1111 110 00 . 10 .... .... 1101 . q:1 . u:1 .... \
20
ISB 1101 0101 0000 0011 0011 ---- 110 11111
26
vm=%vm_dp vn=%vn_dp vd=%vd_dp
21
SB 1101 0101 0000 0011 0011 0000 111 11111
27
+
22
+
28
+# VFM[AS]L
23
+# PSTATE
29
+VFML 1111 110 0 s:1 . 10 .... .... 1000 . 0 . 1 .... \
24
+
30
+ vm=%vm_sp vn=%vn_sp vd=%vd_dp q=0
25
+CFINV 1101 0101 0000 0 000 0100 0000 000 11111
31
+VFML 1111 110 0 s:1 . 10 .... .... 1000 . 1 . 1 .... \
26
+XAFLAG 1101 0101 0000 0 000 0100 0000 001 11111
32
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp q=1
27
+AXFLAG 1101 0101 0000 0 000 0100 0000 010 11111
33
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
28
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
34
index XXXXXXX..XXXXXXX 100644
29
index XXXXXXX..XXXXXXX 100644
35
--- a/target/arm/translate-neon.inc.c
30
--- a/target/arm/tcg/translate-a64.c
36
+++ b/target/arm/translate-neon.inc.c
31
+++ b/target/arm/tcg/translate-a64.c
37
@@ -XXX,XX +XXX,XX @@ static bool trans_VDOT(DisasContext *s, arg_VDOT *a)
32
@@ -XXX,XX +XXX,XX @@ static bool trans_SB(DisasContext *s, arg_SB *a)
38
opr_sz, opr_sz, 0, fn_gvec);
39
return true;
33
return true;
40
}
34
}
35
36
-static void gen_xaflag(void)
37
+static bool trans_CFINV(DisasContext *s, arg_CFINV *a)
38
{
39
- TCGv_i32 z = tcg_temp_new_i32();
40
+ if (!dc_isar_feature(aa64_condm_4, s)) {
41
+ return false;
42
+ }
43
+ tcg_gen_xori_i32(cpu_CF, cpu_CF, 1);
44
+ return true;
45
+}
41
+
46
+
42
+static bool trans_VFML(DisasContext *s, arg_VFML *a)
47
+static bool trans_XAFLAG(DisasContext *s, arg_XAFLAG *a)
43
+{
48
+{
44
+ int opr_sz;
49
+ TCGv_i32 z;
45
+
50
+
46
+ if (!dc_isar_feature(aa32_fhm, s)) {
51
+ if (!dc_isar_feature(aa64_condm_5, s)) {
47
+ return false;
52
+ return false;
48
+ }
53
+ }
49
+
54
+
50
+ /* UNDEF accesses to D16-D31 if they don't exist. */
55
+ z = tcg_temp_new_i32();
51
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
56
52
+ (a->vd & 0x10)) {
57
tcg_gen_setcondi_i32(TCG_COND_EQ, z, cpu_ZF, 0);
58
59
@@ -XXX,XX +XXX,XX @@ static void gen_xaflag(void)
60
61
/* C | Z */
62
tcg_gen_or_i32(cpu_CF, cpu_CF, z);
63
+
64
+ return true;
65
}
66
67
-static void gen_axflag(void)
68
+static bool trans_AXFLAG(DisasContext *s, arg_AXFLAG *a)
69
{
70
+ if (!dc_isar_feature(aa64_condm_5, s)) {
53
+ return false;
71
+ return false;
54
+ }
72
+ }
55
+
73
+
56
+ if (a->vd & a->q) {
74
tcg_gen_sari_i32(cpu_VF, cpu_VF, 31); /* V ? -1 : 0 */
57
+ return false;
75
tcg_gen_andc_i32(cpu_CF, cpu_CF, cpu_VF); /* C & !V */
58
+ }
76
77
@@ -XXX,XX +XXX,XX @@ static void gen_axflag(void)
78
79
tcg_gen_movi_i32(cpu_NF, 0);
80
tcg_gen_movi_i32(cpu_VF, 0);
59
+
81
+
60
+ if (!vfp_access_check(s)) {
61
+ return true;
62
+ }
63
+
64
+ opr_sz = (1 + a->q) * 8;
65
+ tcg_gen_gvec_3_ptr(vfp_reg_offset(1, a->vd),
66
+ vfp_reg_offset(a->q, a->vn),
67
+ vfp_reg_offset(a->q, a->vm),
68
+ cpu_env, opr_sz, opr_sz, a->s, /* is_2 == 0 */
69
+ gen_helper_gvec_fmlal_a32);
70
+ return true;
82
+ return true;
71
+}
72
diff --git a/target/arm/translate.c b/target/arm/translate.c
73
index XXXXXXX..XXXXXXX 100644
74
--- a/target/arm/translate.c
75
+++ b/target/arm/translate.c
76
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
77
return 0;
78
}
83
}
79
84
80
-/* Advanced SIMD three registers of the same length extension.
85
/* MSR (immediate) - move immediate to processor state field */
81
- * 31 25 23 22 20 16 12 11 10 9 8 3 0
86
@@ -XXX,XX +XXX,XX @@ static void handle_msr_i(DisasContext *s, uint32_t insn,
82
- * +---------------+-----+---+-----+----+----+---+----+---+----+---------+----+
87
s->base.is_jmp = DISAS_TOO_MANY;
83
- * | 1 1 1 1 1 1 0 | op1 | D | op2 | Vn | Vd | 1 | o3 | 0 | o4 | N Q M U | Vm |
88
84
- * +---------------+-----+---+-----+----+----+---+----+---+----+---------+----+
89
switch (op) {
85
- */
90
- case 0x00: /* CFINV */
86
-static int disas_neon_insn_3same_ext(DisasContext *s, uint32_t insn)
91
- if (crm != 0 || !dc_isar_feature(aa64_condm_4, s)) {
87
-{
92
- goto do_unallocated;
88
- gen_helper_gvec_3 *fn_gvec = NULL;
93
- }
89
- gen_helper_gvec_3_ptr *fn_gvec_ptr = NULL;
94
- tcg_gen_xori_i32(cpu_CF, cpu_CF, 1);
90
- int rd, rn, rm, opr_sz;
95
- s->base.is_jmp = DISAS_NEXT;
91
- int data = 0;
96
- break;
92
- int off_rn, off_rm;
93
- bool is_long = false, q = extract32(insn, 6, 1);
94
- bool ptr_is_env = false;
95
-
97
-
96
- if ((insn & 0xff300f10) == 0xfc200810) {
98
- case 0x01: /* XAFlag */
97
- /* VFM[AS]L -- 1111 1100 S.10 .... .... 1000 .Q.1 .... */
99
- if (crm != 0 || !dc_isar_feature(aa64_condm_5, s)) {
98
- int is_s = extract32(insn, 23, 1);
100
- goto do_unallocated;
99
- if (!dc_isar_feature(aa32_fhm, s)) {
100
- return 1;
101
- }
101
- }
102
- is_long = true;
102
- gen_xaflag();
103
- data = is_s; /* is_2 == 0 */
103
- s->base.is_jmp = DISAS_NEXT;
104
- fn_gvec_ptr = gen_helper_gvec_fmlal_a32;
104
- break;
105
- ptr_is_env = true;
106
- } else {
107
- return 1;
108
- }
109
-
105
-
110
- VFP_DREG_D(rd, insn);
106
- case 0x02: /* AXFlag */
111
- if (rd & q) {
107
- if (crm != 0 || !dc_isar_feature(aa64_condm_5, s)) {
112
- return 1;
108
- goto do_unallocated;
113
- }
114
- if (q || !is_long) {
115
- VFP_DREG_N(rn, insn);
116
- VFP_DREG_M(rm, insn);
117
- if ((rn | rm) & q & !is_long) {
118
- return 1;
119
- }
109
- }
120
- off_rn = vfp_reg_offset(1, rn);
110
- gen_axflag();
121
- off_rm = vfp_reg_offset(1, rm);
111
- s->base.is_jmp = DISAS_NEXT;
122
- } else {
112
- break;
123
- rn = VFP_SREG_N(insn);
124
- rm = VFP_SREG_M(insn);
125
- off_rn = vfp_reg_offset(0, rn);
126
- off_rm = vfp_reg_offset(0, rm);
127
- }
128
-
113
-
129
- if (s->fp_excp_el) {
114
case 0x03: /* UAO */
130
- gen_exception_insn(s, s->pc_curr, EXCP_UDEF,
115
if (!dc_isar_feature(aa64_uao, s) || s->current_el == 0) {
131
- syn_simd_access_trap(1, 0xe, false), s->fp_excp_el);
116
goto do_unallocated;
132
- return 0;
133
- }
134
- if (!s->vfp_enabled) {
135
- return 1;
136
- }
137
-
138
- opr_sz = (1 + q) * 8;
139
- if (fn_gvec_ptr) {
140
- TCGv_ptr ptr;
141
- if (ptr_is_env) {
142
- ptr = cpu_env;
143
- } else {
144
- ptr = get_fpstatus_ptr(1);
145
- }
146
- tcg_gen_gvec_3_ptr(vfp_reg_offset(1, rd), off_rn, off_rm, ptr,
147
- opr_sz, opr_sz, data, fn_gvec_ptr);
148
- if (!ptr_is_env) {
149
- tcg_temp_free_ptr(ptr);
150
- }
151
- } else {
152
- tcg_gen_gvec_3_ool(vfp_reg_offset(1, rd), off_rn, off_rm,
153
- opr_sz, opr_sz, data, fn_gvec);
154
- }
155
- return 0;
156
-}
157
-
158
/* Advanced SIMD two registers and a scalar extension.
159
* 31 24 23 22 20 16 12 11 10 9 8 3 0
160
* +-----------------+----+---+----+----+----+---+----+---+----+---------+----+
161
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
162
}
163
}
164
}
165
- } else if ((insn & 0x0e000a00) == 0x0c000800
166
- && arm_dc_feature(s, ARM_FEATURE_V8)) {
167
- if (disas_neon_insn_3same_ext(s, insn)) {
168
- goto illegal_op;
169
- }
170
- return;
171
} else if ((insn & 0x0f000a00) == 0x0e000800
172
&& arm_dc_feature(s, ARM_FEATURE_V8)) {
173
if (disas_neon_insn_2reg_scalar_ext(s, insn)) {
174
@@ -XXX,XX +XXX,XX @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn)
175
}
176
break;
177
}
178
- if ((insn & 0xfe000a00) == 0xfc000800
179
+ if ((insn & 0xff000a00) == 0xfe000800
180
&& arm_dc_feature(s, ARM_FEATURE_V8)) {
181
/* The Thumb2 and ARM encodings are identical. */
182
- if (disas_neon_insn_3same_ext(s, insn)) {
183
- goto illegal_op;
184
- }
185
- } else if ((insn & 0xff000a00) == 0xfe000800
186
- && arm_dc_feature(s, ARM_FEATURE_V8)) {
187
- /* The Thumb2 and ARM encodings are identical. */
188
if (disas_neon_insn_2reg_scalar_ext(s, insn)) {
189
goto illegal_op;
190
}
191
--
117
--
192
2.20.1
118
2.34.1
193
194
diff view generated by jsdifflib
1
Convert the Neon comparison ops in the 3-reg-same grouping
1
Convert the MSR (immediate) insn to decodetree. Our implementation
2
to decodetree.
2
has basically no commonality between the different destinations,
3
so we decode the destination register in a64.decode.
3
4
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-18-peter.maydell@linaro.org
7
Message-id: 20230602155223.2040685-6-peter.maydell@linaro.org
7
---
8
---
8
target/arm/neon-dp.decode | 8 ++++++++
9
target/arm/tcg/a64.decode | 13 ++
9
target/arm/translate-neon.inc.c | 22 ++++++++++++++++++++++
10
target/arm/tcg/translate-a64.c | 251 ++++++++++++++++-----------------
10
target/arm/translate.c | 23 +++--------------------
11
2 files changed, 136 insertions(+), 128 deletions(-)
11
3 files changed, 33 insertions(+), 20 deletions(-)
12
12
13
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
14
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-dp.decode
15
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-dp.decode
16
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@ VBSL_3s 1111 001 1 0 . 01 .... .... 0001 ... 1 .... @3same_logic
17
@@ -XXX,XX +XXX,XX @@ SB 1101 0101 0000 0011 0011 0000 111 11111
18
VBIT_3s 1111 001 1 0 . 10 .... .... 0001 ... 1 .... @3same_logic
18
CFINV 1101 0101 0000 0 000 0100 0000 000 11111
19
VBIF_3s 1111 001 1 0 . 11 .... .... 0001 ... 1 .... @3same_logic
19
XAFLAG 1101 0101 0000 0 000 0100 0000 001 11111
20
20
AXFLAG 1101 0101 0000 0 000 0100 0000 010 11111
21
+VCGT_S_3s 1111 001 0 0 . .. .... .... 0011 . . . 0 .... @3same
21
+
22
+VCGT_U_3s 1111 001 1 0 . .. .... .... 0011 . . . 0 .... @3same
22
+# These are architecturally all "MSR (immediate)"; we decode the destination
23
+VCGE_S_3s 1111 001 0 0 . .. .... .... 0011 . . . 1 .... @3same
23
+# register too because there is no commonality in our implementation.
24
+VCGE_U_3s 1111 001 1 0 . .. .... .... 0011 . . . 1 .... @3same
24
+@msr_i .... .... .... . ... .... imm:4 ... .....
25
+
25
+MSR_i_UAO 1101 0101 0000 0 000 0100 .... 011 11111 @msr_i
26
VMAX_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 0 .... @3same
26
+MSR_i_PAN 1101 0101 0000 0 000 0100 .... 100 11111 @msr_i
27
VMAX_U_3s 1111 001 1 0 . .. .... .... 0110 . . . 0 .... @3same
27
+MSR_i_SPSEL 1101 0101 0000 0 000 0100 .... 101 11111 @msr_i
28
VMIN_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 1 .... @3same
28
+MSR_i_SBSS 1101 0101 0000 0 011 0100 .... 001 11111 @msr_i
29
@@ -XXX,XX +XXX,XX @@ VMIN_U_3s 1111 001 1 0 . .. .... .... 0110 . . . 1 .... @3same
29
+MSR_i_DIT 1101 0101 0000 0 011 0100 .... 010 11111 @msr_i
30
30
+MSR_i_TCO 1101 0101 0000 0 011 0100 .... 100 11111 @msr_i
31
VADD_3s 1111 001 0 0 . .. .... .... 1000 . . . 0 .... @3same
31
+MSR_i_DAIFSET 1101 0101 0000 0 011 0100 .... 110 11111 @msr_i
32
VSUB_3s 1111 001 1 0 . .. .... .... 1000 . . . 0 .... @3same
32
+MSR_i_DAIFCLEAR 1101 0101 0000 0 011 0100 .... 111 11111 @msr_i
33
+
33
+MSR_i_SVCR 1101 0101 0000 0 011 0100 0 mask:2 imm:1 011 11111
34
+VTST_3s 1111 001 0 0 . .. .... .... 1000 . . . 1 .... @3same
34
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
35
+VCEQ_3s 1111 001 1 0 . .. .... .... 1000 . . . 1 .... @3same
36
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
37
index XXXXXXX..XXXXXXX 100644
35
index XXXXXXX..XXXXXXX 100644
38
--- a/target/arm/translate-neon.inc.c
36
--- a/target/arm/tcg/translate-a64.c
39
+++ b/target/arm/translate-neon.inc.c
37
+++ b/target/arm/tcg/translate-a64.c
40
@@ -XXX,XX +XXX,XX @@ DO_3SAME_NO_SZ_3(VMAX_S, tcg_gen_gvec_smax)
38
@@ -XXX,XX +XXX,XX @@ static bool trans_AXFLAG(DisasContext *s, arg_AXFLAG *a)
41
DO_3SAME_NO_SZ_3(VMAX_U, tcg_gen_gvec_umax)
39
return true;
42
DO_3SAME_NO_SZ_3(VMIN_S, tcg_gen_gvec_smin)
40
}
43
DO_3SAME_NO_SZ_3(VMIN_U, tcg_gen_gvec_umin)
41
44
+
42
-/* MSR (immediate) - move immediate to processor state field */
45
+#define DO_3SAME_CMP(INSN, COND) \
43
-static void handle_msr_i(DisasContext *s, uint32_t insn,
46
+ static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
44
- unsigned int op1, unsigned int op2, unsigned int crm)
47
+ uint32_t rn_ofs, uint32_t rm_ofs, \
45
+static bool trans_MSR_i_UAO(DisasContext *s, arg_i *a)
48
+ uint32_t oprsz, uint32_t maxsz) \
46
{
49
+ { \
47
- int op = op1 << 3 | op2;
50
+ tcg_gen_gvec_cmp(COND, vece, rd_ofs, rn_ofs, rm_ofs, oprsz, maxsz); \
48
-
51
+ } \
49
- /* End the TB by default, chaining is ok. */
52
+ DO_3SAME_NO_SZ_3(INSN, gen_##INSN##_3s)
50
- s->base.is_jmp = DISAS_TOO_MANY;
53
+
51
-
54
+DO_3SAME_CMP(VCGT_S, TCG_COND_GT)
52
- switch (op) {
55
+DO_3SAME_CMP(VCGT_U, TCG_COND_GTU)
53
- case 0x03: /* UAO */
56
+DO_3SAME_CMP(VCGE_S, TCG_COND_GE)
54
- if (!dc_isar_feature(aa64_uao, s) || s->current_el == 0) {
57
+DO_3SAME_CMP(VCGE_U, TCG_COND_GEU)
55
- goto do_unallocated;
58
+DO_3SAME_CMP(VCEQ, TCG_COND_EQ)
56
- }
59
+
57
- if (crm & 1) {
60
+static void gen_VTST_3s(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
58
- set_pstate_bits(PSTATE_UAO);
61
+ uint32_t rm_ofs, uint32_t oprsz, uint32_t maxsz)
59
- } else {
62
+{
60
- clear_pstate_bits(PSTATE_UAO);
63
+ tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, oprsz, maxsz, &cmtst_op[vece]);
61
- }
64
+}
62
- gen_rebuild_hflags(s);
65
+DO_3SAME_NO_SZ_3(VTST, gen_VTST_3s)
63
- break;
66
diff --git a/target/arm/translate.c b/target/arm/translate.c
64
-
67
index XXXXXXX..XXXXXXX 100644
65
- case 0x04: /* PAN */
68
--- a/target/arm/translate.c
66
- if (!dc_isar_feature(aa64_pan, s) || s->current_el == 0) {
69
+++ b/target/arm/translate.c
67
- goto do_unallocated;
70
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
68
- }
71
u ? &mls_op[size] : &mla_op[size]);
69
- if (crm & 1) {
72
return 0;
70
- set_pstate_bits(PSTATE_PAN);
73
71
- } else {
74
- case NEON_3R_VTST_VCEQ:
72
- clear_pstate_bits(PSTATE_PAN);
75
- if (u) { /* VCEQ */
73
- }
76
- tcg_gen_gvec_cmp(TCG_COND_EQ, size, rd_ofs, rn_ofs, rm_ofs,
74
- gen_rebuild_hflags(s);
77
- vec_size, vec_size);
75
- break;
78
- } else { /* VTST */
76
-
79
- tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs,
77
- case 0x05: /* SPSel */
80
- vec_size, vec_size, &cmtst_op[size]);
78
- if (s->current_el == 0) {
79
- goto do_unallocated;
80
- }
81
- gen_helper_msr_i_spsel(cpu_env, tcg_constant_i32(crm & PSTATE_SP));
82
- break;
83
-
84
- case 0x19: /* SSBS */
85
- if (!dc_isar_feature(aa64_ssbs, s)) {
86
- goto do_unallocated;
87
- }
88
- if (crm & 1) {
89
- set_pstate_bits(PSTATE_SSBS);
90
- } else {
91
- clear_pstate_bits(PSTATE_SSBS);
92
- }
93
- /* Don't need to rebuild hflags since SSBS is a nop */
94
- break;
95
-
96
- case 0x1a: /* DIT */
97
- if (!dc_isar_feature(aa64_dit, s)) {
98
- goto do_unallocated;
99
- }
100
- if (crm & 1) {
101
- set_pstate_bits(PSTATE_DIT);
102
- } else {
103
- clear_pstate_bits(PSTATE_DIT);
104
- }
105
- /* There's no need to rebuild hflags because DIT is a nop */
106
- break;
107
-
108
- case 0x1e: /* DAIFSet */
109
- gen_helper_msr_i_daifset(cpu_env, tcg_constant_i32(crm));
110
- break;
111
-
112
- case 0x1f: /* DAIFClear */
113
- gen_helper_msr_i_daifclear(cpu_env, tcg_constant_i32(crm));
114
- /* For DAIFClear, exit the cpu loop to re-evaluate pending IRQs. */
115
- s->base.is_jmp = DISAS_UPDATE_EXIT;
116
- break;
117
-
118
- case 0x1c: /* TCO */
119
- if (dc_isar_feature(aa64_mte, s)) {
120
- /* Full MTE is enabled -- set the TCO bit as directed. */
121
- if (crm & 1) {
122
- set_pstate_bits(PSTATE_TCO);
123
- } else {
124
- clear_pstate_bits(PSTATE_TCO);
81
- }
125
- }
82
- return 0;
126
- gen_rebuild_hflags(s);
83
-
127
- /* Many factors, including TCO, go into MTE_ACTIVE. */
84
- case NEON_3R_VCGT:
128
- s->base.is_jmp = DISAS_UPDATE_NOCHAIN;
85
- tcg_gen_gvec_cmp(u ? TCG_COND_GTU : TCG_COND_GT, size,
129
- } else if (dc_isar_feature(aa64_mte_insn_reg, s)) {
86
- rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size);
130
- /* Only "instructions accessible at EL0" -- PSTATE.TCO is WI. */
87
- return 0;
131
- s->base.is_jmp = DISAS_NEXT;
88
-
132
- } else {
89
- case NEON_3R_VCGE:
133
- goto do_unallocated;
90
- tcg_gen_gvec_cmp(u ? TCG_COND_GEU : TCG_COND_GE, size,
134
- }
91
- rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size);
135
- break;
92
- return 0;
136
-
93
-
137
- case 0x1b: /* SVCR* */
94
case NEON_3R_VSHL:
138
- if (!dc_isar_feature(aa64_sme, s) || crm < 2 || crm > 7) {
95
/* Note the operation is vshl vd,vm,vn */
139
- goto do_unallocated;
96
tcg_gen_gvec_3(rd_ofs, rm_ofs, rn_ofs, vec_size, vec_size,
140
- }
97
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
141
- if (sme_access_check(s)) {
98
case NEON_3R_LOGIC:
142
- int old = s->pstate_sm | (s->pstate_za << 1);
99
case NEON_3R_VMAX:
143
- int new = (crm & 1) * 3;
100
case NEON_3R_VMIN:
144
- int msk = (crm >> 1) & 3;
101
+ case NEON_3R_VTST_VCEQ:
145
-
102
+ case NEON_3R_VCGT:
146
- if ((old ^ new) & msk) {
103
+ case NEON_3R_VCGE:
147
- /* At least one bit changes. */
104
/* Already handled by decodetree */
148
- gen_helper_set_svcr(cpu_env, tcg_constant_i32(new),
105
return 1;
149
- tcg_constant_i32(msk));
106
}
150
- } else {
151
- s->base.is_jmp = DISAS_NEXT;
152
- }
153
- }
154
- break;
155
-
156
- default:
157
- do_unallocated:
158
- unallocated_encoding(s);
159
- return;
160
+ if (!dc_isar_feature(aa64_uao, s) || s->current_el == 0) {
161
+ return false;
162
}
163
+ if (a->imm & 1) {
164
+ set_pstate_bits(PSTATE_UAO);
165
+ } else {
166
+ clear_pstate_bits(PSTATE_UAO);
167
+ }
168
+ gen_rebuild_hflags(s);
169
+ s->base.is_jmp = DISAS_TOO_MANY;
170
+ return true;
171
+}
172
+
173
+static bool trans_MSR_i_PAN(DisasContext *s, arg_i *a)
174
+{
175
+ if (!dc_isar_feature(aa64_pan, s) || s->current_el == 0) {
176
+ return false;
177
+ }
178
+ if (a->imm & 1) {
179
+ set_pstate_bits(PSTATE_PAN);
180
+ } else {
181
+ clear_pstate_bits(PSTATE_PAN);
182
+ }
183
+ gen_rebuild_hflags(s);
184
+ s->base.is_jmp = DISAS_TOO_MANY;
185
+ return true;
186
+}
187
+
188
+static bool trans_MSR_i_SPSEL(DisasContext *s, arg_i *a)
189
+{
190
+ if (s->current_el == 0) {
191
+ return false;
192
+ }
193
+ gen_helper_msr_i_spsel(cpu_env, tcg_constant_i32(a->imm & PSTATE_SP));
194
+ s->base.is_jmp = DISAS_TOO_MANY;
195
+ return true;
196
+}
197
+
198
+static bool trans_MSR_i_SBSS(DisasContext *s, arg_i *a)
199
+{
200
+ if (!dc_isar_feature(aa64_ssbs, s)) {
201
+ return false;
202
+ }
203
+ if (a->imm & 1) {
204
+ set_pstate_bits(PSTATE_SSBS);
205
+ } else {
206
+ clear_pstate_bits(PSTATE_SSBS);
207
+ }
208
+ /* Don't need to rebuild hflags since SSBS is a nop */
209
+ s->base.is_jmp = DISAS_TOO_MANY;
210
+ return true;
211
+}
212
+
213
+static bool trans_MSR_i_DIT(DisasContext *s, arg_i *a)
214
+{
215
+ if (!dc_isar_feature(aa64_dit, s)) {
216
+ return false;
217
+ }
218
+ if (a->imm & 1) {
219
+ set_pstate_bits(PSTATE_DIT);
220
+ } else {
221
+ clear_pstate_bits(PSTATE_DIT);
222
+ }
223
+ /* There's no need to rebuild hflags because DIT is a nop */
224
+ s->base.is_jmp = DISAS_TOO_MANY;
225
+ return true;
226
+}
227
+
228
+static bool trans_MSR_i_TCO(DisasContext *s, arg_i *a)
229
+{
230
+ if (dc_isar_feature(aa64_mte, s)) {
231
+ /* Full MTE is enabled -- set the TCO bit as directed. */
232
+ if (a->imm & 1) {
233
+ set_pstate_bits(PSTATE_TCO);
234
+ } else {
235
+ clear_pstate_bits(PSTATE_TCO);
236
+ }
237
+ gen_rebuild_hflags(s);
238
+ /* Many factors, including TCO, go into MTE_ACTIVE. */
239
+ s->base.is_jmp = DISAS_UPDATE_NOCHAIN;
240
+ return true;
241
+ } else if (dc_isar_feature(aa64_mte_insn_reg, s)) {
242
+ /* Only "instructions accessible at EL0" -- PSTATE.TCO is WI. */
243
+ return true;
244
+ } else {
245
+ /* Insn not present */
246
+ return false;
247
+ }
248
+}
249
+
250
+static bool trans_MSR_i_DAIFSET(DisasContext *s, arg_i *a)
251
+{
252
+ gen_helper_msr_i_daifset(cpu_env, tcg_constant_i32(a->imm));
253
+ s->base.is_jmp = DISAS_TOO_MANY;
254
+ return true;
255
+}
256
+
257
+static bool trans_MSR_i_DAIFCLEAR(DisasContext *s, arg_i *a)
258
+{
259
+ gen_helper_msr_i_daifclear(cpu_env, tcg_constant_i32(a->imm));
260
+ /* Exit the cpu loop to re-evaluate pending IRQs. */
261
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
262
+ return true;
263
+}
264
+
265
+static bool trans_MSR_i_SVCR(DisasContext *s, arg_MSR_i_SVCR *a)
266
+{
267
+ if (!dc_isar_feature(aa64_sme, s) || a->mask == 0) {
268
+ return false;
269
+ }
270
+ if (sme_access_check(s)) {
271
+ int old = s->pstate_sm | (s->pstate_za << 1);
272
+ int new = a->imm * 3;
273
+
274
+ if ((old ^ new) & a->mask) {
275
+ /* At least one bit changes. */
276
+ gen_helper_set_svcr(cpu_env, tcg_constant_i32(new),
277
+ tcg_constant_i32(a->mask));
278
+ s->base.is_jmp = DISAS_TOO_MANY;
279
+ }
280
+ }
281
+ return true;
282
}
283
284
static void gen_get_nzcv(TCGv_i64 tcg_rt)
285
@@ -XXX,XX +XXX,XX @@ static void disas_system(DisasContext *s, uint32_t insn)
286
rt = extract32(insn, 0, 5);
287
288
if (op0 == 0) {
289
- if (l || rt != 31) {
290
- unallocated_encoding(s);
291
- return;
292
- }
293
- switch (crn) {
294
- case 4: /* MSR (immediate) */
295
- handle_msr_i(s, insn, op1, op2, crm);
296
- break;
297
- default:
298
- unallocated_encoding(s);
299
- break;
300
- }
301
+ unallocated_encoding(s);
302
return;
303
}
304
handle_sys(s, insn, l, op0, op1, op2, crn, crm, rt);
107
--
305
--
108
2.20.1
306
2.34.1
109
110
diff view generated by jsdifflib
1
Convert the Neon logic ops in the 3-reg-same grouping to decodetree.
1
Convert MSR (reg), MRS, SYS, SYSL to decodetree. For QEMU these are
2
Note that for the logic ops the 'size' field forms part of their
2
all essentially the same instruction (system register access).
3
decode and the actual operations are always bitwise.
4
3
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200430181003.21682-16-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-7-peter.maydell@linaro.org
7
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
8
---
8
---
9
target/arm/neon-dp.decode | 12 +++++++++++
9
target/arm/tcg/a64.decode | 8 ++++++++
10
target/arm/translate-neon.inc.c | 19 +++++++++++++++++
10
target/arm/tcg/translate-a64.c | 32 +++++---------------------------
11
target/arm/translate.c | 38 +--------------------------------
11
2 files changed, 13 insertions(+), 27 deletions(-)
12
3 files changed, 32 insertions(+), 37 deletions(-)
13
12
14
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
15
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/neon-dp.decode
15
--- a/target/arm/tcg/a64.decode
17
+++ b/target/arm/neon-dp.decode
16
+++ b/target/arm/tcg/a64.decode
18
@@ -XXX,XX +XXX,XX @@
17
@@ -XXX,XX +XXX,XX @@ MSR_i_TCO 1101 0101 0000 0 011 0100 .... 100 11111 @msr_i
19
@3same .... ... . . . size:2 .... .... .... . q:1 . . .... \
18
MSR_i_DAIFSET 1101 0101 0000 0 011 0100 .... 110 11111 @msr_i
20
&3same vm=%vm_dp vn=%vn_dp vd=%vd_dp
19
MSR_i_DAIFCLEAR 1101 0101 0000 0 011 0100 .... 111 11111 @msr_i
21
20
MSR_i_SVCR 1101 0101 0000 0 011 0100 0 mask:2 imm:1 011 11111
22
+@3same_logic .... ... . . . .. .... .... .... . q:1 .. .... \
23
+ &3same vm=%vm_dp vn=%vn_dp vd=%vd_dp size=0
24
+
21
+
25
+VAND_3s 1111 001 0 0 . 00 .... .... 0001 ... 1 .... @3same_logic
22
+# MRS, MSR (register), SYS, SYSL. These are all essentially the
26
+VBIC_3s 1111 001 0 0 . 01 .... .... 0001 ... 1 .... @3same_logic
23
+# same instruction as far as QEMU is concerned.
27
+VORR_3s 1111 001 0 0 . 10 .... .... 0001 ... 1 .... @3same_logic
24
+# NB: op0 is bits [20:19], but op0=0b00 is other insns, so we have
28
+VORN_3s 1111 001 0 0 . 11 .... .... 0001 ... 1 .... @3same_logic
25
+# to hand-decode it.
29
+VEOR_3s 1111 001 1 0 . 00 .... .... 0001 ... 1 .... @3same_logic
26
+SYS 1101 0101 00 l:1 01 op1:3 crn:4 crm:4 op2:3 rt:5 op0=1
30
+VBSL_3s 1111 001 1 0 . 01 .... .... 0001 ... 1 .... @3same_logic
27
+SYS 1101 0101 00 l:1 10 op1:3 crn:4 crm:4 op2:3 rt:5 op0=2
31
+VBIT_3s 1111 001 1 0 . 10 .... .... 0001 ... 1 .... @3same_logic
28
+SYS 1101 0101 00 l:1 11 op1:3 crn:4 crm:4 op2:3 rt:5 op0=3
32
+VBIF_3s 1111 001 1 0 . 11 .... .... 0001 ... 1 .... @3same_logic
29
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
33
+
34
VADD_3s 1111 001 0 0 . .. .... .... 1000 . . . 0 .... @3same
35
VSUB_3s 1111 001 1 0 . .. .... .... 1000 . . . 0 .... @3same
36
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
37
index XXXXXXX..XXXXXXX 100644
30
index XXXXXXX..XXXXXXX 100644
38
--- a/target/arm/translate-neon.inc.c
31
--- a/target/arm/tcg/translate-a64.c
39
+++ b/target/arm/translate-neon.inc.c
32
+++ b/target/arm/tcg/translate-a64.c
40
@@ -XXX,XX +XXX,XX @@ static bool do_3same(DisasContext *s, arg_3same *a, GVecGen3Fn fn)
33
@@ -XXX,XX +XXX,XX @@ static void gen_sysreg_undef(DisasContext *s, bool isread,
41
34
* These are all essentially the same insn in 'read' and 'write'
42
DO_3SAME(VADD, tcg_gen_gvec_add)
35
* versions, with varying op0 fields.
43
DO_3SAME(VSUB, tcg_gen_gvec_sub)
36
*/
44
+DO_3SAME(VAND, tcg_gen_gvec_and)
37
-static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
45
+DO_3SAME(VBIC, tcg_gen_gvec_andc)
38
+static void handle_sys(DisasContext *s, bool isread,
46
+DO_3SAME(VORR, tcg_gen_gvec_or)
39
unsigned int op0, unsigned int op1, unsigned int op2,
47
+DO_3SAME(VORN, tcg_gen_gvec_orc)
40
unsigned int crn, unsigned int crm, unsigned int rt)
48
+DO_3SAME(VEOR, tcg_gen_gvec_xor)
41
{
49
+
42
@@ -XXX,XX +XXX,XX @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
50
+/* These insns are all gvec_bitsel but with the inputs in various orders. */
43
}
51
+#define DO_3SAME_BITSEL(INSN, O1, O2, O3) \
44
}
52
+ static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
45
53
+ uint32_t rn_ofs, uint32_t rm_ofs, \
46
-/* System
54
+ uint32_t oprsz, uint32_t maxsz) \
47
- * 31 22 21 20 19 18 16 15 12 11 8 7 5 4 0
55
+ { \
48
- * +---------------------+---+-----+-----+-------+-------+-----+------+
56
+ tcg_gen_gvec_bitsel(vece, rd_ofs, O1, O2, O3, oprsz, maxsz); \
49
- * | 1 1 0 1 0 1 0 1 0 0 | L | op0 | op1 | CRn | CRm | op2 | Rt |
57
+ } \
50
- * +---------------------+---+-----+-----+-------+-------+-----+------+
58
+ DO_3SAME(INSN, gen_##INSN##_3s)
51
- */
59
+
52
-static void disas_system(DisasContext *s, uint32_t insn)
60
+DO_3SAME_BITSEL(VBSL, rd_ofs, rn_ofs, rm_ofs)
53
+static bool trans_SYS(DisasContext *s, arg_SYS *a)
61
+DO_3SAME_BITSEL(VBIT, rm_ofs, rn_ofs, rd_ofs)
54
{
62
+DO_3SAME_BITSEL(VBIF, rm_ofs, rd_ofs, rn_ofs)
55
- unsigned int l, op0, op1, crn, crm, op2, rt;
63
diff --git a/target/arm/translate.c b/target/arm/translate.c
56
- l = extract32(insn, 21, 1);
64
index XXXXXXX..XXXXXXX 100644
57
- op0 = extract32(insn, 19, 2);
65
--- a/target/arm/translate.c
58
- op1 = extract32(insn, 16, 3);
66
+++ b/target/arm/translate.c
59
- crn = extract32(insn, 12, 4);
67
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
60
- crm = extract32(insn, 8, 4);
68
}
61
- op2 = extract32(insn, 5, 3);
69
return 1;
62
- rt = extract32(insn, 0, 5);
70
63
-
71
- case NEON_3R_LOGIC: /* Logic ops. */
64
- if (op0 == 0) {
72
- switch ((u << 2) | size) {
65
- unallocated_encoding(s);
73
- case 0: /* VAND */
66
- return;
74
- tcg_gen_gvec_and(0, rd_ofs, rn_ofs, rm_ofs,
67
- }
75
- vec_size, vec_size);
68
- handle_sys(s, insn, l, op0, op1, op2, crn, crm, rt);
76
- break;
69
+ handle_sys(s, a->l, a->op0, a->op1, a->op2, a->crn, a->crm, a->rt);
77
- case 1: /* VBIC */
70
+ return true;
78
- tcg_gen_gvec_andc(0, rd_ofs, rn_ofs, rm_ofs,
71
}
79
- vec_size, vec_size);
72
80
- break;
73
/* Exception generation
81
- case 2: /* VORR */
74
@@ -XXX,XX +XXX,XX @@ static void disas_b_exc_sys(DisasContext *s, uint32_t insn)
82
- tcg_gen_gvec_or(0, rd_ofs, rn_ofs, rm_ofs,
75
switch (extract32(insn, 25, 7)) {
83
- vec_size, vec_size);
76
case 0x6a: /* Exception generation / System */
84
- break;
77
if (insn & (1 << 24)) {
85
- case 3: /* VORN */
78
- if (extract32(insn, 22, 2) == 0) {
86
- tcg_gen_gvec_orc(0, rd_ofs, rn_ofs, rm_ofs,
79
- disas_system(s, insn);
87
- vec_size, vec_size);
80
- } else {
88
- break;
81
- unallocated_encoding(s);
89
- case 4: /* VEOR */
90
- tcg_gen_gvec_xor(0, rd_ofs, rn_ofs, rm_ofs,
91
- vec_size, vec_size);
92
- break;
93
- case 5: /* VBSL */
94
- tcg_gen_gvec_bitsel(MO_8, rd_ofs, rd_ofs, rn_ofs, rm_ofs,
95
- vec_size, vec_size);
96
- break;
97
- case 6: /* VBIT */
98
- tcg_gen_gvec_bitsel(MO_8, rd_ofs, rm_ofs, rn_ofs, rd_ofs,
99
- vec_size, vec_size);
100
- break;
101
- case 7: /* VBIF */
102
- tcg_gen_gvec_bitsel(MO_8, rd_ofs, rm_ofs, rd_ofs, rn_ofs,
103
- vec_size, vec_size);
104
- break;
105
- }
82
- }
106
- return 0;
83
+ unallocated_encoding(s);
107
-
84
} else {
108
case NEON_3R_VQADD:
85
disas_exc(s, insn);
109
tcg_gen_gvec_4(rd_ofs, offsetof(CPUARMState, vfp.qc),
110
rn_ofs, rm_ofs, vec_size, vec_size,
111
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
112
return 0;
113
114
case NEON_3R_VADD_VSUB:
115
+ case NEON_3R_LOGIC:
116
/* Already handled by decodetree */
117
return 1;
118
}
86
}
119
--
87
--
120
2.20.1
88
2.34.1
121
89
122
90
diff view generated by jsdifflib
1
Convert the Neon 3-reg-same VADD and VSUB insns to decodetree.
1
Convert the exception generation instructions SVC, HVC, SMC, BRK and
2
HLT to decodetree.
2
3
3
Note that we don't need the neon_3r_sizes[op] check here because all
4
The old decoder decoded the halting-debug insnns DCPS1, DCPS2 and
4
size values are OK for VADD and VSUB; we'll add this when we convert
5
DCPS3 just in order to then make them UNDEF; as with DRPS, we don't
5
the first insn that has size restrictions.
6
bother to decode them, but document the patterns in a64.decode.
6
7
For this we need one of the GVecGen*Fn typedefs currently in
8
translate-a64.h; move them all to translate.h as a block so they
9
are visible to the 32-bit decoder.
10
7
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
13
Message-id: 20200430181003.21682-15-peter.maydell@linaro.org
10
Message-id: 20230602155223.2040685-8-peter.maydell@linaro.org
14
---
11
---
15
target/arm/translate-a64.h | 9 --------
12
target/arm/tcg/a64.decode | 15 +++
16
target/arm/translate.h | 9 ++++++++
13
target/arm/tcg/translate-a64.c | 173 ++++++++++++---------------------
17
target/arm/neon-dp.decode | 17 +++++++++++++++
14
2 files changed, 79 insertions(+), 109 deletions(-)
18
target/arm/translate-neon.inc.c | 38 +++++++++++++++++++++++++++++++++
19
target/arm/translate.c | 14 ++++--------
20
5 files changed, 68 insertions(+), 19 deletions(-)
21
15
22
diff --git a/target/arm/translate-a64.h b/target/arm/translate-a64.h
16
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
23
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
24
--- a/target/arm/translate-a64.h
18
--- a/target/arm/tcg/a64.decode
25
+++ b/target/arm/translate-a64.h
19
+++ b/target/arm/tcg/a64.decode
26
@@ -XXX,XX +XXX,XX @@ static inline int vec_full_reg_size(DisasContext *s)
20
@@ -XXX,XX +XXX,XX @@ MSR_i_SVCR 1101 0101 0000 0 011 0100 0 mask:2 imm:1 011 11111
27
21
SYS 1101 0101 00 l:1 01 op1:3 crn:4 crm:4 op2:3 rt:5 op0=1
28
bool disas_sve(DisasContext *, uint32_t);
22
SYS 1101 0101 00 l:1 10 op1:3 crn:4 crm:4 op2:3 rt:5 op0=2
29
23
SYS 1101 0101 00 l:1 11 op1:3 crn:4 crm:4 op2:3 rt:5 op0=3
30
-/* Note that the gvec expanders operate on offsets + sizes. */
24
+
31
-typedef void GVecGen2Fn(unsigned, uint32_t, uint32_t, uint32_t, uint32_t);
25
+# Exception generation
32
-typedef void GVecGen2iFn(unsigned, uint32_t, uint32_t, int64_t,
26
+
33
- uint32_t, uint32_t);
27
+@i16 .... .... ... imm:16 ... .. &i
34
-typedef void GVecGen3Fn(unsigned, uint32_t, uint32_t,
28
+SVC 1101 0100 000 ................ 000 01 @i16
35
- uint32_t, uint32_t, uint32_t);
29
+HVC 1101 0100 000 ................ 000 10 @i16
36
-typedef void GVecGen4Fn(unsigned, uint32_t, uint32_t, uint32_t,
30
+SMC 1101 0100 000 ................ 000 11 @i16
37
- uint32_t, uint32_t, uint32_t);
31
+BRK 1101 0100 001 ................ 000 00 @i16
38
-
32
+HLT 1101 0100 010 ................ 000 00 @i16
39
#endif /* TARGET_ARM_TRANSLATE_A64_H */
33
+# These insns always UNDEF unless in halting debug state, which
40
diff --git a/target/arm/translate.h b/target/arm/translate.h
34
+# we don't implement. So we don't need to decode them. The patterns
35
+# are listed here as documentation.
36
+# DCPS1 1101 0100 101 ................ 000 01 @i16
37
+# DCPS2 1101 0100 101 ................ 000 10 @i16
38
+# DCPS3 1101 0100 101 ................ 000 11 @i16
39
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
41
index XXXXXXX..XXXXXXX 100644
40
index XXXXXXX..XXXXXXX 100644
42
--- a/target/arm/translate.h
41
--- a/target/arm/tcg/translate-a64.c
43
+++ b/target/arm/translate.h
42
+++ b/target/arm/tcg/translate-a64.c
44
@@ -XXX,XX +XXX,XX @@ void gen_sshl_i64(TCGv_i64 d, TCGv_i64 a, TCGv_i64 b);
43
@@ -XXX,XX +XXX,XX @@ static bool trans_SYS(DisasContext *s, arg_SYS *a)
45
#define dc_isar_feature(name, ctx) \
46
({ DisasContext *ctx_ = (ctx); isar_feature_##name(ctx_->isar); })
47
48
+/* Note that the gvec expanders operate on offsets + sizes. */
49
+typedef void GVecGen2Fn(unsigned, uint32_t, uint32_t, uint32_t, uint32_t);
50
+typedef void GVecGen2iFn(unsigned, uint32_t, uint32_t, int64_t,
51
+ uint32_t, uint32_t);
52
+typedef void GVecGen3Fn(unsigned, uint32_t, uint32_t,
53
+ uint32_t, uint32_t, uint32_t);
54
+typedef void GVecGen4Fn(unsigned, uint32_t, uint32_t, uint32_t,
55
+ uint32_t, uint32_t, uint32_t);
56
+
57
#endif /* TARGET_ARM_TRANSLATE_H */
58
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
59
index XXXXXXX..XXXXXXX 100644
60
--- a/target/arm/neon-dp.decode
61
+++ b/target/arm/neon-dp.decode
62
@@ -XXX,XX +XXX,XX @@
63
#
64
# This file is processed by scripts/decodetree.py
65
#
66
+# VFP/Neon register fields; same as vfp.decode
67
+%vm_dp 5:1 0:4
68
+%vn_dp 7:1 16:4
69
+%vd_dp 22:1 12:4
70
71
# Encodings for Neon data processing instructions where the T32 encoding
72
# is a simple transformation of the A32 encoding.
73
@@ -XXX,XX +XXX,XX @@
74
# 0b111p_1111_qqqq_qqqq_qqqq_qqqq_qqqq_qqqq
75
# This file works on the A32 encoding only; calling code for T32 has to
76
# transform the insn into the A32 version first.
77
+
78
+######################################################################
79
+# 3-reg-same grouping:
80
+# 1111 001 U 0 D sz:2 Vn:4 Vd:4 opc:4 N Q M op Vm:4
81
+######################################################################
82
+
83
+&3same vm vn vd q size
84
+
85
+@3same .... ... . . . size:2 .... .... .... . q:1 . . .... \
86
+ &3same vm=%vm_dp vn=%vn_dp vd=%vd_dp
87
+
88
+VADD_3s 1111 001 0 0 . .. .... .... 1000 . . . 0 .... @3same
89
+VSUB_3s 1111 001 1 0 . .. .... .... 1000 . . . 0 .... @3same
90
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
91
index XXXXXXX..XXXXXXX 100644
92
--- a/target/arm/translate-neon.inc.c
93
+++ b/target/arm/translate-neon.inc.c
94
@@ -XXX,XX +XXX,XX @@ static bool trans_VLDST_single(DisasContext *s, arg_VLDST_single *a)
95
96
return true;
44
return true;
97
}
45
}
98
+
46
99
+static bool do_3same(DisasContext *s, arg_3same *a, GVecGen3Fn fn)
47
-/* Exception generation
48
- *
49
- * 31 24 23 21 20 5 4 2 1 0
50
- * +-----------------+-----+------------------------+-----+----+
51
- * | 1 1 0 1 0 1 0 0 | opc | imm16 | op2 | LL |
52
- * +-----------------------+------------------------+----------+
53
- */
54
-static void disas_exc(DisasContext *s, uint32_t insn)
55
+static bool trans_SVC(DisasContext *s, arg_i *a)
56
{
57
- int opc = extract32(insn, 21, 3);
58
- int op2_ll = extract32(insn, 0, 5);
59
- int imm16 = extract32(insn, 5, 16);
60
- uint32_t syndrome;
61
-
62
- switch (opc) {
63
- case 0:
64
- /* For SVC, HVC and SMC we advance the single-step state
65
- * machine before taking the exception. This is architecturally
66
- * mandated, to ensure that single-stepping a system call
67
- * instruction works properly.
68
- */
69
- switch (op2_ll) {
70
- case 1: /* SVC */
71
- syndrome = syn_aa64_svc(imm16);
72
- if (s->fgt_svc) {
73
- gen_exception_insn_el(s, 0, EXCP_UDEF, syndrome, 2);
74
- break;
75
- }
76
- gen_ss_advance(s);
77
- gen_exception_insn(s, 4, EXCP_SWI, syndrome);
78
- break;
79
- case 2: /* HVC */
80
- if (s->current_el == 0) {
81
- unallocated_encoding(s);
82
- break;
83
- }
84
- /* The pre HVC helper handles cases when HVC gets trapped
85
- * as an undefined insn by runtime configuration.
86
- */
87
- gen_a64_update_pc(s, 0);
88
- gen_helper_pre_hvc(cpu_env);
89
- gen_ss_advance(s);
90
- gen_exception_insn_el(s, 4, EXCP_HVC, syn_aa64_hvc(imm16), 2);
91
- break;
92
- case 3: /* SMC */
93
- if (s->current_el == 0) {
94
- unallocated_encoding(s);
95
- break;
96
- }
97
- gen_a64_update_pc(s, 0);
98
- gen_helper_pre_smc(cpu_env, tcg_constant_i32(syn_aa64_smc(imm16)));
99
- gen_ss_advance(s);
100
- gen_exception_insn_el(s, 4, EXCP_SMC, syn_aa64_smc(imm16), 3);
101
- break;
102
- default:
103
- unallocated_encoding(s);
104
- break;
105
- }
106
- break;
107
- case 1:
108
- if (op2_ll != 0) {
109
- unallocated_encoding(s);
110
- break;
111
- }
112
- /* BRK */
113
- gen_exception_bkpt_insn(s, syn_aa64_bkpt(imm16));
114
- break;
115
- case 2:
116
- if (op2_ll != 0) {
117
- unallocated_encoding(s);
118
- break;
119
- }
120
- /* HLT. This has two purposes.
121
- * Architecturally, it is an external halting debug instruction.
122
- * Since QEMU doesn't implement external debug, we treat this as
123
- * it is required for halting debug disabled: it will UNDEF.
124
- * Secondly, "HLT 0xf000" is the A64 semihosting syscall instruction.
125
- */
126
- if (semihosting_enabled(s->current_el == 0) && imm16 == 0xf000) {
127
- gen_exception_internal_insn(s, EXCP_SEMIHOST);
128
- } else {
129
- unallocated_encoding(s);
130
- }
131
- break;
132
- case 5:
133
- if (op2_ll < 1 || op2_ll > 3) {
134
- unallocated_encoding(s);
135
- break;
136
- }
137
- /* DCPS1, DCPS2, DCPS3 */
138
- unallocated_encoding(s);
139
- break;
140
- default:
141
- unallocated_encoding(s);
142
- break;
143
+ /*
144
+ * For SVC, HVC and SMC we advance the single-step state
145
+ * machine before taking the exception. This is architecturally
146
+ * mandated, to ensure that single-stepping a system call
147
+ * instruction works properly.
148
+ */
149
+ uint32_t syndrome = syn_aa64_svc(a->imm);
150
+ if (s->fgt_svc) {
151
+ gen_exception_insn_el(s, 0, EXCP_UDEF, syndrome, 2);
152
+ return true;
153
}
154
+ gen_ss_advance(s);
155
+ gen_exception_insn(s, 4, EXCP_SWI, syndrome);
156
+ return true;
157
}
158
159
-/* Branches, exception generating and system instructions */
160
-static void disas_b_exc_sys(DisasContext *s, uint32_t insn)
161
+static bool trans_HVC(DisasContext *s, arg_i *a)
162
{
163
- switch (extract32(insn, 25, 7)) {
164
- case 0x6a: /* Exception generation / System */
165
- if (insn & (1 << 24)) {
166
- unallocated_encoding(s);
167
- } else {
168
- disas_exc(s, insn);
169
- }
170
- break;
171
- default:
172
+ if (s->current_el == 0) {
173
unallocated_encoding(s);
174
- break;
175
+ return true;
176
}
177
+ /*
178
+ * The pre HVC helper handles cases when HVC gets trapped
179
+ * as an undefined insn by runtime configuration.
180
+ */
181
+ gen_a64_update_pc(s, 0);
182
+ gen_helper_pre_hvc(cpu_env);
183
+ /* Architecture requires ss advance before we do the actual work */
184
+ gen_ss_advance(s);
185
+ gen_exception_insn_el(s, 4, EXCP_HVC, syn_aa64_hvc(a->imm), 2);
186
+ return true;
187
+}
188
+
189
+static bool trans_SMC(DisasContext *s, arg_i *a)
100
+{
190
+{
101
+ int vec_size = a->q ? 16 : 8;
191
+ if (s->current_el == 0) {
102
+ int rd_ofs = neon_reg_offset(a->vd, 0);
192
+ unallocated_encoding(s);
103
+ int rn_ofs = neon_reg_offset(a->vn, 0);
104
+ int rm_ofs = neon_reg_offset(a->vm, 0);
105
+
106
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
107
+ return false;
108
+ }
109
+
110
+ /* UNDEF accesses to D16-D31 if they don't exist. */
111
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
112
+ ((a->vd | a->vn | a->vm) & 0x10)) {
113
+ return false;
114
+ }
115
+
116
+ if ((a->vn | a->vm | a->vd) & a->q) {
117
+ return false;
118
+ }
119
+
120
+ if (!vfp_access_check(s)) {
121
+ return true;
193
+ return true;
122
+ }
194
+ }
123
+
195
+ gen_a64_update_pc(s, 0);
124
+ fn(a->size, rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size);
196
+ gen_helper_pre_smc(cpu_env, tcg_constant_i32(syn_aa64_smc(a->imm)));
197
+ /* Architecture requires ss advance before we do the actual work */
198
+ gen_ss_advance(s);
199
+ gen_exception_insn_el(s, 4, EXCP_SMC, syn_aa64_smc(a->imm), 3);
125
+ return true;
200
+ return true;
126
+}
201
+}
127
+
202
+
128
+#define DO_3SAME(INSN, FUNC) \
203
+static bool trans_BRK(DisasContext *s, arg_i *a)
129
+ static bool trans_##INSN##_3s(DisasContext *s, arg_3same *a) \
204
+{
130
+ { \
205
+ gen_exception_bkpt_insn(s, syn_aa64_bkpt(a->imm));
131
+ return do_3same(s, a, FUNC); \
206
+ return true;
207
+}
208
+
209
+static bool trans_HLT(DisasContext *s, arg_i *a)
210
+{
211
+ /*
212
+ * HLT. This has two purposes.
213
+ * Architecturally, it is an external halting debug instruction.
214
+ * Since QEMU doesn't implement external debug, we treat this as
215
+ * it is required for halting debug disabled: it will UNDEF.
216
+ * Secondly, "HLT 0xf000" is the A64 semihosting syscall instruction.
217
+ */
218
+ if (semihosting_enabled(s->current_el == 0) && a->imm == 0xf000) {
219
+ gen_exception_internal_insn(s, EXCP_SEMIHOST);
220
+ } else {
221
+ unallocated_encoding(s);
132
+ }
222
+ }
133
+
223
+ return true;
134
+DO_3SAME(VADD, tcg_gen_gvec_add)
224
}
135
+DO_3SAME(VSUB, tcg_gen_gvec_sub)
225
136
diff --git a/target/arm/translate.c b/target/arm/translate.c
226
/*
137
index XXXXXXX..XXXXXXX 100644
227
@@ -XXX,XX +XXX,XX @@ static bool btype_destination_ok(uint32_t insn, bool bt, int btype)
138
--- a/target/arm/translate.c
228
static void disas_a64_legacy(DisasContext *s, uint32_t insn)
139
+++ b/target/arm/translate.c
229
{
140
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
230
switch (extract32(insn, 25, 4)) {
141
}
231
- case 0xa: case 0xb: /* Branch, exception generation and system insns */
142
return 0;
232
- disas_b_exc_sys(s, insn);
143
233
- break;
144
- case NEON_3R_VADD_VSUB:
234
case 0x4:
145
- if (u) {
235
case 0x6:
146
- tcg_gen_gvec_sub(size, rd_ofs, rn_ofs, rm_ofs,
236
case 0xc:
147
- vec_size, vec_size);
148
- } else {
149
- tcg_gen_gvec_add(size, rd_ofs, rn_ofs, rm_ofs,
150
- vec_size, vec_size);
151
- }
152
- return 0;
153
-
154
case NEON_3R_VQADD:
155
tcg_gen_gvec_4(rd_ofs, offsetof(CPUARMState, vfp.qc),
156
rn_ofs, rm_ofs, vec_size, vec_size,
157
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
158
tcg_gen_gvec_3(rd_ofs, rm_ofs, rn_ofs, vec_size, vec_size,
159
u ? &ushl_op[size] : &sshl_op[size]);
160
return 0;
161
+
162
+ case NEON_3R_VADD_VSUB:
163
+ /* Already handled by decodetree */
164
+ return 1;
165
}
166
167
if (size == 3) {
168
--
237
--
169
2.20.1
238
2.34.1
170
171
diff view generated by jsdifflib
1
Convert the VCMLA (vector) insns in the 3same extension group to
1
Convert the instructions in the load/store exclusive (STXR,
2
decodetree.
2
STLXR, LDXR, LDAXR) and load/store ordered (STLR, STLLR,
3
LDAR, LDLAR) to decodetree.
4
5
Note that for STLR, STLLR, LDAR, LDLAR this fixes an under-decoding
6
in the legacy decoder where we were not checking that the RES1 bits
7
in the Rs and Rt2 fields were set.
8
9
The new function ldst_iss_sf() is equivalent to the existing
10
disas_ldst_compute_iss_sf(), but it takes the pre-decoded 'ext' field
11
rather than taking an undecoded two-bit opc field and extracting
12
'ext' from it. Once all the loads and stores have been converted
13
to decodetree disas_ldst_compute_iss_sf() will be unused and
14
can be deleted.
3
15
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
17
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-5-peter.maydell@linaro.org
18
Message-id: 20230602155223.2040685-9-peter.maydell@linaro.org
7
---
19
---
8
target/arm/neon-shared.decode | 11 ++++++++++
20
target/arm/tcg/a64.decode | 11 +++
9
target/arm/translate-neon.inc.c | 37 +++++++++++++++++++++++++++++++++
21
target/arm/tcg/translate-a64.c | 154 ++++++++++++++++++++-------------
10
target/arm/translate.c | 11 +---------
22
2 files changed, 103 insertions(+), 62 deletions(-)
11
3 files changed, 49 insertions(+), 10 deletions(-)
23
12
24
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
14
index XXXXXXX..XXXXXXX 100644
25
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-shared.decode
26
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-shared.decode
27
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@
28
@@ -XXX,XX +XXX,XX @@ HLT 1101 0100 010 ................ 000 00 @i16
18
# More specifically, this covers:
29
# DCPS1 1101 0100 101 ................ 000 01 @i16
19
# 2reg scalar ext: 0b1111_1110_xxxx_xxxx_xxxx_1x0x_xxxx_xxxx
30
# DCPS2 1101 0100 101 ................ 000 10 @i16
20
# 3same ext: 0b1111_110x_xxxx_xxxx_xxxx_1x0x_xxxx_xxxx
31
# DCPS3 1101 0100 101 ................ 000 11 @i16
21
+
32
+
22
+# VFP/Neon register fields; same as vfp.decode
33
+# Loads and stores
23
+%vm_dp 5:1 0:4
34
+
24
+%vm_sp 0:4 5:1
35
+&stxr rn rt rt2 rs sz lasr
25
+%vn_dp 7:1 16:4
36
+&stlr rn rt sz lasr
26
+%vn_sp 16:4 7:1
37
+@stxr sz:2 ...... ... rs:5 lasr:1 rt2:5 rn:5 rt:5 &stxr
27
+%vd_dp 22:1 12:4
38
+@stlr sz:2 ...... ... ..... lasr:1 ..... rn:5 rt:5 &stlr
28
+%vd_sp 12:4 22:1
39
+STXR .. 001000 000 ..... . ..... ..... ..... @stxr # inc STLXR
29
+
40
+LDXR .. 001000 010 ..... . ..... ..... ..... @stxr # inc LDAXR
30
+VCMLA 1111 110 rot:2 . 1 size:1 .... .... 1000 . q:1 . 0 .... \
41
+STLR .. 001000 100 11111 . 11111 ..... ..... @stlr # inc STLLR
31
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp
42
+LDAR .. 001000 110 11111 . 11111 ..... ..... @stlr # inc LDLAR
32
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
43
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
33
index XXXXXXX..XXXXXXX 100644
44
index XXXXXXX..XXXXXXX 100644
34
--- a/target/arm/translate-neon.inc.c
45
--- a/target/arm/tcg/translate-a64.c
35
+++ b/target/arm/translate-neon.inc.c
46
+++ b/target/arm/tcg/translate-a64.c
36
@@ -XXX,XX +XXX,XX @@
47
@@ -XXX,XX +XXX,XX @@ static bool disas_ldst_compute_iss_sf(int size, bool is_signed, int opc)
37
#include "decode-neon-dp.inc.c"
48
return regsize == 64;
38
#include "decode-neon-ls.inc.c"
49
}
39
#include "decode-neon-shared.inc.c"
50
40
+
51
+static bool ldst_iss_sf(int size, bool sign, bool ext)
41
+static bool trans_VCMLA(DisasContext *s, arg_VCMLA *a)
52
+{
42
+{
53
+
43
+ int opr_sz;
54
+ if (sign) {
44
+ TCGv_ptr fpst;
55
+ /*
45
+ gen_helper_gvec_3_ptr *fn_gvec_ptr;
56
+ * Signed loads are 64 bit results if we are not going to
46
+
57
+ * do a zero-extend from 32 to 64 after the load.
47
+ if (!dc_isar_feature(aa32_vcma, s)
58
+ * (For a store, sign and ext are always false.)
48
+ || (!a->size && !dc_isar_feature(aa32_fp16_arith, s))) {
59
+ */
60
+ return !ext;
61
+ } else {
62
+ /* Unsigned loads/stores work at the specified size */
63
+ return size == MO_64;
64
+ }
65
+}
66
+
67
+static bool trans_STXR(DisasContext *s, arg_stxr *a)
68
+{
69
+ if (a->rn == 31) {
70
+ gen_check_sp_alignment(s);
71
+ }
72
+ if (a->lasr) {
73
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
74
+ }
75
+ gen_store_exclusive(s, a->rs, a->rt, a->rt2, a->rn, a->sz, false);
76
+ return true;
77
+}
78
+
79
+static bool trans_LDXR(DisasContext *s, arg_stxr *a)
80
+{
81
+ if (a->rn == 31) {
82
+ gen_check_sp_alignment(s);
83
+ }
84
+ gen_load_exclusive(s, a->rt, a->rt2, a->rn, a->sz, false);
85
+ if (a->lasr) {
86
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
87
+ }
88
+ return true;
89
+}
90
+
91
+static bool trans_STLR(DisasContext *s, arg_stlr *a)
92
+{
93
+ TCGv_i64 clean_addr;
94
+ MemOp memop;
95
+ bool iss_sf = ldst_iss_sf(a->sz, false, false);
96
+
97
+ /*
98
+ * StoreLORelease is the same as Store-Release for QEMU, but
99
+ * needs the feature-test.
100
+ */
101
+ if (!a->lasr && !dc_isar_feature(aa64_lor, s)) {
49
+ return false;
102
+ return false;
50
+ }
103
+ }
51
+
104
+ /* Generate ISS for non-exclusive accesses including LASR. */
52
+ /* UNDEF accesses to D16-D31 if they don't exist. */
105
+ if (a->rn == 31) {
53
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
106
+ gen_check_sp_alignment(s);
54
+ ((a->vd | a->vn | a->vm) & 0x10)) {
107
+ }
108
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
109
+ memop = check_ordered_align(s, a->rn, 0, true, a->sz);
110
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, a->rn),
111
+ true, a->rn != 31, memop);
112
+ do_gpr_st(s, cpu_reg(s, a->rt), clean_addr, memop, true, a->rt,
113
+ iss_sf, a->lasr);
114
+ return true;
115
+}
116
+
117
+static bool trans_LDAR(DisasContext *s, arg_stlr *a)
118
+{
119
+ TCGv_i64 clean_addr;
120
+ MemOp memop;
121
+ bool iss_sf = ldst_iss_sf(a->sz, false, false);
122
+
123
+ /* LoadLOAcquire is the same as Load-Acquire for QEMU. */
124
+ if (!a->lasr && !dc_isar_feature(aa64_lor, s)) {
55
+ return false;
125
+ return false;
56
+ }
126
+ }
57
+
127
+ /* Generate ISS for non-exclusive accesses including LASR. */
58
+ if ((a->vn | a->vm | a->vd) & a->q) {
128
+ if (a->rn == 31) {
59
+ return false;
129
+ gen_check_sp_alignment(s);
60
+ }
130
+ }
61
+
131
+ memop = check_ordered_align(s, a->rn, 0, false, a->sz);
62
+ if (!vfp_access_check(s)) {
132
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, a->rn),
63
+ return true;
133
+ false, a->rn != 31, memop);
64
+ }
134
+ do_gpr_ld(s, cpu_reg(s, a->rt), clean_addr, memop, false, true,
65
+
135
+ a->rt, iss_sf, a->lasr);
66
+ opr_sz = (1 + a->q) * 8;
136
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
67
+ fpst = get_fpstatus_ptr(1);
137
+ return true;
68
+ fn_gvec_ptr = a->size ? gen_helper_gvec_fcmlas : gen_helper_gvec_fcmlah;
138
+}
69
+ tcg_gen_gvec_3_ptr(vfp_reg_offset(1, a->vd),
139
+
70
+ vfp_reg_offset(1, a->vn),
140
/* Load/store exclusive
71
+ vfp_reg_offset(1, a->vm),
141
*
72
+ fpst, opr_sz, opr_sz, a->rot,
142
* 31 30 29 24 23 22 21 20 16 15 14 10 9 5 4 0
73
+ fn_gvec_ptr);
143
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
74
+ tcg_temp_free_ptr(fpst);
144
int is_lasr = extract32(insn, 15, 1);
75
+ return true;
145
int o2_L_o1_o0 = extract32(insn, 21, 3) * 2 | is_lasr;
76
+}
146
int size = extract32(insn, 30, 2);
77
diff --git a/target/arm/translate.c b/target/arm/translate.c
147
- TCGv_i64 clean_addr;
78
index XXXXXXX..XXXXXXX 100644
148
- MemOp memop;
79
--- a/target/arm/translate.c
149
80
+++ b/target/arm/translate.c
150
switch (o2_L_o1_o0) {
81
@@ -XXX,XX +XXX,XX @@ static int disas_neon_insn_3same_ext(DisasContext *s, uint32_t insn)
151
- case 0x0: /* STXR */
82
bool is_long = false, q = extract32(insn, 6, 1);
152
- case 0x1: /* STLXR */
83
bool ptr_is_env = false;
153
- if (rn == 31) {
84
154
- gen_check_sp_alignment(s);
85
- if ((insn & 0xfe200f10) == 0xfc200800) {
155
- }
86
- /* VCMLA -- 1111 110R R.1S .... .... 1000 ...0 .... */
156
- if (is_lasr) {
87
- int size = extract32(insn, 20, 1);
157
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
88
- data = extract32(insn, 23, 2); /* rot */
158
- }
89
- if (!dc_isar_feature(aa32_vcma, s)
159
- gen_store_exclusive(s, rs, rt, rt2, rn, size, false);
90
- || (!size && !dc_isar_feature(aa32_fp16_arith, s))) {
160
- return;
91
- return 1;
161
-
92
- }
162
- case 0x4: /* LDXR */
93
- fn_gvec_ptr = size ? gen_helper_gvec_fcmlas : gen_helper_gvec_fcmlah;
163
- case 0x5: /* LDAXR */
94
- } else if ((insn & 0xfea00f10) == 0xfc800800) {
164
- if (rn == 31) {
95
+ if ((insn & 0xfea00f10) == 0xfc800800) {
165
- gen_check_sp_alignment(s);
96
/* VCADD -- 1111 110R 1.0S .... .... 1000 ...0 .... */
166
- }
97
int size = extract32(insn, 20, 1);
167
- gen_load_exclusive(s, rt, rt2, rn, size, false);
98
data = extract32(insn, 24, 1); /* rot */
168
- if (is_lasr) {
169
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
170
- }
171
- return;
172
-
173
- case 0x8: /* STLLR */
174
- if (!dc_isar_feature(aa64_lor, s)) {
175
- break;
176
- }
177
- /* StoreLORelease is the same as Store-Release for QEMU. */
178
- /* fall through */
179
- case 0x9: /* STLR */
180
- /* Generate ISS for non-exclusive accesses including LASR. */
181
- if (rn == 31) {
182
- gen_check_sp_alignment(s);
183
- }
184
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
185
- memop = check_ordered_align(s, rn, 0, true, size);
186
- clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
187
- true, rn != 31, memop);
188
- do_gpr_st(s, cpu_reg(s, rt), clean_addr, memop, true, rt,
189
- disas_ldst_compute_iss_sf(size, false, 0), is_lasr);
190
- return;
191
-
192
- case 0xc: /* LDLAR */
193
- if (!dc_isar_feature(aa64_lor, s)) {
194
- break;
195
- }
196
- /* LoadLOAcquire is the same as Load-Acquire for QEMU. */
197
- /* fall through */
198
- case 0xd: /* LDAR */
199
- /* Generate ISS for non-exclusive accesses including LASR. */
200
- if (rn == 31) {
201
- gen_check_sp_alignment(s);
202
- }
203
- memop = check_ordered_align(s, rn, 0, false, size);
204
- clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
205
- false, rn != 31, memop);
206
- do_gpr_ld(s, cpu_reg(s, rt), clean_addr, memop, false, true,
207
- rt, disas_ldst_compute_iss_sf(size, false, 0), is_lasr);
208
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
209
- return;
210
-
211
case 0x2: case 0x3: /* CASP / STXP */
212
if (size & 2) { /* STXP / STLXP */
213
if (rn == 31) {
214
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
215
return;
216
}
217
break;
218
+ default:
219
+ /* Handled in decodetree */
220
+ break;
221
}
222
unallocated_encoding(s);
223
}
99
--
224
--
100
2.20.1
225
2.34.1
101
102
diff view generated by jsdifflib
1
Convert VCMLA (scalar) in the 2reg-scalar-ext group to decodetree.
1
Convert the load/store exclusive pair (LDXP, STXP, LDAXP, STLXP),
2
compare-and-swap pair (CASP, CASPA, CASPAL, CASPL), and compare-and
3
swap (CAS, CASA, CASAL, CASL) instructions to decodetree.
2
4
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200430181003.21682-9-peter.maydell@linaro.org
7
Message-id: 20230602155223.2040685-10-peter.maydell@linaro.org
6
---
8
---
7
target/arm/neon-shared.decode | 5 +++++
9
target/arm/tcg/a64.decode | 11 +++
8
target/arm/translate-neon.inc.c | 40 +++++++++++++++++++++++++++++++++
10
target/arm/tcg/translate-a64.c | 121 ++++++++++++---------------------
9
target/arm/translate.c | 26 +--------------------
11
2 files changed, 53 insertions(+), 79 deletions(-)
10
3 files changed, 46 insertions(+), 25 deletions(-)
11
12
12
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/neon-shared.decode
15
--- a/target/arm/tcg/a64.decode
15
+++ b/target/arm/neon-shared.decode
16
+++ b/target/arm/tcg/a64.decode
16
@@ -XXX,XX +XXX,XX @@ VFML 1111 110 0 s:1 . 10 .... .... 1000 . 0 . 1 .... \
17
@@ -XXX,XX +XXX,XX @@ HLT 1101 0100 010 ................ 000 00 @i16
17
vm=%vm_sp vn=%vn_sp vd=%vd_dp q=0
18
&stlr rn rt sz lasr
18
VFML 1111 110 0 s:1 . 10 .... .... 1000 . 1 . 1 .... \
19
@stxr sz:2 ...... ... rs:5 lasr:1 rt2:5 rn:5 rt:5 &stxr
19
vm=%vm_dp vn=%vn_dp vd=%vd_dp q=1
20
@stlr sz:2 ...... ... ..... lasr:1 ..... rn:5 rt:5 &stlr
21
+%imm1_30_p2 30:1 !function=plus_2
22
+@stxp .. ...... ... rs:5 lasr:1 rt2:5 rn:5 rt:5 &stxr sz=%imm1_30_p2
23
STXR .. 001000 000 ..... . ..... ..... ..... @stxr # inc STLXR
24
LDXR .. 001000 010 ..... . ..... ..... ..... @stxr # inc LDAXR
25
STLR .. 001000 100 11111 . 11111 ..... ..... @stlr # inc STLLR
26
LDAR .. 001000 110 11111 . 11111 ..... ..... @stlr # inc LDLAR
20
+
27
+
21
+VCMLA_scalar 1111 1110 0 . rot:2 .... .... 1000 . q:1 index:1 0 vm:4 \
28
+STXP 1 . 001000 001 ..... . ..... ..... ..... @stxp # inc STLXP
22
+ vn=%vn_dp vd=%vd_dp size=0
29
+LDXP 1 . 001000 011 ..... . ..... ..... ..... @stxp # inc LDAXP
23
+VCMLA_scalar 1111 1110 1 . rot:2 .... .... 1000 . q:1 . 0 .... \
30
+
24
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp size=1 index=0
31
+# CASP, CASPA, CASPAL, CASPL (we don't decode the bits that determine
25
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
32
+# acquire/release semantics because QEMU's cmpxchg always has those)
33
+CASP 0 . 001000 0 - 1 rs:5 - 11111 rn:5 rt:5 sz=%imm1_30_p2
34
+# CAS, CASA, CASAL, CASL
35
+CAS sz:2 001000 1 - 1 rs:5 - 11111 rn:5 rt:5
36
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
26
index XXXXXXX..XXXXXXX 100644
37
index XXXXXXX..XXXXXXX 100644
27
--- a/target/arm/translate-neon.inc.c
38
--- a/target/arm/tcg/translate-a64.c
28
+++ b/target/arm/translate-neon.inc.c
39
+++ b/target/arm/tcg/translate-a64.c
29
@@ -XXX,XX +XXX,XX @@ static bool trans_VFML(DisasContext *s, arg_VFML *a)
40
@@ -XXX,XX +XXX,XX @@ static bool trans_LDAR(DisasContext *s, arg_stlr *a)
30
gen_helper_gvec_fmlal_a32);
31
return true;
41
return true;
32
}
42
}
43
44
-/* Load/store exclusive
45
- *
46
- * 31 30 29 24 23 22 21 20 16 15 14 10 9 5 4 0
47
- * +-----+-------------+----+---+----+------+----+-------+------+------+
48
- * | sz | 0 0 1 0 0 0 | o2 | L | o1 | Rs | o0 | Rt2 | Rn | Rt |
49
- * +-----+-------------+----+---+----+------+----+-------+------+------+
50
- *
51
- * sz: 00 -> 8 bit, 01 -> 16 bit, 10 -> 32 bit, 11 -> 64 bit
52
- * L: 0 -> store, 1 -> load
53
- * o2: 0 -> exclusive, 1 -> not
54
- * o1: 0 -> single register, 1 -> register pair
55
- * o0: 1 -> load-acquire/store-release, 0 -> not
56
- */
57
-static void disas_ldst_excl(DisasContext *s, uint32_t insn)
58
+static bool trans_STXP(DisasContext *s, arg_stxr *a)
59
{
60
- int rt = extract32(insn, 0, 5);
61
- int rn = extract32(insn, 5, 5);
62
- int rt2 = extract32(insn, 10, 5);
63
- int rs = extract32(insn, 16, 5);
64
- int is_lasr = extract32(insn, 15, 1);
65
- int o2_L_o1_o0 = extract32(insn, 21, 3) * 2 | is_lasr;
66
- int size = extract32(insn, 30, 2);
67
-
68
- switch (o2_L_o1_o0) {
69
- case 0x2: case 0x3: /* CASP / STXP */
70
- if (size & 2) { /* STXP / STLXP */
71
- if (rn == 31) {
72
- gen_check_sp_alignment(s);
73
- }
74
- if (is_lasr) {
75
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
76
- }
77
- gen_store_exclusive(s, rs, rt, rt2, rn, size, true);
78
- return;
79
- }
80
- if (rt2 == 31
81
- && ((rt | rs) & 1) == 0
82
- && dc_isar_feature(aa64_atomics, s)) {
83
- /* CASP / CASPL */
84
- gen_compare_and_swap_pair(s, rs, rt, rn, size | 2);
85
- return;
86
- }
87
- break;
88
-
89
- case 0x6: case 0x7: /* CASPA / LDXP */
90
- if (size & 2) { /* LDXP / LDAXP */
91
- if (rn == 31) {
92
- gen_check_sp_alignment(s);
93
- }
94
- gen_load_exclusive(s, rt, rt2, rn, size, true);
95
- if (is_lasr) {
96
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
97
- }
98
- return;
99
- }
100
- if (rt2 == 31
101
- && ((rt | rs) & 1) == 0
102
- && dc_isar_feature(aa64_atomics, s)) {
103
- /* CASPA / CASPAL */
104
- gen_compare_and_swap_pair(s, rs, rt, rn, size | 2);
105
- return;
106
- }
107
- break;
108
-
109
- case 0xa: /* CAS */
110
- case 0xb: /* CASL */
111
- case 0xe: /* CASA */
112
- case 0xf: /* CASAL */
113
- if (rt2 == 31 && dc_isar_feature(aa64_atomics, s)) {
114
- gen_compare_and_swap(s, rs, rt, rn, size);
115
- return;
116
- }
117
- break;
118
- default:
119
- /* Handled in decodetree */
120
- break;
121
+ if (a->rn == 31) {
122
+ gen_check_sp_alignment(s);
123
}
124
- unallocated_encoding(s);
125
+ if (a->lasr) {
126
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
127
+ }
128
+ gen_store_exclusive(s, a->rs, a->rt, a->rt2, a->rn, a->sz, true);
129
+ return true;
130
+}
33
+
131
+
34
+static bool trans_VCMLA_scalar(DisasContext *s, arg_VCMLA_scalar *a)
132
+static bool trans_LDXP(DisasContext *s, arg_stxr *a)
35
+{
133
+{
36
+ gen_helper_gvec_3_ptr *fn_gvec_ptr;
134
+ if (a->rn == 31) {
37
+ int opr_sz;
135
+ gen_check_sp_alignment(s);
38
+ TCGv_ptr fpst;
136
+ }
137
+ gen_load_exclusive(s, a->rt, a->rt2, a->rn, a->sz, true);
138
+ if (a->lasr) {
139
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
140
+ }
141
+ return true;
142
+}
39
+
143
+
40
+ if (!dc_isar_feature(aa32_vcma, s)) {
144
+static bool trans_CASP(DisasContext *s, arg_CASP *a)
145
+{
146
+ if (!dc_isar_feature(aa64_atomics, s)) {
41
+ return false;
147
+ return false;
42
+ }
148
+ }
43
+ if (a->size == 0 && !dc_isar_feature(aa32_fp16_arith, s)) {
149
+ if (((a->rt | a->rs) & 1) != 0) {
44
+ return false;
150
+ return false;
45
+ }
151
+ }
46
+
152
+
47
+ /* UNDEF accesses to D16-D31 if they don't exist. */
153
+ gen_compare_and_swap_pair(s, a->rs, a->rt, a->rn, a->sz);
48
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
154
+ return true;
49
+ ((a->vd | a->vn | a->vm) & 0x10)) {
155
+}
156
+
157
+static bool trans_CAS(DisasContext *s, arg_CAS *a)
158
+{
159
+ if (!dc_isar_feature(aa64_atomics, s)) {
50
+ return false;
160
+ return false;
51
+ }
161
+ }
52
+
162
+ gen_compare_and_swap(s, a->rs, a->rt, a->rn, a->sz);
53
+ if ((a->vd | a->vn) & a->q) {
54
+ return false;
55
+ }
56
+
57
+ if (!vfp_access_check(s)) {
58
+ return true;
59
+ }
60
+
61
+ fn_gvec_ptr = (a->size ? gen_helper_gvec_fcmlas_idx
62
+ : gen_helper_gvec_fcmlah_idx);
63
+ opr_sz = (1 + a->q) * 8;
64
+ fpst = get_fpstatus_ptr(1);
65
+ tcg_gen_gvec_3_ptr(vfp_reg_offset(1, a->vd),
66
+ vfp_reg_offset(1, a->vn),
67
+ vfp_reg_offset(1, a->vm),
68
+ fpst, opr_sz, opr_sz,
69
+ (a->index << 2) | a->rot, fn_gvec_ptr);
70
+ tcg_temp_free_ptr(fpst);
71
+ return true;
163
+ return true;
72
+}
164
}
73
diff --git a/target/arm/translate.c b/target/arm/translate.c
165
74
index XXXXXXX..XXXXXXX 100644
166
/*
75
--- a/target/arm/translate.c
167
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
76
+++ b/target/arm/translate.c
168
static void disas_ldst(DisasContext *s, uint32_t insn)
77
@@ -XXX,XX +XXX,XX @@ static int disas_neon_insn_2reg_scalar_ext(DisasContext *s, uint32_t insn)
169
{
78
bool is_long = false, q = extract32(insn, 6, 1);
170
switch (extract32(insn, 24, 6)) {
79
bool ptr_is_env = false;
171
- case 0x08: /* Load/store exclusive */
80
172
- disas_ldst_excl(s, insn);
81
- if ((insn & 0xff000f10) == 0xfe000800) {
173
- break;
82
- /* VCMLA (indexed) -- 1111 1110 S.RR .... .... 1000 ...0 .... */
174
case 0x18: case 0x1c: /* Load register (literal) */
83
- int rot = extract32(insn, 20, 2);
175
disas_ld_lit(s, insn);
84
- int size = extract32(insn, 23, 1);
176
break;
85
- int index;
86
-
87
- if (!dc_isar_feature(aa32_vcma, s)) {
88
- return 1;
89
- }
90
- if (size == 0) {
91
- if (!dc_isar_feature(aa32_fp16_arith, s)) {
92
- return 1;
93
- }
94
- /* For fp16, rm is just Vm, and index is M. */
95
- rm = extract32(insn, 0, 4);
96
- index = extract32(insn, 5, 1);
97
- } else {
98
- /* For fp32, rm is the usual M:Vm, and index is 0. */
99
- VFP_DREG_M(rm, insn);
100
- index = 0;
101
- }
102
- data = (index << 2) | rot;
103
- fn_gvec_ptr = (size ? gen_helper_gvec_fcmlas_idx
104
- : gen_helper_gvec_fcmlah_idx);
105
- } else if ((insn & 0xffb00f00) == 0xfe200d00) {
106
+ if ((insn & 0xffb00f00) == 0xfe200d00) {
107
/* V[US]DOT -- 1111 1110 0.10 .... .... 1101 .Q.U .... */
108
int u = extract32(insn, 4, 1);
109
110
--
177
--
111
2.20.1
178
2.34.1
112
113
diff view generated by jsdifflib
1
Convert the Neon "load single structure to all lanes" insns to
1
Convert the "Load register (literal)" instruction class to
2
decodetree.
2
decodetree.
3
3
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-13-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-11-peter.maydell@linaro.org
7
---
7
---
8
target/arm/neon-ls.decode | 5 +++
8
target/arm/tcg/a64.decode | 13 ++++++
9
target/arm/translate-neon.inc.c | 73 +++++++++++++++++++++++++++++++++
9
target/arm/tcg/translate-a64.c | 76 ++++++++++------------------------
10
target/arm/translate.c | 55 +------------------------
10
2 files changed, 35 insertions(+), 54 deletions(-)
11
3 files changed, 80 insertions(+), 53 deletions(-)
12
11
13
diff --git a/target/arm/neon-ls.decode b/target/arm/neon-ls.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
14
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-ls.decode
14
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-ls.decode
15
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ LDXP 1 . 001000 011 ..... . ..... ..... ..... @stxp # inc LDAXP
18
17
CASP 0 . 001000 0 - 1 rs:5 - 11111 rn:5 rt:5 sz=%imm1_30_p2
19
VLDST_multiple 1111 0100 0 . l:1 0 rn:4 .... itype:4 size:2 align:2 rm:4 \
18
# CAS, CASA, CASAL, CASL
20
vd=%vd_dp
19
CAS sz:2 001000 1 - 1 rs:5 - 11111 rn:5 rt:5
21
+
20
+
22
+# Neon load single element to all lanes
21
+&ldlit rt imm sz sign
22
+@ldlit .. ... . .. ................... rt:5 &ldlit imm=%imm19
23
+
23
+
24
+VLD_all_lanes 1111 0100 1 . 1 0 rn:4 .... 11 n:2 size:2 t:1 a:1 rm:4 \
24
+LD_lit 00 011 0 00 ................... ..... @ldlit sz=2 sign=0
25
+ vd=%vd_dp
25
+LD_lit 01 011 0 00 ................... ..... @ldlit sz=3 sign=0
26
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
26
+LD_lit 10 011 0 00 ................... ..... @ldlit sz=2 sign=1
27
+LD_lit_v 00 011 1 00 ................... ..... @ldlit sz=2 sign=0
28
+LD_lit_v 01 011 1 00 ................... ..... @ldlit sz=3 sign=0
29
+LD_lit_v 10 011 1 00 ................... ..... @ldlit sz=4 sign=0
30
+
31
+# PRFM
32
+NOP 11 011 0 00 ------------------- -----
33
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
27
index XXXXXXX..XXXXXXX 100644
34
index XXXXXXX..XXXXXXX 100644
28
--- a/target/arm/translate-neon.inc.c
35
--- a/target/arm/tcg/translate-a64.c
29
+++ b/target/arm/translate-neon.inc.c
36
+++ b/target/arm/tcg/translate-a64.c
30
@@ -XXX,XX +XXX,XX @@ static bool trans_VLDST_multiple(DisasContext *s, arg_VLDST_multiple *a)
37
@@ -XXX,XX +XXX,XX @@ static bool trans_CAS(DisasContext *s, arg_CAS *a)
31
gen_neon_ldst_base_update(s, a->rm, a->rn, nregs * interleave * 8);
32
return true;
38
return true;
33
}
39
}
40
41
-/*
42
- * Load register (literal)
43
- *
44
- * 31 30 29 27 26 25 24 23 5 4 0
45
- * +-----+-------+---+-----+-------------------+-------+
46
- * | opc | 0 1 1 | V | 0 0 | imm19 | Rt |
47
- * +-----+-------+---+-----+-------------------+-------+
48
- *
49
- * V: 1 -> vector (simd/fp)
50
- * opc (non-vector): 00 -> 32 bit, 01 -> 64 bit,
51
- * 10-> 32 bit signed, 11 -> prefetch
52
- * opc (vector): 00 -> 32 bit, 01 -> 64 bit, 10 -> 128 bit (11 unallocated)
53
- */
54
-static void disas_ld_lit(DisasContext *s, uint32_t insn)
55
+static bool trans_LD_lit(DisasContext *s, arg_ldlit *a)
56
{
57
- int rt = extract32(insn, 0, 5);
58
- int64_t imm = sextract32(insn, 5, 19) << 2;
59
- bool is_vector = extract32(insn, 26, 1);
60
- int opc = extract32(insn, 30, 2);
61
- bool is_signed = false;
62
- int size = 2;
63
- TCGv_i64 tcg_rt, clean_addr;
64
+ bool iss_sf = ldst_iss_sf(a->sz, a->sign, false);
65
+ TCGv_i64 tcg_rt = cpu_reg(s, a->rt);
66
+ TCGv_i64 clean_addr = tcg_temp_new_i64();
67
+ MemOp memop = finalize_memop(s, a->sz + a->sign * MO_SIGN);
34
+
68
+
35
+static bool trans_VLD_all_lanes(DisasContext *s, arg_VLD_all_lanes *a)
69
+ gen_pc_plus_diff(s, clean_addr, a->imm);
36
+{
70
+ do_gpr_ld(s, tcg_rt, clean_addr, memop,
37
+ /* Neon load single structure to all lanes */
71
+ false, true, a->rt, iss_sf, false);
38
+ int reg, stride, vec_size;
39
+ int vd = a->vd;
40
+ int size = a->size;
41
+ int nregs = a->n + 1;
42
+ TCGv_i32 addr, tmp;
43
+
44
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
45
+ return false;
46
+ }
47
+
48
+ /* UNDEF accesses to D16-D31 if they don't exist */
49
+ if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
50
+ return false;
51
+ }
52
+
53
+ if (size == 3) {
54
+ if (nregs != 4 || a->a == 0) {
55
+ return false;
56
+ }
57
+ /* For VLD4 size == 3 a == 1 means 32 bits at 16 byte alignment */
58
+ size = 2;
59
+ }
60
+ if (nregs == 1 && a->a == 1 && size == 0) {
61
+ return false;
62
+ }
63
+ if (nregs == 3 && a->a == 1) {
64
+ return false;
65
+ }
66
+
67
+ if (!vfp_access_check(s)) {
68
+ return true;
69
+ }
70
+
71
+ /*
72
+ * VLD1 to all lanes: T bit indicates how many Dregs to write.
73
+ * VLD2/3/4 to all lanes: T bit indicates register stride.
74
+ */
75
+ stride = a->t ? 2 : 1;
76
+ vec_size = nregs == 1 ? stride * 8 : 8;
77
+
78
+ tmp = tcg_temp_new_i32();
79
+ addr = tcg_temp_new_i32();
80
+ load_reg_var(s, addr, a->rn);
81
+ for (reg = 0; reg < nregs; reg++) {
82
+ gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s),
83
+ s->be_data | size);
84
+ if ((vd & 1) && vec_size == 16) {
85
+ /*
86
+ * We cannot write 16 bytes at once because the
87
+ * destination is unaligned.
88
+ */
89
+ tcg_gen_gvec_dup_i32(size, neon_reg_offset(vd, 0),
90
+ 8, 8, tmp);
91
+ tcg_gen_gvec_mov(0, neon_reg_offset(vd + 1, 0),
92
+ neon_reg_offset(vd, 0), 8, 8);
93
+ } else {
94
+ tcg_gen_gvec_dup_i32(size, neon_reg_offset(vd, 0),
95
+ vec_size, vec_size, tmp);
96
+ }
97
+ tcg_gen_addi_i32(addr, addr, 1 << size);
98
+ vd += stride;
99
+ }
100
+ tcg_temp_free_i32(tmp);
101
+ tcg_temp_free_i32(addr);
102
+
103
+ gen_neon_ldst_base_update(s, a->rm, a->rn, (1 << size) * nregs);
104
+
105
+ return true;
72
+ return true;
106
+}
73
+}
107
diff --git a/target/arm/translate.c b/target/arm/translate.c
74
+
108
index XXXXXXX..XXXXXXX 100644
75
+static bool trans_LD_lit_v(DisasContext *s, arg_ldlit *a)
109
--- a/target/arm/translate.c
76
+{
110
+++ b/target/arm/translate.c
77
+ /* Load register (literal), vector version */
111
@@ -XXX,XX +XXX,XX @@ static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
78
+ TCGv_i64 clean_addr;
112
int size;
79
MemOp memop;
113
int reg;
80
114
int load;
81
- if (is_vector) {
115
- int vec_size;
82
- if (opc == 3) {
116
TCGv_i32 addr;
83
- unallocated_encoding(s);
117
TCGv_i32 tmp;
84
- return;
118
85
- }
119
@@ -XXX,XX +XXX,XX @@ static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
86
- size = 2 + opc;
120
} else {
87
- if (!fp_access_check(s)) {
121
size = (insn >> 10) & 3;
88
- return;
122
if (size == 3) {
89
- }
123
- /* Load single element to all lanes. */
90
- memop = finalize_memop_asimd(s, size);
124
- int a = (insn >> 4) & 1;
91
- } else {
125
- if (!load) {
92
- if (opc == 3) {
126
- return 1;
93
- /* PRFM (literal) : prefetch */
127
- }
94
- return;
128
- size = (insn >> 6) & 3;
95
- }
129
- nregs = ((insn >> 8) & 3) + 1;
96
- size = 2 + extract32(opc, 0, 1);
97
- is_signed = extract32(opc, 1, 1);
98
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
99
+ if (!fp_access_check(s)) {
100
+ return true;
101
}
130
-
102
-
131
- if (size == 3) {
103
- tcg_rt = cpu_reg(s, rt);
132
- if (nregs != 4 || a == 0) {
133
- return 1;
134
- }
135
- /* For VLD4 size==3 a == 1 means 32 bits at 16 byte alignment */
136
- size = 2;
137
- }
138
- if (nregs == 1 && a == 1 && size == 0) {
139
- return 1;
140
- }
141
- if (nregs == 3 && a == 1) {
142
- return 1;
143
- }
144
- addr = tcg_temp_new_i32();
145
- load_reg_var(s, addr, rn);
146
-
104
-
147
- /* VLD1 to all lanes: bit 5 indicates how many Dregs to write.
105
+ memop = finalize_memop_asimd(s, a->sz);
148
- * VLD2/3/4 to all lanes: bit 5 indicates register stride.
106
clean_addr = tcg_temp_new_i64();
149
- */
107
- gen_pc_plus_diff(s, clean_addr, imm);
150
- stride = (insn & (1 << 5)) ? 2 : 1;
151
- vec_size = nregs == 1 ? stride * 8 : 8;
152
-
108
-
153
- tmp = tcg_temp_new_i32();
109
- if (is_vector) {
154
- for (reg = 0; reg < nregs; reg++) {
110
- do_fp_ld(s, rt, clean_addr, memop);
155
- gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s),
111
- } else {
156
- s->be_data | size);
112
- /* Only unsigned 32bit loads target 32bit registers. */
157
- if ((rd & 1) && vec_size == 16) {
113
- bool iss_sf = opc != 0;
158
- /* We cannot write 16 bytes at once because the
114
- do_gpr_ld(s, tcg_rt, clean_addr, memop, false, true, rt, iss_sf, false);
159
- * destination is unaligned.
115
- }
160
- */
116
+ gen_pc_plus_diff(s, clean_addr, a->imm);
161
- tcg_gen_gvec_dup_i32(size, neon_reg_offset(rd, 0),
117
+ do_fp_ld(s, a->rt, clean_addr, memop);
162
- 8, 8, tmp);
118
+ return true;
163
- tcg_gen_gvec_mov(0, neon_reg_offset(rd + 1, 0),
119
}
164
- neon_reg_offset(rd, 0), 8, 8);
120
165
- } else {
121
/*
166
- tcg_gen_gvec_dup_i32(size, neon_reg_offset(rd, 0),
122
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
167
- vec_size, vec_size, tmp);
123
static void disas_ldst(DisasContext *s, uint32_t insn)
168
- }
124
{
169
- tcg_gen_addi_i32(addr, addr, 1 << size);
125
switch (extract32(insn, 24, 6)) {
170
- rd += stride;
126
- case 0x18: case 0x1c: /* Load register (literal) */
171
- }
127
- disas_ld_lit(s, insn);
172
- tcg_temp_free_i32(tmp);
128
- break;
173
- tcg_temp_free_i32(addr);
129
case 0x28: case 0x29:
174
- stride = (1 << size) * nregs;
130
case 0x2c: case 0x2d: /* Load/store pair (all forms) */
175
+ /* Load single element to all lanes -- handled by decodetree */
131
disas_ldst_pair(s, insn);
176
+ return 1;
177
} else {
178
/* Single element. */
179
int idx = (insn >> 4) & 0xf;
180
--
132
--
181
2.20.1
133
2.34.1
182
183
diff view generated by jsdifflib
1
For ARMv8.2-TTS2UXN, the stage 2 page table walk wants to know
1
Convert the load/store register pair insns (LDP, STP,
2
whether the stage 1 access is for EL0 or not, because whether
2
LDNP, STNP, LDPSW, STGP) to decodetree.
3
exec permission is given can depend on whether this is an EL0
4
or EL1 access. Add a new argument to get_phys_addr_lpae() so
5
the call sites can pass this information in.
6
7
Since get_phys_addr_lpae() doesn't already have a doc comment,
8
add one so we have a place to put the documentation of the
9
semantics of the new s1_is_el0 argument.
10
3
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
5
Message-id: 20230602155223.2040685-12-peter.maydell@linaro.org
13
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
14
Message-id: 20200330210400.11724-4-peter.maydell@linaro.org
15
---
7
---
16
target/arm/helper.c | 29 ++++++++++++++++++++++++++++-
8
target/arm/tcg/a64.decode | 61 +++++
17
1 file changed, 28 insertions(+), 1 deletion(-)
9
target/arm/tcg/translate-a64.c | 422 ++++++++++++++++-----------------
10
2 files changed, 268 insertions(+), 215 deletions(-)
18
11
19
diff --git a/target/arm/helper.c b/target/arm/helper.c
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
20
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
21
--- a/target/arm/helper.c
14
--- a/target/arm/tcg/a64.decode
22
+++ b/target/arm/helper.c
15
+++ b/target/arm/tcg/a64.decode
23
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ LD_lit_v 10 011 1 00 ................... ..... @ldlit sz=4 sign=0
24
17
25
static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
18
# PRFM
26
MMUAccessType access_type, ARMMMUIdx mmu_idx,
19
NOP 11 011 0 00 ------------------- -----
27
+ bool s1_is_el0,
20
+
28
hwaddr *phys_ptr, MemTxAttrs *txattrs, int *prot,
21
+&ldstpair rt2 rt rn imm sz sign w p
29
target_ulong *page_size_ptr,
22
+@ldstpair .. ... . ... . imm:s7 rt2:5 rn:5 rt:5 &ldstpair
30
ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs);
23
+
31
@@ -XXX,XX +XXX,XX @@ static hwaddr S1_ptw_translate(CPUARMState *env, ARMMMUIdx mmu_idx,
24
+# STNP, LDNP: Signed offset, non-temporal hint. We don't emulate caches
32
}
25
+# so we ignore hints about data access patterns, and handle these like
33
26
+# plain signed offset.
34
ret = get_phys_addr_lpae(env, addr, MMU_DATA_LOAD, ARMMMUIdx_Stage2,
27
+STP 00 101 0 000 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
35
+ false,
28
+LDP 00 101 0 000 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
36
&s2pa, &txattrs, &s2prot, &s2size, fi,
29
+STP 10 101 0 000 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
37
pcacheattrs);
30
+LDP 10 101 0 000 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
38
if (ret) {
31
+STP_v 00 101 1 000 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
39
@@ -XXX,XX +XXX,XX @@ static ARMVAParameters aa32_va_parameters(CPUARMState *env, uint32_t va,
32
+LDP_v 00 101 1 000 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
40
};
33
+STP_v 01 101 1 000 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
34
+LDP_v 01 101 1 000 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
35
+STP_v 10 101 1 000 0 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=0
36
+LDP_v 10 101 1 000 1 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=0
37
+
38
+# STP and LDP: post-indexed
39
+STP 00 101 0 001 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=1 w=1
40
+LDP 00 101 0 001 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=1 w=1
41
+LDP 01 101 0 001 1 ....... ..... ..... ..... @ldstpair sz=2 sign=1 p=1 w=1
42
+STP 10 101 0 001 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
43
+LDP 10 101 0 001 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
44
+STP_v 00 101 1 001 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=1 w=1
45
+LDP_v 00 101 1 001 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=1 w=1
46
+STP_v 01 101 1 001 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
47
+LDP_v 01 101 1 001 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
48
+STP_v 10 101 1 001 0 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=1 w=1
49
+LDP_v 10 101 1 001 1 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=1 w=1
50
+
51
+# STP and LDP: offset
52
+STP 00 101 0 010 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
53
+LDP 00 101 0 010 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
54
+LDP 01 101 0 010 1 ....... ..... ..... ..... @ldstpair sz=2 sign=1 p=0 w=0
55
+STP 10 101 0 010 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
56
+LDP 10 101 0 010 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
57
+STP_v 00 101 1 010 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
58
+LDP_v 00 101 1 010 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=0
59
+STP_v 01 101 1 010 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
60
+LDP_v 01 101 1 010 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
61
+STP_v 10 101 1 010 0 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=0
62
+LDP_v 10 101 1 010 1 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=0
63
+
64
+# STP and LDP: pre-indexed
65
+STP 00 101 0 011 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=1
66
+LDP 00 101 0 011 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=1
67
+LDP 01 101 0 011 1 ....... ..... ..... ..... @ldstpair sz=2 sign=1 p=0 w=1
68
+STP 10 101 0 011 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
69
+LDP 10 101 0 011 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
70
+STP_v 00 101 1 011 0 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=1
71
+LDP_v 00 101 1 011 1 ....... ..... ..... ..... @ldstpair sz=2 sign=0 p=0 w=1
72
+STP_v 01 101 1 011 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
73
+LDP_v 01 101 1 011 1 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
74
+STP_v 10 101 1 011 0 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=1
75
+LDP_v 10 101 1 011 1 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p=0 w=1
76
+
77
+# STGP: store tag and pair
78
+STGP 01 101 0 001 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
79
+STGP 01 101 0 010 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
80
+STGP 01 101 0 011 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
81
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
82
index XXXXXXX..XXXXXXX 100644
83
--- a/target/arm/tcg/translate-a64.c
84
+++ b/target/arm/tcg/translate-a64.c
85
@@ -XXX,XX +XXX,XX @@ static bool trans_LD_lit_v(DisasContext *s, arg_ldlit *a)
86
return true;
41
}
87
}
42
88
43
+/**
89
-/*
44
+ * get_phys_addr_lpae: perform one stage of page table walk, LPAE format
90
- * LDNP (Load Pair - non-temporal hint)
45
+ *
91
- * LDP (Load Pair - non vector)
46
+ * Returns false if the translation was successful. Otherwise, phys_ptr, attrs,
92
- * LDPSW (Load Pair Signed Word - non vector)
47
+ * prot and page_size may not be filled in, and the populated fsr value provides
93
- * STNP (Store Pair - non-temporal hint)
48
+ * information on why the translation aborted, in the format of a long-format
94
- * STP (Store Pair - non vector)
49
+ * DFSR/IFSR fault register, with the following caveats:
95
- * LDNP (Load Pair of SIMD&FP - non-temporal hint)
50
+ * * the WnR bit is never set (the caller must do this).
96
- * LDP (Load Pair of SIMD&FP)
51
+ *
97
- * STNP (Store Pair of SIMD&FP - non-temporal hint)
52
+ * @env: CPUARMState
98
- * STP (Store Pair of SIMD&FP)
53
+ * @address: virtual address to get physical address for
99
- *
54
+ * @access_type: MMU_DATA_LOAD, MMU_DATA_STORE or MMU_INST_FETCH
100
- * 31 30 29 27 26 25 24 23 22 21 15 14 10 9 5 4 0
55
+ * @mmu_idx: MMU index indicating required translation regime
101
- * +-----+-------+---+---+-------+---+-----------------------------+
56
+ * @s1_is_el0: if @mmu_idx is ARMMMUIdx_Stage2 (so this is a stage 2 page table
102
- * | opc | 1 0 1 | V | 0 | index | L | imm7 | Rt2 | Rn | Rt |
57
+ * walk), must be true if this is stage 2 of a stage 1+2 walk for an
103
- * +-----+-------+---+---+-------+---+-------+-------+------+------+
58
+ * EL0 access). If @mmu_idx is anything else, @s1_is_el0 is ignored.
104
- *
59
+ * @phys_ptr: set to the physical address corresponding to the virtual address
105
- * opc: LDP/STP/LDNP/STNP 00 -> 32 bit, 10 -> 64 bit
60
+ * @attrs: set to the memory transaction attributes to use
106
- * LDPSW/STGP 01
61
+ * @prot: set to the permissions for the page containing phys_ptr
107
- * LDP/STP/LDNP/STNP (SIMD) 00 -> 32 bit, 01 -> 64 bit, 10 -> 128 bit
62
+ * @page_size_ptr: set to the size of the page containing phys_ptr
108
- * V: 0 -> GPR, 1 -> Vector
63
+ * @fi: set to fault info if the translation fails
109
- * idx: 00 -> signed offset with non-temporal hint, 01 -> post-index,
64
+ * @cacheattrs: (if non-NULL) set to the cacheability/shareability attributes
110
- * 10 -> signed offset, 11 -> pre-index
65
+ */
111
- * L: 0 -> Store 1 -> Load
66
static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
112
- *
67
MMUAccessType access_type, ARMMMUIdx mmu_idx,
113
- * Rt, Rt2 = GPR or SIMD registers to be stored
68
+ bool s1_is_el0,
114
- * Rn = general purpose register containing address
69
hwaddr *phys_ptr, MemTxAttrs *txattrs, int *prot,
115
- * imm7 = signed offset (multiple of 4 or 8 depending on size)
70
target_ulong *page_size_ptr,
116
- */
71
ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs)
117
-static void disas_ldst_pair(DisasContext *s, uint32_t insn)
72
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
118
+static void op_addr_ldstpair_pre(DisasContext *s, arg_ldstpair *a,
73
119
+ TCGv_i64 *clean_addr, TCGv_i64 *dirty_addr,
74
/* S1 is done. Now do S2 translation. */
120
+ uint64_t offset, bool is_store, MemOp mop)
75
ret = get_phys_addr_lpae(env, ipa, access_type, ARMMMUIdx_Stage2,
121
{
76
+ mmu_idx == ARMMMUIdx_E10_0,
122
- int rt = extract32(insn, 0, 5);
77
phys_ptr, attrs, &s2_prot,
123
- int rn = extract32(insn, 5, 5);
78
page_size, fi,
124
- int rt2 = extract32(insn, 10, 5);
79
cacheattrs != NULL ? &cacheattrs2 : NULL);
125
- uint64_t offset = sextract64(insn, 15, 7);
80
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
126
- int index = extract32(insn, 23, 2);
127
- bool is_vector = extract32(insn, 26, 1);
128
- bool is_load = extract32(insn, 22, 1);
129
- int opc = extract32(insn, 30, 2);
130
- bool is_signed = false;
131
- bool postindex = false;
132
- bool wback = false;
133
- bool set_tag = false;
134
- TCGv_i64 clean_addr, dirty_addr;
135
- MemOp mop;
136
- int size;
137
-
138
- if (opc == 3) {
139
- unallocated_encoding(s);
140
- return;
141
- }
142
-
143
- if (is_vector) {
144
- size = 2 + opc;
145
- } else if (opc == 1 && !is_load) {
146
- /* STGP */
147
- if (!dc_isar_feature(aa64_mte_insn_reg, s) || index == 0) {
148
- unallocated_encoding(s);
149
- return;
150
- }
151
- size = 3;
152
- set_tag = true;
153
- } else {
154
- size = 2 + extract32(opc, 1, 1);
155
- is_signed = extract32(opc, 0, 1);
156
- if (!is_load && is_signed) {
157
- unallocated_encoding(s);
158
- return;
159
- }
160
- }
161
-
162
- switch (index) {
163
- case 1: /* post-index */
164
- postindex = true;
165
- wback = true;
166
- break;
167
- case 0:
168
- /* signed offset with "non-temporal" hint. Since we don't emulate
169
- * caches we don't care about hints to the cache system about
170
- * data access patterns, and handle this identically to plain
171
- * signed offset.
172
- */
173
- if (is_signed) {
174
- /* There is no non-temporal-hint version of LDPSW */
175
- unallocated_encoding(s);
176
- return;
177
- }
178
- postindex = false;
179
- break;
180
- case 2: /* signed offset, rn not updated */
181
- postindex = false;
182
- break;
183
- case 3: /* pre-index */
184
- postindex = false;
185
- wback = true;
186
- break;
187
- }
188
-
189
- if (is_vector && !fp_access_check(s)) {
190
- return;
191
- }
192
-
193
- offset <<= (set_tag ? LOG2_TAG_GRANULE : size);
194
-
195
- if (rn == 31) {
196
+ if (a->rn == 31) {
197
gen_check_sp_alignment(s);
81
}
198
}
82
199
83
if (regime_using_lpae_format(env, mmu_idx)) {
200
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
84
- return get_phys_addr_lpae(env, address, access_type, mmu_idx,
201
- if (!postindex) {
85
+ return get_phys_addr_lpae(env, address, access_type, mmu_idx, false,
202
+ *dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
86
phys_ptr, attrs, prot, page_size,
203
+ if (!a->p) {
87
fi, cacheattrs);
204
+ tcg_gen_addi_i64(*dirty_addr, *dirty_addr, offset);
88
} else if (regime_sctlr(env, mmu_idx) & SCTLR_XP) {
205
+ }
206
+
207
+ *clean_addr = gen_mte_checkN(s, *dirty_addr, is_store,
208
+ (a->w || a->rn != 31), 2 << a->sz, mop);
209
+}
210
+
211
+static void op_addr_ldstpair_post(DisasContext *s, arg_ldstpair *a,
212
+ TCGv_i64 dirty_addr, uint64_t offset)
213
+{
214
+ if (a->w) {
215
+ if (a->p) {
216
+ tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
217
+ }
218
+ tcg_gen_mov_i64(cpu_reg_sp(s, a->rn), dirty_addr);
219
+ }
220
+}
221
+
222
+static bool trans_STP(DisasContext *s, arg_ldstpair *a)
223
+{
224
+ uint64_t offset = a->imm << a->sz;
225
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt, tcg_rt2;
226
+ MemOp mop = finalize_memop(s, a->sz);
227
+
228
+ op_addr_ldstpair_pre(s, a, &clean_addr, &dirty_addr, offset, true, mop);
229
+ tcg_rt = cpu_reg(s, a->rt);
230
+ tcg_rt2 = cpu_reg(s, a->rt2);
231
+ /*
232
+ * We built mop above for the single logical access -- rebuild it
233
+ * now for the paired operation.
234
+ *
235
+ * With LSE2, non-sign-extending pairs are treated atomically if
236
+ * aligned, and if unaligned one of the pair will be completely
237
+ * within a 16-byte block and that element will be atomic.
238
+ * Otherwise each element is separately atomic.
239
+ * In all cases, issue one operation with the correct atomicity.
240
+ */
241
+ mop = a->sz + 1;
242
+ if (s->align_mem) {
243
+ mop |= (a->sz == 2 ? MO_ALIGN_4 : MO_ALIGN_8);
244
+ }
245
+ mop = finalize_memop_pair(s, mop);
246
+ if (a->sz == 2) {
247
+ TCGv_i64 tmp = tcg_temp_new_i64();
248
+
249
+ if (s->be_data == MO_LE) {
250
+ tcg_gen_concat32_i64(tmp, tcg_rt, tcg_rt2);
251
+ } else {
252
+ tcg_gen_concat32_i64(tmp, tcg_rt2, tcg_rt);
253
+ }
254
+ tcg_gen_qemu_st_i64(tmp, clean_addr, get_mem_index(s), mop);
255
+ } else {
256
+ TCGv_i128 tmp = tcg_temp_new_i128();
257
+
258
+ if (s->be_data == MO_LE) {
259
+ tcg_gen_concat_i64_i128(tmp, tcg_rt, tcg_rt2);
260
+ } else {
261
+ tcg_gen_concat_i64_i128(tmp, tcg_rt2, tcg_rt);
262
+ }
263
+ tcg_gen_qemu_st_i128(tmp, clean_addr, get_mem_index(s), mop);
264
+ }
265
+ op_addr_ldstpair_post(s, a, dirty_addr, offset);
266
+ return true;
267
+}
268
+
269
+static bool trans_LDP(DisasContext *s, arg_ldstpair *a)
270
+{
271
+ uint64_t offset = a->imm << a->sz;
272
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt, tcg_rt2;
273
+ MemOp mop = finalize_memop(s, a->sz);
274
+
275
+ op_addr_ldstpair_pre(s, a, &clean_addr, &dirty_addr, offset, false, mop);
276
+ tcg_rt = cpu_reg(s, a->rt);
277
+ tcg_rt2 = cpu_reg(s, a->rt2);
278
+
279
+ /*
280
+ * We built mop above for the single logical access -- rebuild it
281
+ * now for the paired operation.
282
+ *
283
+ * With LSE2, non-sign-extending pairs are treated atomically if
284
+ * aligned, and if unaligned one of the pair will be completely
285
+ * within a 16-byte block and that element will be atomic.
286
+ * Otherwise each element is separately atomic.
287
+ * In all cases, issue one operation with the correct atomicity.
288
+ *
289
+ * This treats sign-extending loads like zero-extending loads,
290
+ * since that reuses the most code below.
291
+ */
292
+ mop = a->sz + 1;
293
+ if (s->align_mem) {
294
+ mop |= (a->sz == 2 ? MO_ALIGN_4 : MO_ALIGN_8);
295
+ }
296
+ mop = finalize_memop_pair(s, mop);
297
+ if (a->sz == 2) {
298
+ int o2 = s->be_data == MO_LE ? 32 : 0;
299
+ int o1 = o2 ^ 32;
300
+
301
+ tcg_gen_qemu_ld_i64(tcg_rt, clean_addr, get_mem_index(s), mop);
302
+ if (a->sign) {
303
+ tcg_gen_sextract_i64(tcg_rt2, tcg_rt, o2, 32);
304
+ tcg_gen_sextract_i64(tcg_rt, tcg_rt, o1, 32);
305
+ } else {
306
+ tcg_gen_extract_i64(tcg_rt2, tcg_rt, o2, 32);
307
+ tcg_gen_extract_i64(tcg_rt, tcg_rt, o1, 32);
308
+ }
309
+ } else {
310
+ TCGv_i128 tmp = tcg_temp_new_i128();
311
+
312
+ tcg_gen_qemu_ld_i128(tmp, clean_addr, get_mem_index(s), mop);
313
+ if (s->be_data == MO_LE) {
314
+ tcg_gen_extr_i128_i64(tcg_rt, tcg_rt2, tmp);
315
+ } else {
316
+ tcg_gen_extr_i128_i64(tcg_rt2, tcg_rt, tmp);
317
+ }
318
+ }
319
+ op_addr_ldstpair_post(s, a, dirty_addr, offset);
320
+ return true;
321
+}
322
+
323
+static bool trans_STP_v(DisasContext *s, arg_ldstpair *a)
324
+{
325
+ uint64_t offset = a->imm << a->sz;
326
+ TCGv_i64 clean_addr, dirty_addr;
327
+ MemOp mop;
328
+
329
+ if (!fp_access_check(s)) {
330
+ return true;
331
+ }
332
+
333
+ /* LSE2 does not merge FP pairs; leave these as separate operations. */
334
+ mop = finalize_memop_asimd(s, a->sz);
335
+ op_addr_ldstpair_pre(s, a, &clean_addr, &dirty_addr, offset, true, mop);
336
+ do_fp_st(s, a->rt, clean_addr, mop);
337
+ tcg_gen_addi_i64(clean_addr, clean_addr, 1 << a->sz);
338
+ do_fp_st(s, a->rt2, clean_addr, mop);
339
+ op_addr_ldstpair_post(s, a, dirty_addr, offset);
340
+ return true;
341
+}
342
+
343
+static bool trans_LDP_v(DisasContext *s, arg_ldstpair *a)
344
+{
345
+ uint64_t offset = a->imm << a->sz;
346
+ TCGv_i64 clean_addr, dirty_addr;
347
+ MemOp mop;
348
+
349
+ if (!fp_access_check(s)) {
350
+ return true;
351
+ }
352
+
353
+ /* LSE2 does not merge FP pairs; leave these as separate operations. */
354
+ mop = finalize_memop_asimd(s, a->sz);
355
+ op_addr_ldstpair_pre(s, a, &clean_addr, &dirty_addr, offset, false, mop);
356
+ do_fp_ld(s, a->rt, clean_addr, mop);
357
+ tcg_gen_addi_i64(clean_addr, clean_addr, 1 << a->sz);
358
+ do_fp_ld(s, a->rt2, clean_addr, mop);
359
+ op_addr_ldstpair_post(s, a, dirty_addr, offset);
360
+ return true;
361
+}
362
+
363
+static bool trans_STGP(DisasContext *s, arg_ldstpair *a)
364
+{
365
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt, tcg_rt2;
366
+ uint64_t offset = a->imm << LOG2_TAG_GRANULE;
367
+ MemOp mop;
368
+ TCGv_i128 tmp;
369
+
370
+ if (!dc_isar_feature(aa64_mte_insn_reg, s)) {
371
+ return false;
372
+ }
373
+
374
+ if (a->rn == 31) {
375
+ gen_check_sp_alignment(s);
376
+ }
377
+
378
+ dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
379
+ if (!a->p) {
380
tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
381
}
382
383
- if (set_tag) {
384
- if (!s->ata) {
385
- /*
386
- * TODO: We could rely on the stores below, at least for
387
- * system mode, if we arrange to add MO_ALIGN_16.
388
- */
389
- gen_helper_stg_stub(cpu_env, dirty_addr);
390
- } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
391
- gen_helper_stg_parallel(cpu_env, dirty_addr, dirty_addr);
392
- } else {
393
- gen_helper_stg(cpu_env, dirty_addr, dirty_addr);
394
- }
395
- }
396
-
397
- if (is_vector) {
398
- mop = finalize_memop_asimd(s, size);
399
- } else {
400
- mop = finalize_memop(s, size);
401
- }
402
- clean_addr = gen_mte_checkN(s, dirty_addr, !is_load,
403
- (wback || rn != 31) && !set_tag,
404
- 2 << size, mop);
405
-
406
- if (is_vector) {
407
- /* LSE2 does not merge FP pairs; leave these as separate operations. */
408
- if (is_load) {
409
- do_fp_ld(s, rt, clean_addr, mop);
410
- } else {
411
- do_fp_st(s, rt, clean_addr, mop);
412
- }
413
- tcg_gen_addi_i64(clean_addr, clean_addr, 1 << size);
414
- if (is_load) {
415
- do_fp_ld(s, rt2, clean_addr, mop);
416
- } else {
417
- do_fp_st(s, rt2, clean_addr, mop);
418
- }
419
- } else {
420
- TCGv_i64 tcg_rt = cpu_reg(s, rt);
421
- TCGv_i64 tcg_rt2 = cpu_reg(s, rt2);
422
-
423
+ if (!s->ata) {
424
/*
425
- * We built mop above for the single logical access -- rebuild it
426
- * now for the paired operation.
427
- *
428
- * With LSE2, non-sign-extending pairs are treated atomically if
429
- * aligned, and if unaligned one of the pair will be completely
430
- * within a 16-byte block and that element will be atomic.
431
- * Otherwise each element is separately atomic.
432
- * In all cases, issue one operation with the correct atomicity.
433
- *
434
- * This treats sign-extending loads like zero-extending loads,
435
- * since that reuses the most code below.
436
+ * TODO: We could rely on the stores below, at least for
437
+ * system mode, if we arrange to add MO_ALIGN_16.
438
*/
439
- mop = size + 1;
440
- if (s->align_mem) {
441
- mop |= (size == 2 ? MO_ALIGN_4 : MO_ALIGN_8);
442
- }
443
- mop = finalize_memop_pair(s, mop);
444
-
445
- if (is_load) {
446
- if (size == 2) {
447
- int o2 = s->be_data == MO_LE ? 32 : 0;
448
- int o1 = o2 ^ 32;
449
-
450
- tcg_gen_qemu_ld_i64(tcg_rt, clean_addr, get_mem_index(s), mop);
451
- if (is_signed) {
452
- tcg_gen_sextract_i64(tcg_rt2, tcg_rt, o2, 32);
453
- tcg_gen_sextract_i64(tcg_rt, tcg_rt, o1, 32);
454
- } else {
455
- tcg_gen_extract_i64(tcg_rt2, tcg_rt, o2, 32);
456
- tcg_gen_extract_i64(tcg_rt, tcg_rt, o1, 32);
457
- }
458
- } else {
459
- TCGv_i128 tmp = tcg_temp_new_i128();
460
-
461
- tcg_gen_qemu_ld_i128(tmp, clean_addr, get_mem_index(s), mop);
462
- if (s->be_data == MO_LE) {
463
- tcg_gen_extr_i128_i64(tcg_rt, tcg_rt2, tmp);
464
- } else {
465
- tcg_gen_extr_i128_i64(tcg_rt2, tcg_rt, tmp);
466
- }
467
- }
468
- } else {
469
- if (size == 2) {
470
- TCGv_i64 tmp = tcg_temp_new_i64();
471
-
472
- if (s->be_data == MO_LE) {
473
- tcg_gen_concat32_i64(tmp, tcg_rt, tcg_rt2);
474
- } else {
475
- tcg_gen_concat32_i64(tmp, tcg_rt2, tcg_rt);
476
- }
477
- tcg_gen_qemu_st_i64(tmp, clean_addr, get_mem_index(s), mop);
478
- } else {
479
- TCGv_i128 tmp = tcg_temp_new_i128();
480
-
481
- if (s->be_data == MO_LE) {
482
- tcg_gen_concat_i64_i128(tmp, tcg_rt, tcg_rt2);
483
- } else {
484
- tcg_gen_concat_i64_i128(tmp, tcg_rt2, tcg_rt);
485
- }
486
- tcg_gen_qemu_st_i128(tmp, clean_addr, get_mem_index(s), mop);
487
- }
488
- }
489
+ gen_helper_stg_stub(cpu_env, dirty_addr);
490
+ } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
491
+ gen_helper_stg_parallel(cpu_env, dirty_addr, dirty_addr);
492
+ } else {
493
+ gen_helper_stg(cpu_env, dirty_addr, dirty_addr);
494
}
495
496
- if (wback) {
497
- if (postindex) {
498
- tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
499
- }
500
- tcg_gen_mov_i64(cpu_reg_sp(s, rn), dirty_addr);
501
+ mop = finalize_memop(s, a->sz);
502
+ clean_addr = gen_mte_checkN(s, dirty_addr, true, false, 2 << a->sz, mop);
503
+
504
+ tcg_rt = cpu_reg(s, a->rt);
505
+ tcg_rt2 = cpu_reg(s, a->rt2);
506
+
507
+ assert(a->sz == 3);
508
+
509
+ tmp = tcg_temp_new_i128();
510
+ if (s->be_data == MO_LE) {
511
+ tcg_gen_concat_i64_i128(tmp, tcg_rt, tcg_rt2);
512
+ } else {
513
+ tcg_gen_concat_i64_i128(tmp, tcg_rt2, tcg_rt);
514
}
515
+ tcg_gen_qemu_st_i128(tmp, clean_addr, get_mem_index(s), mop);
516
+
517
+ op_addr_ldstpair_post(s, a, dirty_addr, offset);
518
+ return true;
519
}
520
521
/*
522
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
523
static void disas_ldst(DisasContext *s, uint32_t insn)
524
{
525
switch (extract32(insn, 24, 6)) {
526
- case 0x28: case 0x29:
527
- case 0x2c: case 0x2d: /* Load/store pair (all forms) */
528
- disas_ldst_pair(s, insn);
529
- break;
530
case 0x38: case 0x39:
531
case 0x3c: case 0x3d: /* Load/store register (all forms) */
532
disas_ldst_reg(s, insn);
89
--
533
--
90
2.20.1
534
2.34.1
91
92
diff view generated by jsdifflib
1
Convert the V[US]DOT (vector) insns to decodetree.
1
Convert the load and store instructions which use a 9-bit
2
immediate offset to decodetree.
2
3
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200430181003.21682-7-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-13-peter.maydell@linaro.org
6
---
7
---
7
target/arm/neon-shared.decode | 4 ++++
8
target/arm/tcg/a64.decode | 69 +++++++++++
8
target/arm/translate-neon.inc.c | 32 ++++++++++++++++++++++++++++++++
9
target/arm/tcg/translate-a64.c | 206 ++++++++++++++-------------------
9
target/arm/translate.c | 9 +--------
10
2 files changed, 153 insertions(+), 122 deletions(-)
10
3 files changed, 37 insertions(+), 8 deletions(-)
11
11
12
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/neon-shared.decode
14
--- a/target/arm/tcg/a64.decode
15
+++ b/target/arm/neon-shared.decode
15
+++ b/target/arm/tcg/a64.decode
16
@@ -XXX,XX +XXX,XX @@ VCMLA 1111 110 rot:2 . 1 size:1 .... .... 1000 . q:1 . 0 .... \
16
@@ -XXX,XX +XXX,XX @@ LDP_v 10 101 1 011 1 ....... ..... ..... ..... @ldstpair sz=4 sign=0 p
17
17
STGP 01 101 0 001 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=1 w=1
18
VCADD 1111 110 rot:1 1 . 0 size:1 .... .... 1000 . q:1 . 0 .... \
18
STGP 01 101 0 010 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=0
19
vm=%vm_dp vn=%vn_dp vd=%vd_dp
19
STGP 01 101 0 011 0 ....... ..... ..... ..... @ldstpair sz=3 sign=0 p=0 w=1
20
+
20
+
21
+# VUDOT and VSDOT
21
+# Load/store register (unscaled immediate)
22
+VDOT 1111 110 00 . 10 .... .... 1101 . q:1 . u:1 .... \
22
+&ldst_imm rt rn imm sz sign w p unpriv ext
23
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp
23
+@ldst_imm .. ... . .. .. . imm:s9 .. rn:5 rt:5 &ldst_imm unpriv=0 p=0 w=0
24
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
24
+@ldst_imm_pre .. ... . .. .. . imm:s9 .. rn:5 rt:5 &ldst_imm unpriv=0 p=0 w=1
25
+@ldst_imm_post .. ... . .. .. . imm:s9 .. rn:5 rt:5 &ldst_imm unpriv=0 p=1 w=1
26
+@ldst_imm_user .. ... . .. .. . imm:s9 .. rn:5 rt:5 &ldst_imm unpriv=1 p=0 w=0
27
+
28
+STR_i sz:2 111 0 00 00 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0
29
+LDR_i 00 111 0 00 01 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=1 sz=0
30
+LDR_i 01 111 0 00 01 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=1 sz=1
31
+LDR_i 10 111 0 00 01 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=1 sz=2
32
+LDR_i 11 111 0 00 01 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0 sz=3
33
+LDR_i 00 111 0 00 10 0 ......... 00 ..... ..... @ldst_imm sign=1 ext=0 sz=0
34
+LDR_i 01 111 0 00 10 0 ......... 00 ..... ..... @ldst_imm sign=1 ext=0 sz=1
35
+LDR_i 10 111 0 00 10 0 ......... 00 ..... ..... @ldst_imm sign=1 ext=0 sz=2
36
+LDR_i 00 111 0 00 11 0 ......... 00 ..... ..... @ldst_imm sign=1 ext=1 sz=0
37
+LDR_i 01 111 0 00 11 0 ......... 00 ..... ..... @ldst_imm sign=1 ext=1 sz=1
38
+
39
+STR_i sz:2 111 0 00 00 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0
40
+LDR_i 00 111 0 00 01 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=1 sz=0
41
+LDR_i 01 111 0 00 01 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=1 sz=1
42
+LDR_i 10 111 0 00 01 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=1 sz=2
43
+LDR_i 11 111 0 00 01 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0 sz=3
44
+LDR_i 00 111 0 00 10 0 ......... 01 ..... ..... @ldst_imm_post sign=1 ext=0 sz=0
45
+LDR_i 01 111 0 00 10 0 ......... 01 ..... ..... @ldst_imm_post sign=1 ext=0 sz=1
46
+LDR_i 10 111 0 00 10 0 ......... 01 ..... ..... @ldst_imm_post sign=1 ext=0 sz=2
47
+LDR_i 00 111 0 00 11 0 ......... 01 ..... ..... @ldst_imm_post sign=1 ext=1 sz=0
48
+LDR_i 01 111 0 00 11 0 ......... 01 ..... ..... @ldst_imm_post sign=1 ext=1 sz=1
49
+
50
+STR_i sz:2 111 0 00 00 0 ......... 10 ..... ..... @ldst_imm_user sign=0 ext=0
51
+LDR_i 00 111 0 00 01 0 ......... 10 ..... ..... @ldst_imm_user sign=0 ext=1 sz=0
52
+LDR_i 01 111 0 00 01 0 ......... 10 ..... ..... @ldst_imm_user sign=0 ext=1 sz=1
53
+LDR_i 10 111 0 00 01 0 ......... 10 ..... ..... @ldst_imm_user sign=0 ext=1 sz=2
54
+LDR_i 11 111 0 00 01 0 ......... 10 ..... ..... @ldst_imm_user sign=0 ext=0 sz=3
55
+LDR_i 00 111 0 00 10 0 ......... 10 ..... ..... @ldst_imm_user sign=1 ext=0 sz=0
56
+LDR_i 01 111 0 00 10 0 ......... 10 ..... ..... @ldst_imm_user sign=1 ext=0 sz=1
57
+LDR_i 10 111 0 00 10 0 ......... 10 ..... ..... @ldst_imm_user sign=1 ext=0 sz=2
58
+LDR_i 00 111 0 00 11 0 ......... 10 ..... ..... @ldst_imm_user sign=1 ext=1 sz=0
59
+LDR_i 01 111 0 00 11 0 ......... 10 ..... ..... @ldst_imm_user sign=1 ext=1 sz=1
60
+
61
+STR_i sz:2 111 0 00 00 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0
62
+LDR_i 00 111 0 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=1 sz=0
63
+LDR_i 01 111 0 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=1 sz=1
64
+LDR_i 10 111 0 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=1 sz=2
65
+LDR_i 11 111 0 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0 sz=3
66
+LDR_i 00 111 0 00 10 0 ......... 11 ..... ..... @ldst_imm_pre sign=1 ext=0 sz=0
67
+LDR_i 01 111 0 00 10 0 ......... 11 ..... ..... @ldst_imm_pre sign=1 ext=0 sz=1
68
+LDR_i 10 111 0 00 10 0 ......... 11 ..... ..... @ldst_imm_pre sign=1 ext=0 sz=2
69
+LDR_i 00 111 0 00 11 0 ......... 11 ..... ..... @ldst_imm_pre sign=1 ext=1 sz=0
70
+LDR_i 01 111 0 00 11 0 ......... 11 ..... ..... @ldst_imm_pre sign=1 ext=1 sz=1
71
+
72
+# PRFM : prefetch memory: a no-op for QEMU
73
+NOP 11 111 0 00 10 0 --------- 00 ----- -----
74
+
75
+STR_v_i sz:2 111 1 00 00 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0
76
+STR_v_i 00 111 1 00 10 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0 sz=4
77
+LDR_v_i sz:2 111 1 00 01 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0
78
+LDR_v_i 00 111 1 00 11 0 ......... 00 ..... ..... @ldst_imm sign=0 ext=0 sz=4
79
+
80
+STR_v_i sz:2 111 1 00 00 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0
81
+STR_v_i 00 111 1 00 10 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0 sz=4
82
+LDR_v_i sz:2 111 1 00 01 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0
83
+LDR_v_i 00 111 1 00 11 0 ......... 01 ..... ..... @ldst_imm_post sign=0 ext=0 sz=4
84
+
85
+STR_v_i sz:2 111 1 00 00 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0
86
+STR_v_i 00 111 1 00 10 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0 sz=4
87
+LDR_v_i sz:2 111 1 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0
88
+LDR_v_i 00 111 1 00 11 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0 sz=4
89
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
25
index XXXXXXX..XXXXXXX 100644
90
index XXXXXXX..XXXXXXX 100644
26
--- a/target/arm/translate-neon.inc.c
91
--- a/target/arm/tcg/translate-a64.c
27
+++ b/target/arm/translate-neon.inc.c
92
+++ b/target/arm/tcg/translate-a64.c
28
@@ -XXX,XX +XXX,XX @@ static bool trans_VCADD(DisasContext *s, arg_VCADD *a)
93
@@ -XXX,XX +XXX,XX @@ static bool trans_STGP(DisasContext *s, arg_ldstpair *a)
29
tcg_temp_free_ptr(fpst);
30
return true;
94
return true;
31
}
95
}
32
+
96
33
+static bool trans_VDOT(DisasContext *s, arg_VDOT *a)
97
-/*
34
+{
98
- * Load/store (immediate post-indexed)
35
+ int opr_sz;
99
- * Load/store (immediate pre-indexed)
36
+ gen_helper_gvec_3 *fn_gvec;
100
- * Load/store (unscaled immediate)
37
+
101
- *
38
+ if (!dc_isar_feature(aa32_dp, s)) {
102
- * 31 30 29 27 26 25 24 23 22 21 20 12 11 10 9 5 4 0
39
+ return false;
103
- * +----+-------+---+-----+-----+---+--------+-----+------+------+
40
+ }
104
- * |size| 1 1 1 | V | 0 0 | opc | 0 | imm9 | idx | Rn | Rt |
41
+
105
- * +----+-------+---+-----+-----+---+--------+-----+------+------+
42
+ /* UNDEF accesses to D16-D31 if they don't exist. */
106
- *
43
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
107
- * idx = 01 -> post-indexed, 11 pre-indexed, 00 unscaled imm. (no writeback)
44
+ ((a->vd | a->vn | a->vm) & 0x10)) {
108
- 10 -> unprivileged
45
+ return false;
109
- * V = 0 -> non-vector
46
+ }
110
- * size: 00 -> 8 bit, 01 -> 16 bit, 10 -> 32 bit, 11 -> 64bit
47
+
111
- * opc: 00 -> store, 01 -> loadu, 10 -> loads 64, 11 -> loads 32
48
+ if ((a->vn | a->vm | a->vd) & a->q) {
112
- */
49
+ return false;
113
-static void disas_ldst_reg_imm9(DisasContext *s, uint32_t insn,
50
+ }
114
- int opc,
51
+
115
- int size,
52
+ if (!vfp_access_check(s)) {
116
- int rt,
117
- bool is_vector)
118
+static void op_addr_ldst_imm_pre(DisasContext *s, arg_ldst_imm *a,
119
+ TCGv_i64 *clean_addr, TCGv_i64 *dirty_addr,
120
+ uint64_t offset, bool is_store, MemOp mop)
121
{
122
- int rn = extract32(insn, 5, 5);
123
- int imm9 = sextract32(insn, 12, 9);
124
- int idx = extract32(insn, 10, 2);
125
- bool is_signed = false;
126
- bool is_store = false;
127
- bool is_extended = false;
128
- bool is_unpriv = (idx == 2);
129
- bool iss_valid;
130
- bool post_index;
131
- bool writeback;
132
int memidx;
133
- MemOp memop;
134
- TCGv_i64 clean_addr, dirty_addr;
135
136
- if (is_vector) {
137
- size |= (opc & 2) << 1;
138
- if (size > 4 || is_unpriv) {
139
- unallocated_encoding(s);
140
- return;
141
- }
142
- is_store = ((opc & 1) == 0);
143
- if (!fp_access_check(s)) {
144
- return;
145
- }
146
- memop = finalize_memop_asimd(s, size);
147
- } else {
148
- if (size == 3 && opc == 2) {
149
- /* PRFM - prefetch */
150
- if (idx != 0) {
151
- unallocated_encoding(s);
152
- return;
153
- }
154
- return;
155
- }
156
- if (opc == 3 && size > 1) {
157
- unallocated_encoding(s);
158
- return;
159
- }
160
- is_store = (opc == 0);
161
- is_signed = !is_store && extract32(opc, 1, 1);
162
- is_extended = (size < 3) && extract32(opc, 0, 1);
163
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
164
- }
165
-
166
- switch (idx) {
167
- case 0:
168
- case 2:
169
- post_index = false;
170
- writeback = false;
171
- break;
172
- case 1:
173
- post_index = true;
174
- writeback = true;
175
- break;
176
- case 3:
177
- post_index = false;
178
- writeback = true;
179
- break;
180
- default:
181
- g_assert_not_reached();
182
- }
183
-
184
- iss_valid = !is_vector && !writeback;
185
-
186
- if (rn == 31) {
187
+ if (a->rn == 31) {
188
gen_check_sp_alignment(s);
189
}
190
191
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
192
- if (!post_index) {
193
- tcg_gen_addi_i64(dirty_addr, dirty_addr, imm9);
194
+ *dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
195
+ if (!a->p) {
196
+ tcg_gen_addi_i64(*dirty_addr, *dirty_addr, offset);
197
}
198
+ memidx = a->unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
199
+ *clean_addr = gen_mte_check1_mmuidx(s, *dirty_addr, is_store,
200
+ a->w || a->rn != 31,
201
+ mop, a->unpriv, memidx);
202
+}
203
204
- memidx = is_unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
205
-
206
- clean_addr = gen_mte_check1_mmuidx(s, dirty_addr, is_store,
207
- writeback || rn != 31,
208
- memop, is_unpriv, memidx);
209
-
210
- if (is_vector) {
211
- if (is_store) {
212
- do_fp_st(s, rt, clean_addr, memop);
213
- } else {
214
- do_fp_ld(s, rt, clean_addr, memop);
215
- }
216
- } else {
217
- TCGv_i64 tcg_rt = cpu_reg(s, rt);
218
- bool iss_sf = disas_ldst_compute_iss_sf(size, is_signed, opc);
219
-
220
- if (is_store) {
221
- do_gpr_st_memidx(s, tcg_rt, clean_addr, memop, memidx,
222
- iss_valid, rt, iss_sf, false);
223
- } else {
224
- do_gpr_ld_memidx(s, tcg_rt, clean_addr, memop,
225
- is_extended, memidx,
226
- iss_valid, rt, iss_sf, false);
227
+static void op_addr_ldst_imm_post(DisasContext *s, arg_ldst_imm *a,
228
+ TCGv_i64 dirty_addr, uint64_t offset)
229
+{
230
+ if (a->w) {
231
+ if (a->p) {
232
+ tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
233
}
234
+ tcg_gen_mov_i64(cpu_reg_sp(s, a->rn), dirty_addr);
235
}
236
+}
237
238
- if (writeback) {
239
- TCGv_i64 tcg_rn = cpu_reg_sp(s, rn);
240
- if (post_index) {
241
- tcg_gen_addi_i64(dirty_addr, dirty_addr, imm9);
242
- }
243
- tcg_gen_mov_i64(tcg_rn, dirty_addr);
244
+static bool trans_STR_i(DisasContext *s, arg_ldst_imm *a)
245
+{
246
+ bool iss_sf, iss_valid = !a->w;
247
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt;
248
+ int memidx = a->unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
249
+ MemOp mop = finalize_memop(s, a->sz + a->sign * MO_SIGN);
250
+
251
+ op_addr_ldst_imm_pre(s, a, &clean_addr, &dirty_addr, a->imm, true, mop);
252
+
253
+ tcg_rt = cpu_reg(s, a->rt);
254
+ iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
255
+
256
+ do_gpr_st_memidx(s, tcg_rt, clean_addr, mop, memidx,
257
+ iss_valid, a->rt, iss_sf, false);
258
+ op_addr_ldst_imm_post(s, a, dirty_addr, a->imm);
259
+ return true;
260
+}
261
+
262
+static bool trans_LDR_i(DisasContext *s, arg_ldst_imm *a)
263
+{
264
+ bool iss_sf, iss_valid = !a->w;
265
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt;
266
+ int memidx = a->unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
267
+ MemOp mop = finalize_memop(s, a->sz + a->sign * MO_SIGN);
268
+
269
+ op_addr_ldst_imm_pre(s, a, &clean_addr, &dirty_addr, a->imm, false, mop);
270
+
271
+ tcg_rt = cpu_reg(s, a->rt);
272
+ iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
273
+
274
+ do_gpr_ld_memidx(s, tcg_rt, clean_addr, mop,
275
+ a->ext, memidx, iss_valid, a->rt, iss_sf, false);
276
+ op_addr_ldst_imm_post(s, a, dirty_addr, a->imm);
277
+ return true;
278
+}
279
+
280
+static bool trans_STR_v_i(DisasContext *s, arg_ldst_imm *a)
281
+{
282
+ TCGv_i64 clean_addr, dirty_addr;
283
+ MemOp mop;
284
+
285
+ if (!fp_access_check(s)) {
286
+ return true;
287
}
288
+ mop = finalize_memop_asimd(s, a->sz);
289
+ op_addr_ldst_imm_pre(s, a, &clean_addr, &dirty_addr, a->imm, true, mop);
290
+ do_fp_st(s, a->rt, clean_addr, mop);
291
+ op_addr_ldst_imm_post(s, a, dirty_addr, a->imm);
292
+ return true;
293
+}
294
+
295
+static bool trans_LDR_v_i(DisasContext *s, arg_ldst_imm *a)
296
+{
297
+ TCGv_i64 clean_addr, dirty_addr;
298
+ MemOp mop;
299
+
300
+ if (!fp_access_check(s)) {
53
+ return true;
301
+ return true;
54
+ }
302
+ }
55
+
303
+ mop = finalize_memop_asimd(s, a->sz);
56
+ opr_sz = (1 + a->q) * 8;
304
+ op_addr_ldst_imm_pre(s, a, &clean_addr, &dirty_addr, a->imm, false, mop);
57
+ fn_gvec = a->u ? gen_helper_gvec_udot_b : gen_helper_gvec_sdot_b;
305
+ do_fp_ld(s, a->rt, clean_addr, mop);
58
+ tcg_gen_gvec_3_ool(vfp_reg_offset(1, a->vd),
306
+ op_addr_ldst_imm_post(s, a, dirty_addr, a->imm);
59
+ vfp_reg_offset(1, a->vn),
60
+ vfp_reg_offset(1, a->vm),
61
+ opr_sz, opr_sz, 0, fn_gvec);
62
+ return true;
307
+ return true;
63
+}
308
}
64
diff --git a/target/arm/translate.c b/target/arm/translate.c
309
65
index XXXXXXX..XXXXXXX 100644
310
/*
66
--- a/target/arm/translate.c
311
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg(DisasContext *s, uint32_t insn)
67
+++ b/target/arm/translate.c
312
switch (extract32(insn, 24, 2)) {
68
@@ -XXX,XX +XXX,XX @@ static int disas_neon_insn_3same_ext(DisasContext *s, uint32_t insn)
313
case 0:
69
bool is_long = false, q = extract32(insn, 6, 1);
314
if (extract32(insn, 21, 1) == 0) {
70
bool ptr_is_env = false;
315
- /* Load/store register (unscaled immediate)
71
316
- * Load/store immediate pre/post-indexed
72
- if ((insn & 0xfeb00f00) == 0xfc200d00) {
317
- * Load/store register unprivileged
73
- /* V[US]DOT -- 1111 1100 0.10 .... .... 1101 .Q.U .... */
318
- */
74
- bool u = extract32(insn, 4, 1);
319
- disas_ldst_reg_imm9(s, insn, opc, size, rt, is_vector);
75
- if (!dc_isar_feature(aa32_dp, s)) {
320
- return;
76
- return 1;
321
+ break;
77
- }
322
}
78
- fn_gvec = u ? gen_helper_gvec_udot_b : gen_helper_gvec_sdot_b;
323
switch (extract32(insn, 10, 2)) {
79
- } else if ((insn & 0xff300f10) == 0xfc200810) {
324
case 0:
80
+ if ((insn & 0xff300f10) == 0xfc200810) {
81
/* VFM[AS]L -- 1111 1100 S.10 .... .... 1000 .Q.1 .... */
82
int is_s = extract32(insn, 23, 1);
83
if (!dc_isar_feature(aa32_fhm, s)) {
84
--
325
--
85
2.20.1
326
2.34.1
86
87
diff view generated by jsdifflib
1
Convert the Neon "load/store single structure to one lane" insns to
1
Convert the LDR and STR instructions which use a 12-bit immediate
2
decodetree.
2
offset to decodetree. We can reuse the existing LDR and STR
3
3
trans functions for these.
4
As this is the last set of insns in the neon load/store group,
5
we can remove the whole disas_neon_ls_insn() function.
6
4
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200430181003.21682-14-peter.maydell@linaro.org
7
Message-id: 20230602155223.2040685-14-peter.maydell@linaro.org
10
---
8
---
11
target/arm/neon-ls.decode | 11 +++
9
target/arm/tcg/a64.decode | 25 ++++++++
12
target/arm/translate-neon.inc.c | 89 +++++++++++++++++++
10
target/arm/tcg/translate-a64.c | 104 +++++----------------------------
13
target/arm/translate.c | 147 --------------------------------
11
2 files changed, 41 insertions(+), 88 deletions(-)
14
3 files changed, 100 insertions(+), 147 deletions(-)
15
12
16
diff --git a/target/arm/neon-ls.decode b/target/arm/neon-ls.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
17
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
18
--- a/target/arm/neon-ls.decode
15
--- a/target/arm/tcg/a64.decode
19
+++ b/target/arm/neon-ls.decode
16
+++ b/target/arm/tcg/a64.decode
20
@@ -XXX,XX +XXX,XX @@ VLDST_multiple 1111 0100 0 . l:1 0 rn:4 .... itype:4 size:2 align:2 rm:4 \
17
@@ -XXX,XX +XXX,XX @@ STR_v_i sz:2 111 1 00 00 0 ......... 11 ..... ..... @ldst_imm_pre sign=0
21
18
STR_v_i 00 111 1 00 10 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0 sz=4
22
VLD_all_lanes 1111 0100 1 . 1 0 rn:4 .... 11 n:2 size:2 t:1 a:1 rm:4 \
19
LDR_v_i sz:2 111 1 00 01 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0
23
vd=%vd_dp
20
LDR_v_i 00 111 1 00 11 0 ......... 11 ..... ..... @ldst_imm_pre sign=0 ext=0 sz=4
24
+
21
+
25
+# Neon load/store single structure to one lane
22
+# Load/store with an unsigned 12 bit immediate, which is scaled by the
26
+%imm1_5_p1 5:1 !function=plus1
23
+# element size. The function gets the sz:imm and returns the scaled immediate.
27
+%imm1_6_p1 6:1 !function=plus1
24
+%uimm_scaled 10:12 sz:3 !function=uimm_scaled
28
+
25
+
29
+VLDST_single 1111 0100 1 . l:1 0 rn:4 .... 00 n:2 reg_idx:3 align:1 rm:4 \
26
+@ldst_uimm .. ... . .. .. ............ rn:5 rt:5 &ldst_imm unpriv=0 p=0 w=0 imm=%uimm_scaled
30
+ vd=%vd_dp size=0 stride=1
27
+
31
+VLDST_single 1111 0100 1 . l:1 0 rn:4 .... 01 n:2 reg_idx:2 align:2 rm:4 \
28
+STR_i sz:2 111 0 01 00 ............ ..... ..... @ldst_uimm sign=0 ext=0
32
+ vd=%vd_dp size=1 stride=%imm1_5_p1
29
+LDR_i 00 111 0 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=1 sz=0
33
+VLDST_single 1111 0100 1 . l:1 0 rn:4 .... 10 n:2 reg_idx:1 align:3 rm:4 \
30
+LDR_i 01 111 0 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=1 sz=1
34
+ vd=%vd_dp size=2 stride=%imm1_6_p1
31
+LDR_i 10 111 0 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=1 sz=2
35
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
32
+LDR_i 11 111 0 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=0 sz=3
33
+LDR_i 00 111 0 01 10 ............ ..... ..... @ldst_uimm sign=1 ext=0 sz=0
34
+LDR_i 01 111 0 01 10 ............ ..... ..... @ldst_uimm sign=1 ext=0 sz=1
35
+LDR_i 10 111 0 01 10 ............ ..... ..... @ldst_uimm sign=1 ext=0 sz=2
36
+LDR_i 00 111 0 01 11 ............ ..... ..... @ldst_uimm sign=1 ext=1 sz=0
37
+LDR_i 01 111 0 01 11 ............ ..... ..... @ldst_uimm sign=1 ext=1 sz=1
38
+
39
+# PRFM
40
+NOP 11 111 0 01 10 ------------ ----- -----
41
+
42
+STR_v_i sz:2 111 1 01 00 ............ ..... ..... @ldst_uimm sign=0 ext=0
43
+STR_v_i 00 111 1 01 10 ............ ..... ..... @ldst_uimm sign=0 ext=0 sz=4
44
+LDR_v_i sz:2 111 1 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=0
45
+LDR_v_i 00 111 1 01 11 ............ ..... ..... @ldst_uimm sign=0 ext=0 sz=4
46
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
36
index XXXXXXX..XXXXXXX 100644
47
index XXXXXXX..XXXXXXX 100644
37
--- a/target/arm/translate-neon.inc.c
48
--- a/target/arm/tcg/translate-a64.c
38
+++ b/target/arm/translate-neon.inc.c
49
+++ b/target/arm/tcg/translate-a64.c
39
@@ -XXX,XX +XXX,XX @@
50
@@ -XXX,XX +XXX,XX @@ enum a64_shift_type {
40
* It might be possible to convert it to a standalone .c file eventually.
51
A64_SHIFT_TYPE_ROR = 3
41
*/
52
};
42
53
43
+static inline int plus1(DisasContext *s, int x)
54
+/*
55
+ * Helpers for extracting complex instruction fields
56
+ */
57
+
58
+/*
59
+ * For load/store with an unsigned 12 bit immediate scaled by the element
60
+ * size. The input has the immediate field in bits [14:3] and the element
61
+ * size in [2:0].
62
+ */
63
+static int uimm_scaled(DisasContext *s, int x)
44
+{
64
+{
45
+ return x + 1;
65
+ unsigned imm = x >> 3;
66
+ unsigned scale = extract32(x, 0, 3);
67
+ return imm << scale;
46
+}
68
+}
47
+
69
+
48
/* Include the generated Neon decoder */
70
/*
49
#include "decode-neon-dp.inc.c"
71
* Include the generated decoders.
50
#include "decode-neon-ls.inc.c"
72
*/
51
@@ -XXX,XX +XXX,XX @@ static bool trans_VLD_all_lanes(DisasContext *s, arg_VLD_all_lanes *a)
73
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
52
74
}
53
return true;
54
}
75
}
55
+
76
56
+static bool trans_VLDST_single(DisasContext *s, arg_VLDST_single *a)
77
-/*
57
+{
78
- * Load/store (unsigned immediate)
58
+ /* Neon load/store single structure to one lane */
79
- *
59
+ int reg;
80
- * 31 30 29 27 26 25 24 23 22 21 10 9 5
60
+ int nregs = a->n + 1;
81
- * +----+-------+---+-----+-----+------------+-------+------+
61
+ int vd = a->vd;
82
- * |size| 1 1 1 | V | 0 1 | opc | imm12 | Rn | Rt |
62
+ TCGv_i32 addr, tmp;
83
- * +----+-------+---+-----+-----+------------+-------+------+
63
+
84
- *
64
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
85
- * For non-vector:
65
+ return false;
86
- * size: 00-> byte, 01 -> 16 bit, 10 -> 32bit, 11 -> 64bit
66
+ }
87
- * opc: 00 -> store, 01 -> loadu, 10 -> loads 64, 11 -> loads 32
67
+
88
- * For vector:
68
+ /* UNDEF accesses to D16-D31 if they don't exist */
89
- * size is opc<1>:size<1:0> so 100 -> 128 bit; 110 and 111 unallocated
69
+ if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
90
- * opc<0>: 0 -> store, 1 -> load
70
+ return false;
91
- * Rn: base address register (inc SP)
71
+ }
92
- * Rt: target register
72
+
93
- */
73
+ /* Catch the UNDEF cases. This is unavoidably a bit messy. */
94
-static void disas_ldst_reg_unsigned_imm(DisasContext *s, uint32_t insn,
74
+ switch (nregs) {
95
- int opc,
75
+ case 1:
96
- int size,
76
+ if (((a->align & (1 << a->size)) != 0) ||
97
- int rt,
77
+ (a->size == 2 && ((a->align & 3) == 1 || (a->align & 3) == 2))) {
98
- bool is_vector)
78
+ return false;
99
-{
79
+ }
100
- int rn = extract32(insn, 5, 5);
80
+ break;
101
- unsigned int imm12 = extract32(insn, 10, 12);
81
+ case 3:
102
- unsigned int offset;
82
+ if ((a->align & 1) != 0) {
103
- TCGv_i64 clean_addr, dirty_addr;
83
+ return false;
104
- bool is_store;
84
+ }
105
- bool is_signed = false;
85
+ /* fall through */
106
- bool is_extended = false;
86
+ case 2:
107
- MemOp memop;
87
+ if (a->size == 2 && (a->align & 2) != 0) {
88
+ return false;
89
+ }
90
+ break;
91
+ case 4:
92
+ if ((a->size == 2) && ((a->align & 3) == 3)) {
93
+ return false;
94
+ }
95
+ break;
96
+ default:
97
+ abort();
98
+ }
99
+ if ((vd + a->stride * (nregs - 1)) > 31) {
100
+ /*
101
+ * Attempts to write off the end of the register file are
102
+ * UNPREDICTABLE; we choose to UNDEF because otherwise we would
103
+ * access off the end of the array that holds the register data.
104
+ */
105
+ return false;
106
+ }
107
+
108
+ if (!vfp_access_check(s)) {
109
+ return true;
110
+ }
111
+
112
+ tmp = tcg_temp_new_i32();
113
+ addr = tcg_temp_new_i32();
114
+ load_reg_var(s, addr, a->rn);
115
+ /*
116
+ * TODO: if we implemented alignment exceptions, we should check
117
+ * addr against the alignment encoded in a->align here.
118
+ */
119
+ for (reg = 0; reg < nregs; reg++) {
120
+ if (a->l) {
121
+ gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s),
122
+ s->be_data | a->size);
123
+ neon_store_element(vd, a->reg_idx, a->size, tmp);
124
+ } else { /* Store */
125
+ neon_load_element(tmp, vd, a->reg_idx, a->size);
126
+ gen_aa32_st_i32(s, tmp, addr, get_mem_index(s),
127
+ s->be_data | a->size);
128
+ }
129
+ vd += a->stride;
130
+ tcg_gen_addi_i32(addr, addr, 1 << a->size);
131
+ }
132
+ tcg_temp_free_i32(addr);
133
+ tcg_temp_free_i32(tmp);
134
+
135
+ gen_neon_ldst_base_update(s, a->rm, a->rn, (1 << a->size) * nregs);
136
+
137
+ return true;
138
+}
139
diff --git a/target/arm/translate.c b/target/arm/translate.c
140
index XXXXXXX..XXXXXXX 100644
141
--- a/target/arm/translate.c
142
+++ b/target/arm/translate.c
143
@@ -XXX,XX +XXX,XX @@ static void gen_neon_trn_u16(TCGv_i32 t0, TCGv_i32 t1)
144
tcg_temp_free_i32(rd);
145
}
146
147
-
108
-
148
-/* Translate a NEON load/store element instruction. Return nonzero if the
109
- if (is_vector) {
149
- instruction is invalid. */
110
- size |= (opc & 2) << 1;
150
-static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
111
- if (size > 4) {
151
-{
112
- unallocated_encoding(s);
152
- int rd, rn, rm;
113
- return;
153
- int nregs;
114
- }
154
- int stride;
115
- is_store = !extract32(opc, 0, 1);
155
- int size;
116
- if (!fp_access_check(s)) {
156
- int reg;
117
- return;
157
- int load;
118
- }
158
- TCGv_i32 addr;
119
- memop = finalize_memop_asimd(s, size);
159
- TCGv_i32 tmp;
120
- } else {
160
-
121
- if (size == 3 && opc == 2) {
161
- if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
122
- /* PRFM - prefetch */
162
- return 1;
123
- return;
124
- }
125
- if (opc == 3 && size > 1) {
126
- unallocated_encoding(s);
127
- return;
128
- }
129
- is_store = (opc == 0);
130
- is_signed = !is_store && extract32(opc, 1, 1);
131
- is_extended = (size < 3) && extract32(opc, 0, 1);
132
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
163
- }
133
- }
164
-
134
-
165
- /* FIXME: this access check should not take precedence over UNDEF
135
- if (rn == 31) {
166
- * for invalid encodings; we will generate incorrect syndrome information
136
- gen_check_sp_alignment(s);
167
- * for attempts to execute invalid vfp/neon encodings with FP disabled.
168
- */
169
- if (s->fp_excp_el) {
170
- gen_exception_insn(s, s->pc_curr, EXCP_UDEF,
171
- syn_simd_access_trap(1, 0xe, false), s->fp_excp_el);
172
- return 0;
173
- }
137
- }
138
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
139
- offset = imm12 << size;
140
- tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
174
-
141
-
175
- if (!s->vfp_enabled)
142
- clean_addr = gen_mte_check1(s, dirty_addr, is_store, rn != 31, memop);
176
- return 1;
143
-
177
- VFP_DREG_D(rd, insn);
144
- if (is_vector) {
178
- rn = (insn >> 16) & 0xf;
145
- if (is_store) {
179
- rm = insn & 0xf;
146
- do_fp_st(s, rt, clean_addr, memop);
180
- load = (insn & (1 << 21)) != 0;
147
- } else {
181
- if ((insn & (1 << 23)) == 0) {
148
- do_fp_ld(s, rt, clean_addr, memop);
182
- /* Load store all elements -- handled already by decodetree */
149
- }
183
- return 1;
184
- } else {
150
- } else {
185
- size = (insn >> 10) & 3;
151
- TCGv_i64 tcg_rt = cpu_reg(s, rt);
186
- if (size == 3) {
152
- bool iss_sf = disas_ldst_compute_iss_sf(size, is_signed, opc);
187
- /* Load single element to all lanes -- handled by decodetree */
153
- if (is_store) {
188
- return 1;
154
- do_gpr_st(s, tcg_rt, clean_addr, memop, true, rt, iss_sf, false);
189
- } else {
155
- } else {
190
- /* Single element. */
156
- do_gpr_ld(s, tcg_rt, clean_addr, memop,
191
- int idx = (insn >> 4) & 0xf;
157
- is_extended, true, rt, iss_sf, false);
192
- int reg_idx;
193
- switch (size) {
194
- case 0:
195
- reg_idx = (insn >> 5) & 7;
196
- stride = 1;
197
- break;
198
- case 1:
199
- reg_idx = (insn >> 6) & 3;
200
- stride = (insn & (1 << 5)) ? 2 : 1;
201
- break;
202
- case 2:
203
- reg_idx = (insn >> 7) & 1;
204
- stride = (insn & (1 << 6)) ? 2 : 1;
205
- break;
206
- default:
207
- abort();
208
- }
209
- nregs = ((insn >> 8) & 3) + 1;
210
- /* Catch the UNDEF cases. This is unavoidably a bit messy. */
211
- switch (nregs) {
212
- case 1:
213
- if (((idx & (1 << size)) != 0) ||
214
- (size == 2 && ((idx & 3) == 1 || (idx & 3) == 2))) {
215
- return 1;
216
- }
217
- break;
218
- case 3:
219
- if ((idx & 1) != 0) {
220
- return 1;
221
- }
222
- /* fall through */
223
- case 2:
224
- if (size == 2 && (idx & 2) != 0) {
225
- return 1;
226
- }
227
- break;
228
- case 4:
229
- if ((size == 2) && ((idx & 3) == 3)) {
230
- return 1;
231
- }
232
- break;
233
- default:
234
- abort();
235
- }
236
- if ((rd + stride * (nregs - 1)) > 31) {
237
- /* Attempts to write off the end of the register file
238
- * are UNPREDICTABLE; we choose to UNDEF because otherwise
239
- * the neon_load_reg() would write off the end of the array.
240
- */
241
- return 1;
242
- }
243
- tmp = tcg_temp_new_i32();
244
- addr = tcg_temp_new_i32();
245
- load_reg_var(s, addr, rn);
246
- for (reg = 0; reg < nregs; reg++) {
247
- if (load) {
248
- gen_aa32_ld_i32(s, tmp, addr, get_mem_index(s),
249
- s->be_data | size);
250
- neon_store_element(rd, reg_idx, size, tmp);
251
- } else { /* Store */
252
- neon_load_element(tmp, rd, reg_idx, size);
253
- gen_aa32_st_i32(s, tmp, addr, get_mem_index(s),
254
- s->be_data | size);
255
- }
256
- rd += stride;
257
- tcg_gen_addi_i32(addr, addr, 1 << size);
258
- }
259
- tcg_temp_free_i32(addr);
260
- tcg_temp_free_i32(tmp);
261
- stride = nregs * (1 << size);
262
- }
158
- }
263
- }
159
- }
264
- if (rm != 15) {
265
- TCGv_i32 base;
266
-
267
- base = load_reg(s, rn);
268
- if (rm == 13) {
269
- tcg_gen_addi_i32(base, base, stride);
270
- } else {
271
- TCGv_i32 index;
272
- index = load_reg(s, rm);
273
- tcg_gen_add_i32(base, base, index);
274
- tcg_temp_free_i32(index);
275
- }
276
- store_reg(s, rn, base);
277
- }
278
- return 0;
279
-}
160
-}
280
-
161
-
281
static inline void gen_neon_narrow(int size, TCGv_i32 dest, TCGv_i64 src)
162
/* Atomic memory operations
282
{
163
*
283
switch (size) {
164
* 31 30 27 26 24 22 21 16 15 12 10 5 0
284
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
165
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg(DisasContext *s, uint32_t insn)
285
}
286
return;
166
return;
287
}
167
}
288
- if ((insn & 0x0f100000) == 0x04000000) {
289
- /* NEON load/store. */
290
- if (disas_neon_ls_insn(s, insn)) {
291
- goto illegal_op;
292
- }
293
- return;
294
- }
295
if ((insn & 0x0e000f00) == 0x0c000100) {
296
if (arm_dc_feature(s, ARM_FEATURE_IWMMXT)) {
297
/* iWMMXt register transfer. */
298
@@ -XXX,XX +XXX,XX @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn)
299
}
300
break;
168
break;
301
case 12:
169
- case 1:
302
- if ((insn & 0x01100000) == 0x01000000) {
170
- disas_ldst_reg_unsigned_imm(s, insn, opc, size, rt, is_vector);
303
- if (disas_neon_ls_insn(s, insn)) {
171
- return;
304
- goto illegal_op;
172
}
305
- }
173
unallocated_encoding(s);
306
- break;
174
}
307
- }
308
goto illegal_op;
309
default:
310
illegal_op:
311
--
175
--
312
2.20.1
176
2.34.1
313
314
diff view generated by jsdifflib
1
Convert the VCADD (vector) insns to decodetree.
1
Convert the LDR and STR instructions which take a register
2
plus register offset to decodetree.
2
3
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200430181003.21682-6-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-15-peter.maydell@linaro.org
6
---
7
---
7
target/arm/neon-shared.decode | 3 +++
8
target/arm/tcg/a64.decode | 22 +++++
8
target/arm/translate-neon.inc.c | 37 +++++++++++++++++++++++++++++++++
9
target/arm/tcg/translate-a64.c | 173 +++++++++++++++------------------
9
target/arm/translate.c | 11 +---------
10
2 files changed, 103 insertions(+), 92 deletions(-)
10
3 files changed, 41 insertions(+), 10 deletions(-)
11
11
12
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/neon-shared.decode
14
--- a/target/arm/tcg/a64.decode
15
+++ b/target/arm/neon-shared.decode
15
+++ b/target/arm/tcg/a64.decode
16
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ STR_v_i sz:2 111 1 01 00 ............ ..... ..... @ldst_uimm sign=0 ext=
17
17
STR_v_i 00 111 1 01 10 ............ ..... ..... @ldst_uimm sign=0 ext=0 sz=4
18
VCMLA 1111 110 rot:2 . 1 size:1 .... .... 1000 . q:1 . 0 .... \
18
LDR_v_i sz:2 111 1 01 01 ............ ..... ..... @ldst_uimm sign=0 ext=0
19
vm=%vm_dp vn=%vn_dp vd=%vd_dp
19
LDR_v_i 00 111 1 01 11 ............ ..... ..... @ldst_uimm sign=0 ext=0 sz=4
20
+
20
+
21
+VCADD 1111 110 rot:1 1 . 0 size:1 .... .... 1000 . q:1 . 0 .... \
21
+# Load/store with register offset
22
+ vm=%vm_dp vn=%vn_dp vd=%vd_dp
22
+&ldst rm rn rt sign ext sz opt s
23
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
23
+@ldst .. ... . .. .. . rm:5 opt:3 s:1 .. rn:5 rt:5 &ldst
24
+STR sz:2 111 0 00 00 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0
25
+LDR 00 111 0 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=1 sz=0
26
+LDR 01 111 0 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=1 sz=1
27
+LDR 10 111 0 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=1 sz=2
28
+LDR 11 111 0 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0 sz=3
29
+LDR 00 111 0 00 10 1 ..... ... . 10 ..... ..... @ldst sign=1 ext=0 sz=0
30
+LDR 01 111 0 00 10 1 ..... ... . 10 ..... ..... @ldst sign=1 ext=0 sz=1
31
+LDR 10 111 0 00 10 1 ..... ... . 10 ..... ..... @ldst sign=1 ext=0 sz=2
32
+LDR 00 111 0 00 11 1 ..... ... . 10 ..... ..... @ldst sign=1 ext=1 sz=0
33
+LDR 01 111 0 00 11 1 ..... ... . 10 ..... ..... @ldst sign=1 ext=1 sz=1
34
+
35
+# PRFM
36
+NOP 11 111 0 00 10 1 ----- -1- - 10 ----- -----
37
+
38
+STR_v sz:2 111 1 00 00 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0
39
+STR_v 00 111 1 00 10 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0 sz=4
40
+LDR_v sz:2 111 1 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0
41
+LDR_v 00 111 1 00 11 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0 sz=4
42
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
24
index XXXXXXX..XXXXXXX 100644
43
index XXXXXXX..XXXXXXX 100644
25
--- a/target/arm/translate-neon.inc.c
44
--- a/target/arm/tcg/translate-a64.c
26
+++ b/target/arm/translate-neon.inc.c
45
+++ b/target/arm/tcg/translate-a64.c
27
@@ -XXX,XX +XXX,XX @@ static bool trans_VCMLA(DisasContext *s, arg_VCMLA *a)
46
@@ -XXX,XX +XXX,XX @@ static bool trans_LDR_v_i(DisasContext *s, arg_ldst_imm *a)
28
tcg_temp_free_ptr(fpst);
29
return true;
47
return true;
30
}
48
}
31
+
49
32
+static bool trans_VCADD(DisasContext *s, arg_VCADD *a)
50
-/*
33
+{
51
- * Load/store (register offset)
34
+ int opr_sz;
52
- *
35
+ TCGv_ptr fpst;
53
- * 31 30 29 27 26 25 24 23 22 21 20 16 15 13 12 11 10 9 5 4 0
36
+ gen_helper_gvec_3_ptr *fn_gvec_ptr;
54
- * +----+-------+---+-----+-----+---+------+-----+--+-----+----+----+
37
+
55
- * |size| 1 1 1 | V | 0 0 | opc | 1 | Rm | opt | S| 1 0 | Rn | Rt |
38
+ if (!dc_isar_feature(aa32_vcma, s)
56
- * +----+-------+---+-----+-----+---+------+-----+--+-----+----+----+
39
+ || (!a->size && !dc_isar_feature(aa32_fp16_arith, s))) {
57
- *
40
+ return false;
58
- * For non-vector:
41
+ }
59
- * size: 00-> byte, 01 -> 16 bit, 10 -> 32bit, 11 -> 64bit
42
+
60
- * opc: 00 -> store, 01 -> loadu, 10 -> loads 64, 11 -> loads 32
43
+ /* UNDEF accesses to D16-D31 if they don't exist. */
61
- * For vector:
44
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
62
- * size is opc<1>:size<1:0> so 100 -> 128 bit; 110 and 111 unallocated
45
+ ((a->vd | a->vn | a->vm) & 0x10)) {
63
- * opc<0>: 0 -> store, 1 -> load
46
+ return false;
64
- * V: 1 -> vector/simd
47
+ }
65
- * opt: extend encoding (see DecodeRegExtend)
48
+
66
- * S: if S=1 then scale (essentially index by sizeof(size))
49
+ if ((a->vn | a->vm | a->vd) & a->q) {
67
- * Rt: register to transfer into/out of
50
+ return false;
68
- * Rn: address register or SP for base
51
+ }
69
- * Rm: offset register or ZR for offset
52
+
70
- */
53
+ if (!vfp_access_check(s)) {
71
-static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
72
- int opc,
73
- int size,
74
- int rt,
75
- bool is_vector)
76
+static void op_addr_ldst_pre(DisasContext *s, arg_ldst *a,
77
+ TCGv_i64 *clean_addr, TCGv_i64 *dirty_addr,
78
+ bool is_store, MemOp memop)
79
{
80
- int rn = extract32(insn, 5, 5);
81
- int shift = extract32(insn, 12, 1);
82
- int rm = extract32(insn, 16, 5);
83
- int opt = extract32(insn, 13, 3);
84
- bool is_signed = false;
85
- bool is_store = false;
86
- bool is_extended = false;
87
- TCGv_i64 tcg_rm, clean_addr, dirty_addr;
88
- MemOp memop;
89
+ TCGv_i64 tcg_rm;
90
91
- if (extract32(opt, 1, 1) == 0) {
92
- unallocated_encoding(s);
93
- return;
94
- }
95
-
96
- if (is_vector) {
97
- size |= (opc & 2) << 1;
98
- if (size > 4) {
99
- unallocated_encoding(s);
100
- return;
101
- }
102
- is_store = !extract32(opc, 0, 1);
103
- if (!fp_access_check(s)) {
104
- return;
105
- }
106
- memop = finalize_memop_asimd(s, size);
107
- } else {
108
- if (size == 3 && opc == 2) {
109
- /* PRFM - prefetch */
110
- return;
111
- }
112
- if (opc == 3 && size > 1) {
113
- unallocated_encoding(s);
114
- return;
115
- }
116
- is_store = (opc == 0);
117
- is_signed = !is_store && extract32(opc, 1, 1);
118
- is_extended = (size < 3) && extract32(opc, 0, 1);
119
- memop = finalize_memop(s, size + is_signed * MO_SIGN);
120
- }
121
-
122
- if (rn == 31) {
123
+ if (a->rn == 31) {
124
gen_check_sp_alignment(s);
125
}
126
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
127
+ *dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
128
129
- tcg_rm = read_cpu_reg(s, rm, 1);
130
- ext_and_shift_reg(tcg_rm, tcg_rm, opt, shift ? size : 0);
131
+ tcg_rm = read_cpu_reg(s, a->rm, 1);
132
+ ext_and_shift_reg(tcg_rm, tcg_rm, a->opt, a->s ? a->sz : 0);
133
134
- tcg_gen_add_i64(dirty_addr, dirty_addr, tcg_rm);
135
+ tcg_gen_add_i64(*dirty_addr, *dirty_addr, tcg_rm);
136
+ *clean_addr = gen_mte_check1(s, *dirty_addr, is_store, true, memop);
137
+}
138
139
- clean_addr = gen_mte_check1(s, dirty_addr, is_store, true, memop);
140
+static bool trans_LDR(DisasContext *s, arg_ldst *a)
141
+{
142
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt;
143
+ bool iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
144
+ MemOp memop;
145
146
- if (is_vector) {
147
- if (is_store) {
148
- do_fp_st(s, rt, clean_addr, memop);
149
- } else {
150
- do_fp_ld(s, rt, clean_addr, memop);
151
- }
152
- } else {
153
- TCGv_i64 tcg_rt = cpu_reg(s, rt);
154
- bool iss_sf = disas_ldst_compute_iss_sf(size, is_signed, opc);
155
-
156
- if (is_store) {
157
- do_gpr_st(s, tcg_rt, clean_addr, memop,
158
- true, rt, iss_sf, false);
159
- } else {
160
- do_gpr_ld(s, tcg_rt, clean_addr, memop,
161
- is_extended, true, rt, iss_sf, false);
162
- }
163
+ if (extract32(a->opt, 1, 1) == 0) {
164
+ return false;
165
}
166
+
167
+ memop = finalize_memop(s, a->sz + a->sign * MO_SIGN);
168
+ op_addr_ldst_pre(s, a, &clean_addr, &dirty_addr, false, memop);
169
+ tcg_rt = cpu_reg(s, a->rt);
170
+ do_gpr_ld(s, tcg_rt, clean_addr, memop,
171
+ a->ext, true, a->rt, iss_sf, false);
172
+ return true;
173
+}
174
+
175
+static bool trans_STR(DisasContext *s, arg_ldst *a)
176
+{
177
+ TCGv_i64 clean_addr, dirty_addr, tcg_rt;
178
+ bool iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
179
+ MemOp memop;
180
+
181
+ if (extract32(a->opt, 1, 1) == 0) {
182
+ return false;
183
+ }
184
+
185
+ memop = finalize_memop(s, a->sz);
186
+ op_addr_ldst_pre(s, a, &clean_addr, &dirty_addr, true, memop);
187
+ tcg_rt = cpu_reg(s, a->rt);
188
+ do_gpr_st(s, tcg_rt, clean_addr, memop, true, a->rt, iss_sf, false);
189
+ return true;
190
+}
191
+
192
+static bool trans_LDR_v(DisasContext *s, arg_ldst *a)
193
+{
194
+ TCGv_i64 clean_addr, dirty_addr;
195
+ MemOp memop;
196
+
197
+ if (extract32(a->opt, 1, 1) == 0) {
198
+ return false;
199
+ }
200
+
201
+ if (!fp_access_check(s)) {
54
+ return true;
202
+ return true;
55
+ }
203
+ }
56
+
204
+
57
+ opr_sz = (1 + a->q) * 8;
205
+ memop = finalize_memop_asimd(s, a->sz);
58
+ fpst = get_fpstatus_ptr(1);
206
+ op_addr_ldst_pre(s, a, &clean_addr, &dirty_addr, false, memop);
59
+ fn_gvec_ptr = a->size ? gen_helper_gvec_fcadds : gen_helper_gvec_fcaddh;
207
+ do_fp_ld(s, a->rt, clean_addr, memop);
60
+ tcg_gen_gvec_3_ptr(vfp_reg_offset(1, a->vd),
208
+ return true;
61
+ vfp_reg_offset(1, a->vn),
209
+}
62
+ vfp_reg_offset(1, a->vm),
210
+
63
+ fpst, opr_sz, opr_sz, a->rot,
211
+static bool trans_STR_v(DisasContext *s, arg_ldst *a)
64
+ fn_gvec_ptr);
212
+{
65
+ tcg_temp_free_ptr(fpst);
213
+ TCGv_i64 clean_addr, dirty_addr;
66
+ return true;
214
+ MemOp memop;
67
+}
215
+
68
diff --git a/target/arm/translate.c b/target/arm/translate.c
216
+ if (extract32(a->opt, 1, 1) == 0) {
69
index XXXXXXX..XXXXXXX 100644
217
+ return false;
70
--- a/target/arm/translate.c
218
+ }
71
+++ b/target/arm/translate.c
219
+
72
@@ -XXX,XX +XXX,XX @@ static int disas_neon_insn_3same_ext(DisasContext *s, uint32_t insn)
220
+ if (!fp_access_check(s)) {
73
bool is_long = false, q = extract32(insn, 6, 1);
221
+ return true;
74
bool ptr_is_env = false;
222
+ }
75
223
+
76
- if ((insn & 0xfea00f10) == 0xfc800800) {
224
+ memop = finalize_memop_asimd(s, a->sz);
77
- /* VCADD -- 1111 110R 1.0S .... .... 1000 ...0 .... */
225
+ op_addr_ldst_pre(s, a, &clean_addr, &dirty_addr, true, memop);
78
- int size = extract32(insn, 20, 1);
226
+ do_fp_st(s, a->rt, clean_addr, memop);
79
- data = extract32(insn, 24, 1); /* rot */
227
+ return true;
80
- if (!dc_isar_feature(aa32_vcma, s)
228
}
81
- || (!size && !dc_isar_feature(aa32_fp16_arith, s))) {
229
82
- return 1;
230
/* Atomic memory operations
83
- }
231
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_ldapr_stlr(DisasContext *s, uint32_t insn)
84
- fn_gvec_ptr = size ? gen_helper_gvec_fcadds : gen_helper_gvec_fcaddh;
232
static void disas_ldst_reg(DisasContext *s, uint32_t insn)
85
- } else if ((insn & 0xfeb00f00) == 0xfc200d00) {
233
{
86
+ if ((insn & 0xfeb00f00) == 0xfc200d00) {
234
int rt = extract32(insn, 0, 5);
87
/* V[US]DOT -- 1111 1100 0.10 .... .... 1101 .Q.U .... */
235
- int opc = extract32(insn, 22, 2);
88
bool u = extract32(insn, 4, 1);
236
bool is_vector = extract32(insn, 26, 1);
89
if (!dc_isar_feature(aa32_dp, s)) {
237
int size = extract32(insn, 30, 2);
238
239
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg(DisasContext *s, uint32_t insn)
240
disas_ldst_atomic(s, insn, size, rt, is_vector);
241
return;
242
case 2:
243
- disas_ldst_reg_roffset(s, insn, opc, size, rt, is_vector);
244
- return;
245
+ break;
246
default:
247
disas_ldst_pac(s, insn, size, rt, is_vector);
248
return;
90
--
249
--
91
2.20.1
250
2.34.1
92
93
diff view generated by jsdifflib
1
We define ARMMMUIdx_Stage2 as being an MMU index which uses a QEMU
1
Convert the insns in the atomic memory operations group to
2
TLB. However we never actually use the TLB -- all stage 2 lookups
2
decodetree.
3
are done by direct calls to get_phys_addr_lpae() followed by a
4
physical address load via address_space_ld*().
5
6
Remove Stage2 from the list of ARM MMU indexes which correspond to
7
real core MMU indexes, and instead put it in the set of "NOTLB" ARM
8
MMU indexes.
9
10
This allows us to drop NB_MMU_MODES to 11. It also means we can
11
safely add support for the ARMv8.3-TTS2UXN extension, which adds
12
permission bits to the stage 2 descriptors which define execute
13
permission separatel for EL0 and EL1; supporting that while keeping
14
Stage2 in a QEMU TLB would require us to use separate TLBs for
15
"Stage2 for an EL0 access" and "Stage2 for an EL1 access", which is a
16
lot of extra complication given we aren't even using the QEMU TLB.
17
18
In the process of updating the comment on our MMU index use,
19
fix a couple of other minor errors:
20
* NS EL2 EL2&0 was missing from the list in the comment
21
* some text hadn't been updated from when we bumped NB_MMU_MODES
22
above 8
23
3
24
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
25
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
26
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
27
Message-id: 20200330210400.11724-2-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-16-peter.maydell@linaro.org
28
---
7
---
29
target/arm/cpu-param.h | 2 +-
8
target/arm/tcg/a64.decode | 15 ++++
30
target/arm/cpu.h | 21 +++++---
9
target/arm/tcg/translate-a64.c | 153 ++++++++++++---------------------
31
target/arm/helper.c | 112 ++++-------------------------------------
10
2 files changed, 70 insertions(+), 98 deletions(-)
32
3 files changed, 27 insertions(+), 108 deletions(-)
33
11
34
diff --git a/target/arm/cpu-param.h b/target/arm/cpu-param.h
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
35
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
36
--- a/target/arm/cpu-param.h
14
--- a/target/arm/tcg/a64.decode
37
+++ b/target/arm/cpu-param.h
15
+++ b/target/arm/tcg/a64.decode
38
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ STR_v sz:2 111 1 00 00 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0
39
# define TARGET_PAGE_BITS_MIN 10
17
STR_v 00 111 1 00 10 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0 sz=4
40
#endif
18
LDR_v sz:2 111 1 00 01 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0
41
19
LDR_v 00 111 1 00 11 1 ..... ... . 10 ..... ..... @ldst sign=0 ext=0 sz=4
42
-#define NB_MMU_MODES 12
20
+
43
+#define NB_MMU_MODES 11
21
+# Atomic memory operations
44
22
+&atomic rs rn rt a r sz
45
#endif
23
+@atomic sz:2 ... . .. a:1 r:1 . rs:5 . ... .. rn:5 rt:5 &atomic
46
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
24
+LDADD .. 111 0 00 . . 1 ..... 0000 00 ..... ..... @atomic
25
+LDCLR .. 111 0 00 . . 1 ..... 0001 00 ..... ..... @atomic
26
+LDEOR .. 111 0 00 . . 1 ..... 0010 00 ..... ..... @atomic
27
+LDSET .. 111 0 00 . . 1 ..... 0011 00 ..... ..... @atomic
28
+LDSMAX .. 111 0 00 . . 1 ..... 0100 00 ..... ..... @atomic
29
+LDSMIN .. 111 0 00 . . 1 ..... 0101 00 ..... ..... @atomic
30
+LDUMAX .. 111 0 00 . . 1 ..... 0110 00 ..... ..... @atomic
31
+LDUMIN .. 111 0 00 . . 1 ..... 0111 00 ..... ..... @atomic
32
+SWP .. 111 0 00 . . 1 ..... 1000 00 ..... ..... @atomic
33
+
34
+LDAPR sz:2 111 0 00 1 0 1 11111 1100 00 rn:5 rt:5
35
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
47
index XXXXXXX..XXXXXXX 100644
36
index XXXXXXX..XXXXXXX 100644
48
--- a/target/arm/cpu.h
37
--- a/target/arm/tcg/translate-a64.c
49
+++ b/target/arm/cpu.h
38
+++ b/target/arm/tcg/translate-a64.c
50
@@ -XXX,XX +XXX,XX @@ bool write_cpustate_to_list(ARMCPU *cpu, bool kvm_sync);
39
@@ -XXX,XX +XXX,XX @@ static bool trans_STR_v(DisasContext *s, arg_ldst *a)
51
* handling via the TLB. The only way to do a stage 1 translation without
40
return true;
52
* the immediate stage 2 translation is via the ATS or AT system insns,
53
* which can be slow-pathed and always do a page table walk.
54
+ * The only use of stage 2 translations is either as part of an s1+2
55
+ * lookup or when loading the descriptors during a stage 1 page table walk,
56
+ * and in both those cases we don't use the TLB.
57
* 4. we can also safely fold together the "32 bit EL3" and "64 bit EL3"
58
* translation regimes, because they map reasonably well to each other
59
* and they can't both be active at the same time.
60
@@ -XXX,XX +XXX,XX @@ bool write_cpustate_to_list(ARMCPU *cpu, bool kvm_sync);
61
* NS EL1 EL1&0 stage 1+2 (aka NS PL1)
62
* NS EL1 EL1&0 stage 1+2 +PAN
63
* NS EL0 EL2&0
64
+ * NS EL2 EL2&0
65
* NS EL2 EL2&0 +PAN
66
* NS EL2 (aka NS PL2)
67
* S EL0 EL1&0 (aka S PL0)
68
* S EL1 EL1&0 (not used if EL3 is 32 bit)
69
* S EL1 EL1&0 +PAN
70
* S EL3 (aka S PL1)
71
- * NS EL1&0 stage 2
72
*
73
- * for a total of 12 different mmu_idx.
74
+ * for a total of 11 different mmu_idx.
75
*
76
* R profile CPUs have an MPU, but can use the same set of MMU indexes
77
* as A profile. They only need to distinguish NS EL0 and NS EL1 (and
78
@@ -XXX,XX +XXX,XX @@ bool write_cpustate_to_list(ARMCPU *cpu, bool kvm_sync);
79
* are not quite the same -- different CPU types (most notably M profile
80
* vs A/R profile) would like to use MMU indexes with different semantics,
81
* but since we don't ever need to use all of those in a single CPU we
82
- * can avoid setting NB_MMU_MODES to more than 8. The lower bits of
83
+ * can avoid having to set NB_MMU_MODES to "total number of A profile MMU
84
+ * modes + total number of M profile MMU modes". The lower bits of
85
* ARMMMUIdx are the core TLB mmu index, and the higher bits are always
86
* the same for any particular CPU.
87
* Variables of type ARMMUIdx are always full values, and the core
88
@@ -XXX,XX +XXX,XX @@ typedef enum ARMMMUIdx {
89
ARMMMUIdx_SE10_1_PAN = 9 | ARM_MMU_IDX_A,
90
ARMMMUIdx_SE3 = 10 | ARM_MMU_IDX_A,
91
92
- ARMMMUIdx_Stage2 = 11 | ARM_MMU_IDX_A,
93
-
94
/*
95
* These are not allocated TLBs and are used only for AT system
96
* instructions or for the first stage of an S12 page table walk.
97
@@ -XXX,XX +XXX,XX @@ typedef enum ARMMMUIdx {
98
ARMMMUIdx_Stage1_E0 = 0 | ARM_MMU_IDX_NOTLB,
99
ARMMMUIdx_Stage1_E1 = 1 | ARM_MMU_IDX_NOTLB,
100
ARMMMUIdx_Stage1_E1_PAN = 2 | ARM_MMU_IDX_NOTLB,
101
+ /*
102
+ * Not allocated a TLB: used only for second stage of an S12 page
103
+ * table walk, or for descriptor loads during first stage of an S1
104
+ * page table walk. Note that if we ever want to have a TLB for this
105
+ * then various TLB flush insns which currently are no-ops or flush
106
+ * only stage 1 MMU indexes will need to change to flush stage 2.
107
+ */
108
+ ARMMMUIdx_Stage2 = 3 | ARM_MMU_IDX_NOTLB,
109
110
/*
111
* M-profile.
112
@@ -XXX,XX +XXX,XX @@ typedef enum ARMMMUIdxBit {
113
TO_CORE_BIT(SE10_1),
114
TO_CORE_BIT(SE10_1_PAN),
115
TO_CORE_BIT(SE3),
116
- TO_CORE_BIT(Stage2),
117
118
TO_CORE_BIT(MUser),
119
TO_CORE_BIT(MPriv),
120
diff --git a/target/arm/helper.c b/target/arm/helper.c
121
index XXXXXXX..XXXXXXX 100644
122
--- a/target/arm/helper.c
123
+++ b/target/arm/helper.c
124
@@ -XXX,XX +XXX,XX @@ static void tlbiall_nsnh_write(CPUARMState *env, const ARMCPRegInfo *ri,
125
tlb_flush_by_mmuidx(cs,
126
ARMMMUIdxBit_E10_1 |
127
ARMMMUIdxBit_E10_1_PAN |
128
- ARMMMUIdxBit_E10_0 |
129
- ARMMMUIdxBit_Stage2);
130
+ ARMMMUIdxBit_E10_0);
131
}
41
}
132
42
133
static void tlbiall_nsnh_is_write(CPUARMState *env, const ARMCPRegInfo *ri,
43
-/* Atomic memory operations
134
@@ -XXX,XX +XXX,XX @@ static void tlbiall_nsnh_is_write(CPUARMState *env, const ARMCPRegInfo *ri,
44
- *
135
tlb_flush_by_mmuidx_all_cpus_synced(cs,
45
- * 31 30 27 26 24 22 21 16 15 12 10 5 0
136
ARMMMUIdxBit_E10_1 |
46
- * +------+-------+---+-----+-----+---+----+----+-----+-----+----+-----+
137
ARMMMUIdxBit_E10_1_PAN |
47
- * | size | 1 1 1 | V | 0 0 | A R | 1 | Rs | o3 | opc | 0 0 | Rn | Rt |
138
- ARMMMUIdxBit_E10_0 |
48
- * +------+-------+---+-----+-----+--------+----+-----+-----+----+-----+
139
- ARMMMUIdxBit_Stage2);
49
- *
140
+ ARMMMUIdxBit_E10_0);
50
- * Rt: the result register
141
}
51
- * Rn: base address or SP
142
52
- * Rs: the source register for the operation
143
-static void tlbiipas2_write(CPUARMState *env, const ARMCPRegInfo *ri,
53
- * V: vector flag (always 0 as of v8.3)
144
- uint64_t value)
54
- * A: acquire flag
145
-{
55
- * R: release flag
146
- /* Invalidate by IPA. This has to invalidate any structures that
56
- */
147
- * contain only stage 2 translation information, but does not need
57
-static void disas_ldst_atomic(DisasContext *s, uint32_t insn,
148
- * to apply to structures that contain combined stage 1 and stage 2
58
- int size, int rt, bool is_vector)
149
- * translation information.
59
+
150
- * This must NOP if EL2 isn't implemented or SCR_EL3.NS is zero.
60
+static bool do_atomic_ld(DisasContext *s, arg_atomic *a, AtomicThreeOpFn *fn,
151
- */
61
+ int sign, bool invert)
152
- CPUState *cs = env_cpu(env);
62
{
153
- uint64_t pageaddr;
63
- int rs = extract32(insn, 16, 5);
154
-
64
- int rn = extract32(insn, 5, 5);
155
- if (!arm_feature(env, ARM_FEATURE_EL2) || !(env->cp15.scr_el3 & SCR_NS)) {
65
- int o3_opc = extract32(insn, 12, 4);
66
- bool r = extract32(insn, 22, 1);
67
- bool a = extract32(insn, 23, 1);
68
- TCGv_i64 tcg_rs, tcg_rt, clean_addr;
69
- AtomicThreeOpFn *fn = NULL;
70
- MemOp mop = size;
71
+ MemOp mop = a->sz | sign;
72
+ TCGv_i64 clean_addr, tcg_rs, tcg_rt;
73
74
- if (is_vector || !dc_isar_feature(aa64_atomics, s)) {
75
- unallocated_encoding(s);
156
- return;
76
- return;
157
- }
77
- }
158
-
78
- switch (o3_opc) {
159
- pageaddr = sextract64(value << 12, 0, 40);
79
- case 000: /* LDADD */
160
-
80
- fn = tcg_gen_atomic_fetch_add_i64;
161
- tlb_flush_page_by_mmuidx(cs, pageaddr, ARMMMUIdxBit_Stage2);
81
- break;
162
-}
82
- case 001: /* LDCLR */
163
-
83
- fn = tcg_gen_atomic_fetch_and_i64;
164
-static void tlbiipas2_is_write(CPUARMState *env, const ARMCPRegInfo *ri,
84
- break;
165
- uint64_t value)
85
- case 002: /* LDEOR */
166
-{
86
- fn = tcg_gen_atomic_fetch_xor_i64;
167
- CPUState *cs = env_cpu(env);
87
- break;
168
- uint64_t pageaddr;
88
- case 003: /* LDSET */
169
-
89
- fn = tcg_gen_atomic_fetch_or_i64;
170
- if (!arm_feature(env, ARM_FEATURE_EL2) || !(env->cp15.scr_el3 & SCR_NS)) {
90
- break;
91
- case 004: /* LDSMAX */
92
- fn = tcg_gen_atomic_fetch_smax_i64;
93
- mop |= MO_SIGN;
94
- break;
95
- case 005: /* LDSMIN */
96
- fn = tcg_gen_atomic_fetch_smin_i64;
97
- mop |= MO_SIGN;
98
- break;
99
- case 006: /* LDUMAX */
100
- fn = tcg_gen_atomic_fetch_umax_i64;
101
- break;
102
- case 007: /* LDUMIN */
103
- fn = tcg_gen_atomic_fetch_umin_i64;
104
- break;
105
- case 010: /* SWP */
106
- fn = tcg_gen_atomic_xchg_i64;
107
- break;
108
- case 014: /* LDAPR, LDAPRH, LDAPRB */
109
- if (!dc_isar_feature(aa64_rcpc_8_3, s) ||
110
- rs != 31 || a != 1 || r != 0) {
111
- unallocated_encoding(s);
112
- return;
113
- }
114
- break;
115
- default:
116
- unallocated_encoding(s);
171
- return;
117
- return;
172
- }
118
- }
173
-
119
-
174
- pageaddr = sextract64(value << 12, 0, 40);
120
- if (rn == 31) {
175
-
121
+ if (a->rn == 31) {
176
- tlb_flush_page_by_mmuidx_all_cpus_synced(cs, pageaddr,
122
gen_check_sp_alignment(s);
177
- ARMMMUIdxBit_Stage2);
178
-}
179
180
static void tlbiall_hyp_write(CPUARMState *env, const ARMCPRegInfo *ri,
181
uint64_t value)
182
@@ -XXX,XX +XXX,XX @@ static void vttbr_write(CPUARMState *env, const ARMCPRegInfo *ri,
183
tlb_flush_by_mmuidx(cs,
184
ARMMMUIdxBit_E10_1 |
185
ARMMMUIdxBit_E10_1_PAN |
186
- ARMMMUIdxBit_E10_0 |
187
- ARMMMUIdxBit_Stage2);
188
+ ARMMMUIdxBit_E10_0);
189
raw_write(env, ri, value);
190
}
123
}
191
}
124
-
192
@@ -XXX,XX +XXX,XX @@ static int alle1_tlbmask(CPUARMState *env)
125
- mop = check_atomic_align(s, rn, mop);
193
return ARMMMUIdxBit_SE10_1 |
126
- clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn), false, rn != 31, mop);
194
ARMMMUIdxBit_SE10_1_PAN |
127
-
195
ARMMMUIdxBit_SE10_0;
128
- if (o3_opc == 014) {
196
- } else if (arm_feature(env, ARM_FEATURE_EL2)) {
129
- /*
197
- return ARMMMUIdxBit_E10_1 |
130
- * LDAPR* are a special case because they are a simple load, not a
198
- ARMMMUIdxBit_E10_1_PAN |
131
- * fetch-and-do-something op.
199
- ARMMMUIdxBit_E10_0 |
132
- * The architectural consistency requirements here are weaker than
200
- ARMMMUIdxBit_Stage2;
133
- * full load-acquire (we only need "load-acquire processor consistent"),
201
} else {
134
- * but we choose to implement them as full LDAQ.
202
return ARMMMUIdxBit_E10_1 |
135
- */
203
ARMMMUIdxBit_E10_1_PAN |
136
- do_gpr_ld(s, cpu_reg(s, rt), clean_addr, mop, false,
204
@@ -XXX,XX +XXX,XX @@ static void tlbi_aa64_vae3is_write(CPUARMState *env, const ARMCPRegInfo *ri,
137
- true, rt, disas_ldst_compute_iss_sf(size, false, 0), true);
205
ARMMMUIdxBit_SE3);
138
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
206
}
207
208
-static void tlbi_aa64_ipas2e1_write(CPUARMState *env, const ARMCPRegInfo *ri,
209
- uint64_t value)
210
-{
211
- /* Invalidate by IPA. This has to invalidate any structures that
212
- * contain only stage 2 translation information, but does not need
213
- * to apply to structures that contain combined stage 1 and stage 2
214
- * translation information.
215
- * This must NOP if EL2 isn't implemented or SCR_EL3.NS is zero.
216
- */
217
- ARMCPU *cpu = env_archcpu(env);
218
- CPUState *cs = CPU(cpu);
219
- uint64_t pageaddr;
220
-
221
- if (!arm_feature(env, ARM_FEATURE_EL2) || !(env->cp15.scr_el3 & SCR_NS)) {
222
- return;
139
- return;
223
- }
140
- }
224
-
141
-
225
- pageaddr = sextract64(value << 12, 0, 48);
142
- tcg_rs = read_cpu_reg(s, rs, true);
226
-
143
- tcg_rt = cpu_reg(s, rt);
227
- tlb_flush_page_by_mmuidx(cs, pageaddr, ARMMMUIdxBit_Stage2);
144
-
228
-}
145
- if (o3_opc == 1) { /* LDCLR */
229
-
146
+ mop = check_atomic_align(s, a->rn, mop);
230
-static void tlbi_aa64_ipas2e1is_write(CPUARMState *env, const ARMCPRegInfo *ri,
147
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, a->rn), false,
231
- uint64_t value)
148
+ a->rn != 31, mop);
232
-{
149
+ tcg_rs = read_cpu_reg(s, a->rs, true);
233
- CPUState *cs = env_cpu(env);
150
+ tcg_rt = cpu_reg(s, a->rt);
234
- uint64_t pageaddr;
151
+ if (invert) {
235
-
152
tcg_gen_not_i64(tcg_rs, tcg_rs);
236
- if (!arm_feature(env, ARM_FEATURE_EL2) || !(env->cp15.scr_el3 & SCR_NS)) {
153
}
237
- return;
154
-
238
- }
155
- /* The tcg atomic primitives are all full barriers. Therefore we
239
-
156
+ /*
240
- pageaddr = sextract64(value << 12, 0, 48);
157
+ * The tcg atomic primitives are all full barriers. Therefore we
241
-
158
* can ignore the Acquire and Release bits of this instruction.
242
- tlb_flush_page_by_mmuidx_all_cpus_synced(cs, pageaddr,
159
*/
243
- ARMMMUIdxBit_Stage2);
160
fn(tcg_rt, clean_addr, tcg_rs, get_mem_index(s), mop);
244
-}
161
245
-
162
if (mop & MO_SIGN) {
246
static CPAccessResult aa64_zva_access(CPUARMState *env, const ARMCPRegInfo *ri,
163
- switch (size) {
247
bool isread)
164
+ switch (a->sz) {
248
{
165
case MO_8:
249
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo v8_cp_reginfo[] = {
166
tcg_gen_ext8u_i64(tcg_rt, tcg_rt);
250
.writefn = tlbi_aa64_vae1_write },
167
break;
251
{ .name = "TLBI_IPAS2E1IS", .state = ARM_CP_STATE_AA64,
168
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_atomic(DisasContext *s, uint32_t insn,
252
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 0, .opc2 = 1,
169
g_assert_not_reached();
253
- .access = PL2_W, .type = ARM_CP_NO_RAW,
170
}
254
- .writefn = tlbi_aa64_ipas2e1is_write },
171
}
255
+ .access = PL2_W, .type = ARM_CP_NOP },
172
+ return true;
256
{ .name = "TLBI_IPAS2LE1IS", .state = ARM_CP_STATE_AA64,
173
+}
257
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 0, .opc2 = 5,
174
+
258
- .access = PL2_W, .type = ARM_CP_NO_RAW,
175
+TRANS_FEAT(LDADD, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_add_i64, 0, false)
259
- .writefn = tlbi_aa64_ipas2e1is_write },
176
+TRANS_FEAT(LDCLR, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_and_i64, 0, true)
260
+ .access = PL2_W, .type = ARM_CP_NOP },
177
+TRANS_FEAT(LDEOR, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_xor_i64, 0, false)
261
{ .name = "TLBI_ALLE1IS", .state = ARM_CP_STATE_AA64,
178
+TRANS_FEAT(LDSET, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_or_i64, 0, false)
262
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 3, .opc2 = 4,
179
+TRANS_FEAT(LDSMAX, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_smax_i64, MO_SIGN, false)
263
.access = PL2_W, .type = ARM_CP_NO_RAW,
180
+TRANS_FEAT(LDSMIN, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_smin_i64, MO_SIGN, false)
264
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo v8_cp_reginfo[] = {
181
+TRANS_FEAT(LDUMAX, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_umax_i64, 0, false)
265
.writefn = tlbi_aa64_alle1is_write },
182
+TRANS_FEAT(LDUMIN, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_fetch_umin_i64, 0, false)
266
{ .name = "TLBI_IPAS2E1", .state = ARM_CP_STATE_AA64,
183
+TRANS_FEAT(SWP, aa64_atomics, do_atomic_ld, a, tcg_gen_atomic_xchg_i64, 0, false)
267
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 4, .opc2 = 1,
184
+
268
- .access = PL2_W, .type = ARM_CP_NO_RAW,
185
+static bool trans_LDAPR(DisasContext *s, arg_LDAPR *a)
269
- .writefn = tlbi_aa64_ipas2e1_write },
186
+{
270
+ .access = PL2_W, .type = ARM_CP_NOP },
187
+ bool iss_sf = ldst_iss_sf(a->sz, false, false);
271
{ .name = "TLBI_IPAS2LE1", .state = ARM_CP_STATE_AA64,
188
+ TCGv_i64 clean_addr;
272
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 4, .opc2 = 5,
189
+ MemOp mop;
273
- .access = PL2_W, .type = ARM_CP_NO_RAW,
190
+
274
- .writefn = tlbi_aa64_ipas2e1_write },
191
+ if (!dc_isar_feature(aa64_atomics, s) ||
275
+ .access = PL2_W, .type = ARM_CP_NOP },
192
+ !dc_isar_feature(aa64_rcpc_8_3, s)) {
276
{ .name = "TLBI_ALLE1", .state = ARM_CP_STATE_AA64,
193
+ return false;
277
.opc0 = 1, .opc1 = 4, .crn = 8, .crm = 7, .opc2 = 4,
194
+ }
278
.access = PL2_W, .type = ARM_CP_NO_RAW,
195
+ if (a->rn == 31) {
279
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo v8_cp_reginfo[] = {
196
+ gen_check_sp_alignment(s);
280
.writefn = tlbimva_hyp_is_write },
197
+ }
281
{ .name = "TLBIIPAS2",
198
+ mop = check_atomic_align(s, a->rn, a->sz);
282
.cp = 15, .opc1 = 4, .crn = 8, .crm = 4, .opc2 = 1,
199
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, a->rn), false,
283
- .type = ARM_CP_NO_RAW, .access = PL2_W,
200
+ a->rn != 31, mop);
284
- .writefn = tlbiipas2_write },
201
+ /*
285
+ .type = ARM_CP_NOP, .access = PL2_W },
202
+ * LDAPR* are a special case because they are a simple load, not a
286
{ .name = "TLBIIPAS2IS",
203
+ * fetch-and-do-something op.
287
.cp = 15, .opc1 = 4, .crn = 8, .crm = 0, .opc2 = 1,
204
+ * The architectural consistency requirements here are weaker than
288
- .type = ARM_CP_NO_RAW, .access = PL2_W,
205
+ * full load-acquire (we only need "load-acquire processor consistent"),
289
- .writefn = tlbiipas2_is_write },
206
+ * but we choose to implement them as full LDAQ.
290
+ .type = ARM_CP_NOP, .access = PL2_W },
207
+ */
291
{ .name = "TLBIIPAS2L",
208
+ do_gpr_ld(s, cpu_reg(s, a->rt), clean_addr, mop, false,
292
.cp = 15, .opc1 = 4, .crn = 8, .crm = 4, .opc2 = 5,
209
+ true, a->rt, iss_sf, true);
293
- .type = ARM_CP_NO_RAW, .access = PL2_W,
210
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
294
- .writefn = tlbiipas2_write },
211
+ return true;
295
+ .type = ARM_CP_NOP, .access = PL2_W },
212
}
296
{ .name = "TLBIIPAS2LIS",
213
297
.cp = 15, .opc1 = 4, .crn = 8, .crm = 0, .opc2 = 5,
214
/*
298
- .type = ARM_CP_NO_RAW, .access = PL2_W,
215
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg(DisasContext *s, uint32_t insn)
299
- .writefn = tlbiipas2_is_write },
216
}
300
+ .type = ARM_CP_NOP, .access = PL2_W },
217
switch (extract32(insn, 10, 2)) {
301
/* 32 bit cache operations */
218
case 0:
302
{ .name = "ICIALLUIS", .cp = 15, .opc1 = 0, .crn = 7, .crm = 1, .opc2 = 0,
219
- disas_ldst_atomic(s, insn, size, rt, is_vector);
303
.type = ARM_CP_NOP, .access = PL1_W, .accessfn = aa64_cacheop_pou_access },
220
- return;
221
case 2:
222
break;
223
default:
304
--
224
--
305
2.20.1
225
2.34.1
306
307
diff view generated by jsdifflib
1
Convert the VFM[AS]L (scalar) insns in the 2reg-scalar-ext group
1
Convert the instructions in the load/store register (pointer
2
to decodetree. These are the last ones in the group so we can remove
2
authentication) group ot decodetree: LDRAA, LDRAB.
3
all the legacy decode for the group.
4
5
Note that in disas_thumb2_insn() the parts of this encoding space
6
where the decodetree decoder returns false will correctly be directed
7
to illegal_op by the "(insn & (1 << 28))" check so they won't fall
8
into disas_coproc_insn() by mistake.
9
3
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
Message-id: 20200430181003.21682-11-peter.maydell@linaro.org
7
Message-id: 20230602155223.2040685-17-peter.maydell@linaro.org
13
---
8
---
14
target/arm/neon-shared.decode | 7 +++
9
target/arm/tcg/a64.decode | 7 +++
15
target/arm/translate-neon.inc.c | 32 ++++++++++
10
target/arm/tcg/translate-a64.c | 83 +++++++---------------------------
16
target/arm/translate.c | 107 +-------------------------------
11
2 files changed, 23 insertions(+), 67 deletions(-)
17
3 files changed, 40 insertions(+), 106 deletions(-)
18
12
19
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
13
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
20
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
21
--- a/target/arm/neon-shared.decode
15
--- a/target/arm/tcg/a64.decode
22
+++ b/target/arm/neon-shared.decode
16
+++ b/target/arm/tcg/a64.decode
23
@@ -XXX,XX +XXX,XX @@ VCMLA_scalar 1111 1110 1 . rot:2 .... .... 1000 . q:1 . 0 .... \
17
@@ -XXX,XX +XXX,XX @@ LDUMIN .. 111 0 00 . . 1 ..... 0111 00 ..... ..... @atomic
24
18
SWP .. 111 0 00 . . 1 ..... 1000 00 ..... ..... @atomic
25
VDOT_scalar 1111 1110 0 . 10 .... .... 1101 . q:1 index:1 u:1 rm:4 \
19
26
vm=%vm_dp vn=%vn_dp vd=%vd_dp
20
LDAPR sz:2 111 0 00 1 0 1 11111 1100 00 rn:5 rt:5
27
+
21
+
28
+%vfml_scalar_q0_rm 0:3 5:1
22
+# Load/store register (pointer authentication)
29
+%vfml_scalar_q1_index 5:1 3:1
23
+
30
+VFML_scalar 1111 1110 0 . 0 s:1 .... .... 1000 . 0 . 1 index:1 ... \
24
+# LDRA immediate is 10 bits signed and scaled, but the bits aren't all contiguous
31
+ rm=%vfml_scalar_q0_rm vn=%vn_sp vd=%vd_dp q=0
25
+%ldra_imm 22:s1 12:9 !function=times_2
32
+VFML_scalar 1111 1110 0 . 0 s:1 .... .... 1000 . 1 . 1 . rm:3 \
26
+
33
+ index=%vfml_scalar_q1_index vn=%vn_dp vd=%vd_dp q=1
27
+LDRA 11 111 0 00 m:1 . 1 ......... w:1 1 rn:5 rt:5 imm=%ldra_imm
34
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
28
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
35
index XXXXXXX..XXXXXXX 100644
29
index XXXXXXX..XXXXXXX 100644
36
--- a/target/arm/translate-neon.inc.c
30
--- a/target/arm/tcg/translate-a64.c
37
+++ b/target/arm/translate-neon.inc.c
31
+++ b/target/arm/tcg/translate-a64.c
38
@@ -XXX,XX +XXX,XX @@ static bool trans_VDOT_scalar(DisasContext *s, arg_VDOT_scalar *a)
32
@@ -XXX,XX +XXX,XX @@ static bool trans_LDAPR(DisasContext *s, arg_LDAPR *a)
39
tcg_temp_free_ptr(fpst);
40
return true;
33
return true;
41
}
34
}
42
+
35
43
+static bool trans_VFML_scalar(DisasContext *s, arg_VFML_scalar *a)
36
-/*
44
+{
37
- * PAC memory operations
45
+ int opr_sz;
38
- *
46
+
39
- * 31 30 27 26 24 22 21 12 11 10 5 0
47
+ if (!dc_isar_feature(aa32_fhm, s)) {
40
- * +------+-------+---+-----+-----+---+--------+---+---+----+-----+
41
- * | size | 1 1 1 | V | 0 0 | M S | 1 | imm9 | W | 1 | Rn | Rt |
42
- * +------+-------+---+-----+-----+---+--------+---+---+----+-----+
43
- *
44
- * Rt: the result register
45
- * Rn: base address or SP
46
- * V: vector flag (always 0 as of v8.3)
47
- * M: clear for key DA, set for key DB
48
- * W: pre-indexing flag
49
- * S: sign for imm9.
50
- */
51
-static void disas_ldst_pac(DisasContext *s, uint32_t insn,
52
- int size, int rt, bool is_vector)
53
+static bool trans_LDRA(DisasContext *s, arg_LDRA *a)
54
{
55
- int rn = extract32(insn, 5, 5);
56
- bool is_wback = extract32(insn, 11, 1);
57
- bool use_key_a = !extract32(insn, 23, 1);
58
- int offset;
59
TCGv_i64 clean_addr, dirty_addr, tcg_rt;
60
MemOp memop;
61
62
- if (size != 3 || is_vector || !dc_isar_feature(aa64_pauth, s)) {
63
- unallocated_encoding(s);
64
- return;
65
+ /* Load with pointer authentication */
66
+ if (!dc_isar_feature(aa64_pauth, s)) {
48
+ return false;
67
+ return false;
49
+ }
68
}
50
+
69
51
+ /* UNDEF accesses to D16-D31 if they don't exist. */
70
- if (rn == 31) {
52
+ if (!dc_isar_feature(aa32_simd_r32, s) &&
71
+ if (a->rn == 31) {
53
+ ((a->vd & 0x10) || (a->q && (a->vn & 0x10)))) {
72
gen_check_sp_alignment(s);
54
+ return false;
73
}
55
+ }
74
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
56
+
75
+ dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
57
+ if (a->vd & a->q) {
76
58
+ return false;
77
if (s->pauth_active) {
59
+ }
78
- if (use_key_a) {
60
+
79
+ if (!a->m) {
61
+ if (!vfp_access_check(s)) {
80
gen_helper_autda(dirty_addr, cpu_env, dirty_addr,
62
+ return true;
81
tcg_constant_i64(0));
63
+ }
82
} else {
64
+
83
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pac(DisasContext *s, uint32_t insn,
65
+ opr_sz = (1 + a->q) * 8;
84
}
66
+ tcg_gen_gvec_3_ptr(vfp_reg_offset(1, a->vd),
85
}
67
+ vfp_reg_offset(a->q, a->vn),
86
68
+ vfp_reg_offset(a->q, a->rm),
87
- /* Form the 10-bit signed, scaled offset. */
69
+ cpu_env, opr_sz, opr_sz,
88
- offset = (extract32(insn, 22, 1) << 9) | extract32(insn, 12, 9);
70
+ (a->index << 2) | a->s, /* is_2 == 0 */
89
- offset = sextract32(offset << size, 0, 10 + size);
71
+ gen_helper_gvec_fmlal_idx_a32);
90
- tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
91
+ tcg_gen_addi_i64(dirty_addr, dirty_addr, a->imm);
92
93
- memop = finalize_memop(s, size);
94
+ memop = finalize_memop(s, MO_64);
95
96
/* Note that "clean" and "dirty" here refer to TBI not PAC. */
97
clean_addr = gen_mte_check1(s, dirty_addr, false,
98
- is_wback || rn != 31, memop);
99
+ a->w || a->rn != 31, memop);
100
101
- tcg_rt = cpu_reg(s, rt);
102
+ tcg_rt = cpu_reg(s, a->rt);
103
do_gpr_ld(s, tcg_rt, clean_addr, memop,
104
- /* extend */ false, /* iss_valid */ !is_wback,
105
- /* iss_srt */ rt, /* iss_sf */ true, /* iss_ar */ false);
106
+ /* extend */ false, /* iss_valid */ !a->w,
107
+ /* iss_srt */ a->rt, /* iss_sf */ true, /* iss_ar */ false);
108
109
- if (is_wback) {
110
- tcg_gen_mov_i64(cpu_reg_sp(s, rn), dirty_addr);
111
+ if (a->w) {
112
+ tcg_gen_mov_i64(cpu_reg_sp(s, a->rn), dirty_addr);
113
}
72
+ return true;
114
+ return true;
73
+}
74
diff --git a/target/arm/translate.c b/target/arm/translate.c
75
index XXXXXXX..XXXXXXX 100644
76
--- a/target/arm/translate.c
77
+++ b/target/arm/translate.c
78
@@ -XXX,XX +XXX,XX @@ static int disas_dsp_insn(DisasContext *s, uint32_t insn)
79
}
115
}
80
116
81
#define VFP_REG_SHR(x, n) (((n) > 0) ? (x) >> (n) : (x) << -(n))
117
/*
82
-#define VFP_SREG(insn, bigbit, smallbit) \
118
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_ldapr_stlr(DisasContext *s, uint32_t insn)
83
- ((VFP_REG_SHR(insn, bigbit - 1) & 0x1e) | (((insn) >> (smallbit)) & 1))
119
}
84
#define VFP_DREG(reg, insn, bigbit, smallbit) do { \
85
if (dc_isar_feature(aa32_simd_r32, s)) { \
86
reg = (((insn) >> (bigbit)) & 0x0f) \
87
@@ -XXX,XX +XXX,XX @@ static int disas_dsp_insn(DisasContext *s, uint32_t insn)
88
reg = ((insn) >> (bigbit)) & 0x0f; \
89
}} while (0)
90
91
-#define VFP_SREG_D(insn) VFP_SREG(insn, 12, 22)
92
#define VFP_DREG_D(reg, insn) VFP_DREG(reg, insn, 12, 22)
93
-#define VFP_SREG_N(insn) VFP_SREG(insn, 16, 7)
94
#define VFP_DREG_N(reg, insn) VFP_DREG(reg, insn, 16, 7)
95
-#define VFP_SREG_M(insn) VFP_SREG(insn, 0, 5)
96
#define VFP_DREG_M(reg, insn) VFP_DREG(reg, insn, 0, 5)
97
98
static void gen_neon_dup_low16(TCGv_i32 var)
99
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
100
return 0;
101
}
120
}
102
121
103
-/* Advanced SIMD two registers and a scalar extension.
122
-/* Load/store register (all forms) */
104
- * 31 24 23 22 20 16 12 11 10 9 8 3 0
123
-static void disas_ldst_reg(DisasContext *s, uint32_t insn)
105
- * +-----------------+----+---+----+----+----+---+----+---+----+---------+----+
124
-{
106
- * | 1 1 1 1 1 1 1 0 | o1 | D | o2 | Vn | Vd | 1 | o3 | 0 | o4 | N Q M U | Vm |
125
- int rt = extract32(insn, 0, 5);
107
- * +-----------------+----+---+----+----+----+---+----+---+----+---------+----+
126
- bool is_vector = extract32(insn, 26, 1);
108
- *
127
- int size = extract32(insn, 30, 2);
109
- */
110
-
128
-
111
-static int disas_neon_insn_2reg_scalar_ext(DisasContext *s, uint32_t insn)
129
- switch (extract32(insn, 24, 2)) {
112
-{
130
- case 0:
113
- gen_helper_gvec_3 *fn_gvec = NULL;
131
- if (extract32(insn, 21, 1) == 0) {
114
- gen_helper_gvec_3_ptr *fn_gvec_ptr = NULL;
132
- break;
115
- int rd, rn, rm, opr_sz, data;
116
- int off_rn, off_rm;
117
- bool is_long = false, q = extract32(insn, 6, 1);
118
- bool ptr_is_env = false;
119
-
120
- if ((insn & 0xffa00f10) == 0xfe000810) {
121
- /* VFM[AS]L -- 1111 1110 0.0S .... .... 1000 .Q.1 .... */
122
- int is_s = extract32(insn, 20, 1);
123
- int vm20 = extract32(insn, 0, 3);
124
- int vm3 = extract32(insn, 3, 1);
125
- int m = extract32(insn, 5, 1);
126
- int index;
127
-
128
- if (!dc_isar_feature(aa32_fhm, s)) {
129
- return 1;
130
- }
133
- }
131
- if (q) {
134
- switch (extract32(insn, 10, 2)) {
132
- rm = vm20;
135
- case 0:
133
- index = m * 2 + vm3;
136
- case 2:
134
- } else {
137
- break;
135
- rm = vm20 * 2 + m;
138
- default:
136
- index = vm3;
139
- disas_ldst_pac(s, insn, size, rt, is_vector);
140
- return;
137
- }
141
- }
138
- is_long = true;
142
- break;
139
- data = (index << 2) | is_s; /* is_2 == 0 */
140
- fn_gvec_ptr = gen_helper_gvec_fmlal_idx_a32;
141
- ptr_is_env = true;
142
- } else {
143
- return 1;
144
- }
143
- }
145
-
144
- unallocated_encoding(s);
146
- VFP_DREG_D(rd, insn);
147
- if (rd & q) {
148
- return 1;
149
- }
150
- if (q || !is_long) {
151
- VFP_DREG_N(rn, insn);
152
- if (rn & q & !is_long) {
153
- return 1;
154
- }
155
- off_rn = vfp_reg_offset(1, rn);
156
- off_rm = vfp_reg_offset(1, rm);
157
- } else {
158
- rn = VFP_SREG_N(insn);
159
- off_rn = vfp_reg_offset(0, rn);
160
- off_rm = vfp_reg_offset(0, rm);
161
- }
162
- if (s->fp_excp_el) {
163
- gen_exception_insn(s, s->pc_curr, EXCP_UDEF,
164
- syn_simd_access_trap(1, 0xe, false), s->fp_excp_el);
165
- return 0;
166
- }
167
- if (!s->vfp_enabled) {
168
- return 1;
169
- }
170
-
171
- opr_sz = (1 + q) * 8;
172
- if (fn_gvec_ptr) {
173
- TCGv_ptr ptr;
174
- if (ptr_is_env) {
175
- ptr = cpu_env;
176
- } else {
177
- ptr = get_fpstatus_ptr(1);
178
- }
179
- tcg_gen_gvec_3_ptr(vfp_reg_offset(1, rd), off_rn, off_rm, ptr,
180
- opr_sz, opr_sz, data, fn_gvec_ptr);
181
- if (!ptr_is_env) {
182
- tcg_temp_free_ptr(ptr);
183
- }
184
- } else {
185
- tcg_gen_gvec_3_ool(vfp_reg_offset(1, rd), off_rn, off_rm,
186
- opr_sz, opr_sz, data, fn_gvec);
187
- }
188
- return 0;
189
-}
145
-}
190
-
146
-
191
static int disas_coproc_insn(DisasContext *s, uint32_t insn)
147
/* AdvSIMD load/store multiple structures
148
*
149
* 31 30 29 23 22 21 16 15 12 11 10 9 5 4 0
150
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
151
static void disas_ldst(DisasContext *s, uint32_t insn)
192
{
152
{
193
int cpnum, is64, crn, crm, opc1, opc2, isread, rt, rt2;
153
switch (extract32(insn, 24, 6)) {
194
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
154
- case 0x38: case 0x39:
195
}
155
- case 0x3c: case 0x3d: /* Load/store register (all forms) */
196
}
156
- disas_ldst_reg(s, insn);
197
}
157
- break;
198
- } else if ((insn & 0x0f000a00) == 0x0e000800
158
case 0x0c: /* AdvSIMD load/store multiple structures */
199
- && arm_dc_feature(s, ARM_FEATURE_V8)) {
159
disas_ldst_multiple_struct(s, insn);
200
- if (disas_neon_insn_2reg_scalar_ext(s, insn)) {
160
break;
201
- goto illegal_op;
202
- }
203
- return;
204
}
205
goto illegal_op;
206
}
207
@@ -XXX,XX +XXX,XX @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn)
208
}
209
break;
210
}
211
- if ((insn & 0xff000a00) == 0xfe000800
212
- && arm_dc_feature(s, ARM_FEATURE_V8)) {
213
- /* The Thumb2 and ARM encodings are identical. */
214
- if (disas_neon_insn_2reg_scalar_ext(s, insn)) {
215
- goto illegal_op;
216
- }
217
- } else if (((insn >> 24) & 3) == 3) {
218
+ if (((insn >> 24) & 3) == 3) {
219
/* Translate into the equivalent ARM encoding. */
220
insn = (insn & 0xe2ffffff) | ((insn & (1 << 28)) >> 4) | (1 << 28);
221
if (disas_neon_data_insn(s, insn)) {
222
--
161
--
223
2.20.1
162
2.34.1
224
163
225
164
diff view generated by jsdifflib
1
Convert the Neon 3-reg-same VMAX and VMIN insns to decodetree.
1
Convert the instructions in the LDAPR/STLR (unscaled immediate)
2
group to decodetree.
2
3
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200430181003.21682-17-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-18-peter.maydell@linaro.org
6
---
7
---
7
target/arm/neon-dp.decode | 5 +++++
8
target/arm/tcg/a64.decode | 10 +++
8
target/arm/translate-neon.inc.c | 14 ++++++++++++++
9
target/arm/tcg/translate-a64.c | 132 ++++++++++++---------------------
9
target/arm/translate.c | 21 ++-------------------
10
2 files changed, 56 insertions(+), 86 deletions(-)
10
3 files changed, 21 insertions(+), 19 deletions(-)
11
11
12
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/neon-dp.decode
14
--- a/target/arm/tcg/a64.decode
15
+++ b/target/arm/neon-dp.decode
15
+++ b/target/arm/tcg/a64.decode
16
@@ -XXX,XX +XXX,XX @@ VBSL_3s 1111 001 1 0 . 01 .... .... 0001 ... 1 .... @3same_logic
16
@@ -XXX,XX +XXX,XX @@ LDAPR sz:2 111 0 00 1 0 1 11111 1100 00 rn:5 rt:5
17
VBIT_3s 1111 001 1 0 . 10 .... .... 0001 ... 1 .... @3same_logic
17
%ldra_imm 22:s1 12:9 !function=times_2
18
VBIF_3s 1111 001 1 0 . 11 .... .... 0001 ... 1 .... @3same_logic
18
19
19
LDRA 11 111 0 00 m:1 . 1 ......... w:1 1 rn:5 rt:5 imm=%ldra_imm
20
+VMAX_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 0 .... @3same
20
+
21
+VMAX_U_3s 1111 001 1 0 . .. .... .... 0110 . . . 0 .... @3same
21
+&ldapr_stlr_i rn rt imm sz sign ext
22
+VMIN_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 1 .... @3same
22
+@ldapr_stlr_i .. ...... .. . imm:9 .. rn:5 rt:5 &ldapr_stlr_i
23
+VMIN_U_3s 1111 001 1 0 . .. .... .... 0110 . . . 1 .... @3same
23
+STLR_i sz:2 011001 00 0 ......... 00 ..... ..... @ldapr_stlr_i sign=0 ext=0
24
+
24
+LDAPR_i sz:2 011001 01 0 ......... 00 ..... ..... @ldapr_stlr_i sign=0 ext=0
25
VADD_3s 1111 001 0 0 . .. .... .... 1000 . . . 0 .... @3same
25
+LDAPR_i 00 011001 10 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=0 sz=0
26
VSUB_3s 1111 001 1 0 . .. .... .... 1000 . . . 0 .... @3same
26
+LDAPR_i 01 011001 10 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=0 sz=1
27
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
27
+LDAPR_i 10 011001 10 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=0 sz=2
28
+LDAPR_i 00 011001 11 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=1 sz=0
29
+LDAPR_i 01 011001 11 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=1 sz=1
30
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
28
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
29
--- a/target/arm/translate-neon.inc.c
32
--- a/target/arm/tcg/translate-a64.c
30
+++ b/target/arm/translate-neon.inc.c
33
+++ b/target/arm/tcg/translate-a64.c
31
@@ -XXX,XX +XXX,XX @@ DO_3SAME(VEOR, tcg_gen_gvec_xor)
34
@@ -XXX,XX +XXX,XX @@ static void gen_compare_and_swap_pair(DisasContext *s, int rs, int rt,
32
DO_3SAME_BITSEL(VBSL, rd_ofs, rn_ofs, rm_ofs)
35
}
33
DO_3SAME_BITSEL(VBIT, rm_ofs, rn_ofs, rd_ofs)
36
}
34
DO_3SAME_BITSEL(VBIF, rm_ofs, rd_ofs, rn_ofs)
37
35
+
38
-/* Update the Sixty-Four bit (SF) registersize. This logic is derived
36
+#define DO_3SAME_NO_SZ_3(INSN, FUNC) \
39
+/*
37
+ static bool trans_##INSN##_3s(DisasContext *s, arg_3same *a) \
40
+ * Compute the ISS.SF bit for syndrome information if an exception
38
+ { \
41
+ * is taken on a load or store. This indicates whether the instruction
39
+ if (a->size == 3) { \
42
+ * is accessing a 32-bit or 64-bit register. This logic is derived
40
+ return false; \
43
* from the ARMv8 specs for LDR (Shared decode for all encodings).
41
+ } \
44
*/
42
+ return do_3same(s, a, FUNC); \
45
-static bool disas_ldst_compute_iss_sf(int size, bool is_signed, int opc)
46
-{
47
- int opc0 = extract32(opc, 0, 1);
48
- int regsize;
49
-
50
- if (is_signed) {
51
- regsize = opc0 ? 32 : 64;
52
- } else {
53
- regsize = size == 3 ? 64 : 32;
54
- }
55
- return regsize == 64;
56
-}
57
-
58
static bool ldst_iss_sf(int size, bool sign, bool ext)
59
{
60
61
@@ -XXX,XX +XXX,XX @@ static bool trans_LDRA(DisasContext *s, arg_LDRA *a)
62
return true;
63
}
64
65
-/*
66
- * LDAPR/STLR (unscaled immediate)
67
- *
68
- * 31 30 24 22 21 12 10 5 0
69
- * +------+-------------+-----+---+--------+-----+----+-----+
70
- * | size | 0 1 1 0 0 1 | opc | 0 | imm9 | 0 0 | Rn | Rt |
71
- * +------+-------------+-----+---+--------+-----+----+-----+
72
- *
73
- * Rt: source or destination register
74
- * Rn: base register
75
- * imm9: unscaled immediate offset
76
- * opc: 00: STLUR*, 01/10/11: various LDAPUR*
77
- * size: size of load/store
78
- */
79
-static void disas_ldst_ldapr_stlr(DisasContext *s, uint32_t insn)
80
+static bool trans_LDAPR_i(DisasContext *s, arg_ldapr_stlr_i *a)
81
{
82
- int rt = extract32(insn, 0, 5);
83
- int rn = extract32(insn, 5, 5);
84
- int offset = sextract32(insn, 12, 9);
85
- int opc = extract32(insn, 22, 2);
86
- int size = extract32(insn, 30, 2);
87
TCGv_i64 clean_addr, dirty_addr;
88
- bool is_store = false;
89
- bool extend = false;
90
- bool iss_sf;
91
- MemOp mop = size;
92
+ MemOp mop = a->sz | (a->sign ? MO_SIGN : 0);
93
+ bool iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
94
95
if (!dc_isar_feature(aa64_rcpc_8_4, s)) {
96
- unallocated_encoding(s);
97
- return;
98
+ return false;
99
}
100
101
- switch (opc) {
102
- case 0: /* STLURB */
103
- is_store = true;
104
- break;
105
- case 1: /* LDAPUR* */
106
- break;
107
- case 2: /* LDAPURS* 64-bit variant */
108
- if (size == 3) {
109
- unallocated_encoding(s);
110
- return;
111
- }
112
- mop |= MO_SIGN;
113
- break;
114
- case 3: /* LDAPURS* 32-bit variant */
115
- if (size > 1) {
116
- unallocated_encoding(s);
117
- return;
118
- }
119
- mop |= MO_SIGN;
120
- extend = true; /* zero-extend 32->64 after signed load */
121
- break;
122
- default:
123
- g_assert_not_reached();
124
- }
125
-
126
- iss_sf = disas_ldst_compute_iss_sf(size, (mop & MO_SIGN) != 0, opc);
127
-
128
- if (rn == 31) {
129
+ if (a->rn == 31) {
130
gen_check_sp_alignment(s);
131
}
132
133
- mop = check_ordered_align(s, rn, offset, is_store, mop);
134
-
135
- dirty_addr = read_cpu_reg_sp(s, rn, 1);
136
- tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
137
+ mop = check_ordered_align(s, a->rn, a->imm, false, mop);
138
+ dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
139
+ tcg_gen_addi_i64(dirty_addr, dirty_addr, a->imm);
140
clean_addr = clean_data_tbi(s, dirty_addr);
141
142
- if (is_store) {
143
- /* Store-Release semantics */
144
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
145
- do_gpr_st(s, cpu_reg(s, rt), clean_addr, mop, true, rt, iss_sf, true);
146
- } else {
147
- /*
148
- * Load-AcquirePC semantics; we implement as the slightly more
149
- * restrictive Load-Acquire.
150
- */
151
- do_gpr_ld(s, cpu_reg(s, rt), clean_addr, mop,
152
- extend, true, rt, iss_sf, true);
153
- tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
154
+ /*
155
+ * Load-AcquirePC semantics; we implement as the slightly more
156
+ * restrictive Load-Acquire.
157
+ */
158
+ do_gpr_ld(s, cpu_reg(s, a->rt), clean_addr, mop, a->ext, true,
159
+ a->rt, iss_sf, true);
160
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
161
+ return true;
162
+}
163
+
164
+static bool trans_STLR_i(DisasContext *s, arg_ldapr_stlr_i *a)
165
+{
166
+ TCGv_i64 clean_addr, dirty_addr;
167
+ MemOp mop = a->sz;
168
+ bool iss_sf = ldst_iss_sf(a->sz, a->sign, a->ext);
169
+
170
+ if (!dc_isar_feature(aa64_rcpc_8_4, s)) {
171
+ return false;
172
}
173
+
174
+ /* TODO: ARMv8.4-LSE SCTLR.nAA */
175
+
176
+ if (a->rn == 31) {
177
+ gen_check_sp_alignment(s);
43
+ }
178
+ }
44
+
179
+
45
+DO_3SAME_NO_SZ_3(VMAX_S, tcg_gen_gvec_smax)
180
+ mop = check_ordered_align(s, a->rn, a->imm, true, mop);
46
+DO_3SAME_NO_SZ_3(VMAX_U, tcg_gen_gvec_umax)
181
+ dirty_addr = read_cpu_reg_sp(s, a->rn, 1);
47
+DO_3SAME_NO_SZ_3(VMIN_S, tcg_gen_gvec_smin)
182
+ tcg_gen_addi_i64(dirty_addr, dirty_addr, a->imm);
48
+DO_3SAME_NO_SZ_3(VMIN_U, tcg_gen_gvec_umin)
183
+ clean_addr = clean_data_tbi(s, dirty_addr);
49
diff --git a/target/arm/translate.c b/target/arm/translate.c
184
+
50
index XXXXXXX..XXXXXXX 100644
185
+ /* Store-Release semantics */
51
--- a/target/arm/translate.c
186
+ tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
52
+++ b/target/arm/translate.c
187
+ do_gpr_st(s, cpu_reg(s, a->rt), clean_addr, mop, true, a->rt, iss_sf, true);
53
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
188
+ return true;
54
rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size);
189
}
55
return 0;
190
56
191
/* AdvSIMD load/store multiple structures
57
- case NEON_3R_VMAX:
192
@@ -XXX,XX +XXX,XX @@ static void disas_ldst(DisasContext *s, uint32_t insn)
58
- if (u) {
193
case 0x19:
59
- tcg_gen_gvec_umax(size, rd_ofs, rn_ofs, rm_ofs,
194
if (extract32(insn, 21, 1) != 0) {
60
- vec_size, vec_size);
195
disas_ldst_tag(s, insn);
61
- } else {
196
- } else if (extract32(insn, 10, 2) == 0) {
62
- tcg_gen_gvec_smax(size, rd_ofs, rn_ofs, rm_ofs,
197
- disas_ldst_ldapr_stlr(s, insn);
63
- vec_size, vec_size);
198
} else {
64
- }
199
unallocated_encoding(s);
65
- return 0;
66
- case NEON_3R_VMIN:
67
- if (u) {
68
- tcg_gen_gvec_umin(size, rd_ofs, rn_ofs, rm_ofs,
69
- vec_size, vec_size);
70
- } else {
71
- tcg_gen_gvec_smin(size, rd_ofs, rn_ofs, rm_ofs,
72
- vec_size, vec_size);
73
- }
74
- return 0;
75
-
76
case NEON_3R_VSHL:
77
/* Note the operation is vshl vd,vm,vn */
78
tcg_gen_gvec_3(rd_ofs, rm_ofs, rn_ofs, vec_size, vec_size,
79
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
80
81
case NEON_3R_VADD_VSUB:
82
case NEON_3R_LOGIC:
83
+ case NEON_3R_VMAX:
84
+ case NEON_3R_VMIN:
85
/* Already handled by decodetree */
86
return 1;
87
}
200
}
88
--
201
--
89
2.20.1
202
2.34.1
90
91
diff view generated by jsdifflib
1
Convert the Neon "load/store multiple structures" insns to decodetree.
1
Convert the instructions in the ASIMD load/store multiple structures
2
instruction classes to decodetree.
2
3
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200430181003.21682-12-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-19-peter.maydell@linaro.org
6
---
7
---
7
target/arm/neon-ls.decode | 7 ++
8
target/arm/tcg/a64.decode | 20 +++
8
target/arm/translate-neon.inc.c | 124 ++++++++++++++++++++++++++++++++
9
target/arm/tcg/translate-a64.c | 222 ++++++++++++++++-----------------
9
target/arm/translate.c | 91 +----------------------
10
2 files changed, 131 insertions(+), 111 deletions(-)
10
3 files changed, 133 insertions(+), 89 deletions(-)
11
11
12
diff --git a/target/arm/neon-ls.decode b/target/arm/neon-ls.decode
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
13
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/neon-ls.decode
14
--- a/target/arm/tcg/a64.decode
15
+++ b/target/arm/neon-ls.decode
15
+++ b/target/arm/tcg/a64.decode
16
@@ -XXX,XX +XXX,XX @@
16
@@ -XXX,XX +XXX,XX @@ LDAPR_i 01 011001 10 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext
17
# 0b1111_1001_xxx0_xxxx_xxxx_xxxx_xxxx_xxxx
17
LDAPR_i 10 011001 10 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=0 sz=2
18
# This file works on the A32 encoding only; calling code for T32 has to
18
LDAPR_i 00 011001 11 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=1 sz=0
19
# transform the insn into the A32 version first.
19
LDAPR_i 01 011001 11 0 ......... 00 ..... ..... @ldapr_stlr_i sign=1 ext=1 sz=1
20
+
20
+
21
+%vd_dp 22:1 12:4
21
+# Load/store multiple structures
22
+
22
+# The 4-bit opcode in [15:12] encodes repeat count and structure elements
23
+# Neon load/store multiple structures
23
+&ldst_mult rm rn rt sz q p rpt selem
24
+
24
+@ldst_mult . q:1 ...... p:1 . . rm:5 .... sz:2 rn:5 rt:5 &ldst_mult
25
+VLDST_multiple 1111 0100 0 . l:1 0 rn:4 .... itype:4 size:2 align:2 rm:4 \
25
+ST_mult 0 . 001100 . 0 0 ..... 0000 .. ..... ..... @ldst_mult rpt=1 selem=4
26
+ vd=%vd_dp
26
+ST_mult 0 . 001100 . 0 0 ..... 0010 .. ..... ..... @ldst_mult rpt=4 selem=1
27
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
27
+ST_mult 0 . 001100 . 0 0 ..... 0100 .. ..... ..... @ldst_mult rpt=1 selem=3
28
+ST_mult 0 . 001100 . 0 0 ..... 0110 .. ..... ..... @ldst_mult rpt=3 selem=1
29
+ST_mult 0 . 001100 . 0 0 ..... 0111 .. ..... ..... @ldst_mult rpt=1 selem=1
30
+ST_mult 0 . 001100 . 0 0 ..... 1000 .. ..... ..... @ldst_mult rpt=1 selem=2
31
+ST_mult 0 . 001100 . 0 0 ..... 1010 .. ..... ..... @ldst_mult rpt=2 selem=1
32
+
33
+LD_mult 0 . 001100 . 1 0 ..... 0000 .. ..... ..... @ldst_mult rpt=1 selem=4
34
+LD_mult 0 . 001100 . 1 0 ..... 0010 .. ..... ..... @ldst_mult rpt=4 selem=1
35
+LD_mult 0 . 001100 . 1 0 ..... 0100 .. ..... ..... @ldst_mult rpt=1 selem=3
36
+LD_mult 0 . 001100 . 1 0 ..... 0110 .. ..... ..... @ldst_mult rpt=3 selem=1
37
+LD_mult 0 . 001100 . 1 0 ..... 0111 .. ..... ..... @ldst_mult rpt=1 selem=1
38
+LD_mult 0 . 001100 . 1 0 ..... 1000 .. ..... ..... @ldst_mult rpt=1 selem=2
39
+LD_mult 0 . 001100 . 1 0 ..... 1010 .. ..... ..... @ldst_mult rpt=2 selem=1
40
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
28
index XXXXXXX..XXXXXXX 100644
41
index XXXXXXX..XXXXXXX 100644
29
--- a/target/arm/translate-neon.inc.c
42
--- a/target/arm/tcg/translate-a64.c
30
+++ b/target/arm/translate-neon.inc.c
43
+++ b/target/arm/tcg/translate-a64.c
31
@@ -XXX,XX +XXX,XX @@ static bool trans_VFML_scalar(DisasContext *s, arg_VFML_scalar *a)
44
@@ -XXX,XX +XXX,XX @@ static bool trans_STLR_i(DisasContext *s, arg_ldapr_stlr_i *a)
32
gen_helper_gvec_fmlal_idx_a32);
33
return true;
45
return true;
34
}
46
}
35
+
47
36
+static struct {
48
-/* AdvSIMD load/store multiple structures
37
+ int nregs;
49
- *
38
+ int interleave;
50
- * 31 30 29 23 22 21 16 15 12 11 10 9 5 4 0
39
+ int spacing;
51
- * +---+---+---------------+---+-------------+--------+------+------+------+
40
+} const neon_ls_element_type[11] = {
52
- * | 0 | Q | 0 0 1 1 0 0 0 | L | 0 0 0 0 0 0 | opcode | size | Rn | Rt |
41
+ {1, 4, 1},
53
- * +---+---+---------------+---+-------------+--------+------+------+------+
42
+ {1, 4, 2},
54
- *
43
+ {4, 1, 1},
55
- * AdvSIMD load/store multiple structures (post-indexed)
44
+ {2, 2, 2},
56
- *
45
+ {1, 3, 1},
57
- * 31 30 29 23 22 21 20 16 15 12 11 10 9 5 4 0
46
+ {1, 3, 2},
58
- * +---+---+---------------+---+---+---------+--------+------+------+------+
47
+ {3, 1, 1},
59
- * | 0 | Q | 0 0 1 1 0 0 1 | L | 0 | Rm | opcode | size | Rn | Rt |
48
+ {1, 1, 1},
60
- * +---+---+---------------+---+---+---------+--------+------+------+------+
49
+ {1, 2, 1},
61
- *
50
+ {1, 2, 2},
62
- * Rt: first (or only) SIMD&FP register to be transferred
51
+ {2, 1, 1}
63
- * Rn: base address or SP
52
+};
64
- * Rm (post-index only): post-index register (when !31) or size dependent #imm
53
+
65
- */
54
+static void gen_neon_ldst_base_update(DisasContext *s, int rm, int rn,
66
-static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
55
+ int stride)
67
+static bool trans_LD_mult(DisasContext *s, arg_ldst_mult *a)
68
{
69
- int rt = extract32(insn, 0, 5);
70
- int rn = extract32(insn, 5, 5);
71
- int rm = extract32(insn, 16, 5);
72
- int size = extract32(insn, 10, 2);
73
- int opcode = extract32(insn, 12, 4);
74
- bool is_store = !extract32(insn, 22, 1);
75
- bool is_postidx = extract32(insn, 23, 1);
76
- bool is_q = extract32(insn, 30, 1);
77
TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
78
MemOp endian, align, mop;
79
80
int total; /* total bytes */
81
int elements; /* elements per vector */
82
- int rpt; /* num iterations */
83
- int selem; /* structure elements */
84
int r;
85
+ int size = a->sz;
86
87
- if (extract32(insn, 31, 1) || extract32(insn, 21, 1)) {
88
- unallocated_encoding(s);
89
- return;
90
+ if (!a->p && a->rm != 0) {
91
+ /* For non-postindexed accesses the Rm field must be 0 */
92
+ return false;
93
}
94
-
95
- if (!is_postidx && rm != 0) {
96
- unallocated_encoding(s);
97
- return;
98
+ if (size == 3 && !a->q && a->selem != 1) {
99
+ return false;
100
}
101
-
102
- /* From the shared decode logic */
103
- switch (opcode) {
104
- case 0x0:
105
- rpt = 1;
106
- selem = 4;
107
- break;
108
- case 0x2:
109
- rpt = 4;
110
- selem = 1;
111
- break;
112
- case 0x4:
113
- rpt = 1;
114
- selem = 3;
115
- break;
116
- case 0x6:
117
- rpt = 3;
118
- selem = 1;
119
- break;
120
- case 0x7:
121
- rpt = 1;
122
- selem = 1;
123
- break;
124
- case 0x8:
125
- rpt = 1;
126
- selem = 2;
127
- break;
128
- case 0xa:
129
- rpt = 2;
130
- selem = 1;
131
- break;
132
- default:
133
- unallocated_encoding(s);
134
- return;
135
- }
136
-
137
- if (size == 3 && !is_q && selem != 1) {
138
- /* reserved */
139
- unallocated_encoding(s);
140
- return;
141
- }
142
-
143
if (!fp_access_check(s)) {
144
- return;
145
+ return true;
146
}
147
148
- if (rn == 31) {
149
+ if (a->rn == 31) {
150
gen_check_sp_alignment(s);
151
}
152
153
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
154
endian = MO_LE;
155
}
156
157
- total = rpt * selem * (is_q ? 16 : 8);
158
- tcg_rn = cpu_reg_sp(s, rn);
159
+ total = a->rpt * a->selem * (a->q ? 16 : 8);
160
+ tcg_rn = cpu_reg_sp(s, a->rn);
161
162
/*
163
* Issue the MTE check vs the logical repeat count, before we
164
* promote consecutive little-endian elements below.
165
*/
166
- clean_addr = gen_mte_checkN(s, tcg_rn, is_store, is_postidx || rn != 31,
167
- total, finalize_memop_asimd(s, size));
168
+ clean_addr = gen_mte_checkN(s, tcg_rn, false, a->p || a->rn != 31, total,
169
+ finalize_memop_asimd(s, size));
170
171
/*
172
* Consecutive little-endian elements from a single register
173
* can be promoted to a larger little-endian operation.
174
*/
175
align = MO_ALIGN;
176
- if (selem == 1 && endian == MO_LE) {
177
+ if (a->selem == 1 && endian == MO_LE) {
178
align = pow2_align(size);
179
size = 3;
180
}
181
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
182
}
183
mop = endian | size | align;
184
185
- elements = (is_q ? 16 : 8) >> size;
186
+ elements = (a->q ? 16 : 8) >> size;
187
tcg_ebytes = tcg_constant_i64(1 << size);
188
- for (r = 0; r < rpt; r++) {
189
+ for (r = 0; r < a->rpt; r++) {
190
int e;
191
for (e = 0; e < elements; e++) {
192
int xs;
193
- for (xs = 0; xs < selem; xs++) {
194
- int tt = (rt + r + xs) % 32;
195
- if (is_store) {
196
- do_vec_st(s, tt, e, clean_addr, mop);
197
- } else {
198
- do_vec_ld(s, tt, e, clean_addr, mop);
199
- }
200
+ for (xs = 0; xs < a->selem; xs++) {
201
+ int tt = (a->rt + r + xs) % 32;
202
+ do_vec_ld(s, tt, e, clean_addr, mop);
203
tcg_gen_add_i64(clean_addr, clean_addr, tcg_ebytes);
204
}
205
}
206
}
207
208
- if (!is_store) {
209
- /* For non-quad operations, setting a slice of the low
210
- * 64 bits of the register clears the high 64 bits (in
211
- * the ARM ARM pseudocode this is implicit in the fact
212
- * that 'rval' is a 64 bit wide variable).
213
- * For quad operations, we might still need to zero the
214
- * high bits of SVE.
215
- */
216
- for (r = 0; r < rpt * selem; r++) {
217
- int tt = (rt + r) % 32;
218
- clear_vec_high(s, is_q, tt);
219
+ /*
220
+ * For non-quad operations, setting a slice of the low 64 bits of
221
+ * the register clears the high 64 bits (in the ARM ARM pseudocode
222
+ * this is implicit in the fact that 'rval' is a 64 bit wide
223
+ * variable). For quad operations, we might still need to zero
224
+ * the high bits of SVE.
225
+ */
226
+ for (r = 0; r < a->rpt * a->selem; r++) {
227
+ int tt = (a->rt + r) % 32;
228
+ clear_vec_high(s, a->q, tt);
229
+ }
230
+
231
+ if (a->p) {
232
+ if (a->rm == 31) {
233
+ tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
234
+ } else {
235
+ tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, a->rm));
236
+ }
237
+ }
238
+ return true;
239
+}
240
+
241
+static bool trans_ST_mult(DisasContext *s, arg_ldst_mult *a)
56
+{
242
+{
57
+ if (rm != 15) {
243
+ TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
58
+ TCGv_i32 base;
244
+ MemOp endian, align, mop;
59
+
245
+
60
+ base = load_reg(s, rn);
246
+ int total; /* total bytes */
61
+ if (rm == 13) {
247
+ int elements; /* elements per vector */
62
+ tcg_gen_addi_i32(base, base, stride);
248
+ int r;
63
+ } else {
249
+ int size = a->sz;
64
+ TCGv_i32 index;
250
+
65
+ index = load_reg(s, rm);
251
+ if (!a->p && a->rm != 0) {
66
+ tcg_gen_add_i32(base, base, index);
252
+ /* For non-postindexed accesses the Rm field must be 0 */
67
+ tcg_temp_free_i32(index);
68
+ }
69
+ store_reg(s, rn, base);
70
+ }
71
+}
72
+
73
+static bool trans_VLDST_multiple(DisasContext *s, arg_VLDST_multiple *a)
74
+{
75
+ /* Neon load/store multiple structures */
76
+ int nregs, interleave, spacing, reg, n;
77
+ MemOp endian = s->be_data;
78
+ int mmu_idx = get_mem_index(s);
79
+ int size = a->size;
80
+ TCGv_i64 tmp64;
81
+ TCGv_i32 addr, tmp;
82
+
83
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
84
+ return false;
253
+ return false;
85
+ }
254
+ }
86
+
255
+ if (size == 3 && !a->q && a->selem != 1) {
87
+ /* UNDEF accesses to D16-D31 if they don't exist */
88
+ if (!dc_isar_feature(aa32_simd_r32, s) && (a->vd & 0x10)) {
89
+ return false;
256
+ return false;
90
+ }
257
+ }
91
+ if (a->itype > 10) {
258
+ if (!fp_access_check(s)) {
92
+ return false;
93
+ }
94
+ /* Catch UNDEF cases for bad values of align field */
95
+ switch (a->itype & 0xc) {
96
+ case 4:
97
+ if (a->align >= 2) {
98
+ return false;
99
+ }
100
+ break;
101
+ case 8:
102
+ if (a->align == 3) {
103
+ return false;
104
+ }
105
+ break;
106
+ default:
107
+ break;
108
+ }
109
+ nregs = neon_ls_element_type[a->itype].nregs;
110
+ interleave = neon_ls_element_type[a->itype].interleave;
111
+ spacing = neon_ls_element_type[a->itype].spacing;
112
+ if (size == 3 && (interleave | spacing) != 1) {
113
+ return false;
114
+ }
115
+
116
+ if (!vfp_access_check(s)) {
117
+ return true;
259
+ return true;
118
+ }
260
+ }
119
+
261
+
262
+ if (a->rn == 31) {
263
+ gen_check_sp_alignment(s);
264
+ }
265
+
120
+ /* For our purposes, bytes are always little-endian. */
266
+ /* For our purposes, bytes are always little-endian. */
267
+ endian = s->be_data;
121
+ if (size == 0) {
268
+ if (size == 0) {
122
+ endian = MO_LE;
269
+ endian = MO_LE;
123
+ }
270
+ }
271
+
272
+ total = a->rpt * a->selem * (a->q ? 16 : 8);
273
+ tcg_rn = cpu_reg_sp(s, a->rn);
274
+
275
+ /*
276
+ * Issue the MTE check vs the logical repeat count, before we
277
+ * promote consecutive little-endian elements below.
278
+ */
279
+ clean_addr = gen_mte_checkN(s, tcg_rn, true, a->p || a->rn != 31, total,
280
+ finalize_memop_asimd(s, size));
281
+
124
+ /*
282
+ /*
125
+ * Consecutive little-endian elements from a single register
283
+ * Consecutive little-endian elements from a single register
126
+ * can be promoted to a larger little-endian operation.
284
+ * can be promoted to a larger little-endian operation.
127
+ */
285
+ */
128
+ if (interleave == 1 && endian == MO_LE) {
286
+ align = MO_ALIGN;
287
+ if (a->selem == 1 && endian == MO_LE) {
288
+ align = pow2_align(size);
129
+ size = 3;
289
+ size = 3;
130
+ }
290
+ }
131
+ tmp64 = tcg_temp_new_i64();
291
+ if (!s->align_mem) {
132
+ addr = tcg_temp_new_i32();
292
+ align = 0;
133
+ tmp = tcg_const_i32(1 << size);
293
+ }
134
+ load_reg_var(s, addr, a->rn);
294
+ mop = endian | size | align;
135
+ for (reg = 0; reg < nregs; reg++) {
295
+
136
+ for (n = 0; n < 8 >> size; n++) {
296
+ elements = (a->q ? 16 : 8) >> size;
297
+ tcg_ebytes = tcg_constant_i64(1 << size);
298
+ for (r = 0; r < a->rpt; r++) {
299
+ int e;
300
+ for (e = 0; e < elements; e++) {
137
+ int xs;
301
+ int xs;
138
+ for (xs = 0; xs < interleave; xs++) {
302
+ for (xs = 0; xs < a->selem; xs++) {
139
+ int tt = a->vd + reg + spacing * xs;
303
+ int tt = (a->rt + r + xs) % 32;
140
+
304
+ do_vec_st(s, tt, e, clean_addr, mop);
141
+ if (a->l) {
305
+ tcg_gen_add_i64(clean_addr, clean_addr, tcg_ebytes);
142
+ gen_aa32_ld_i64(s, tmp64, addr, mmu_idx, endian | size);
143
+ neon_store_element64(tt, n, size, tmp64);
144
+ } else {
145
+ neon_load_element64(tmp64, tt, n, size);
146
+ gen_aa32_st_i64(s, tmp64, addr, mmu_idx, endian | size);
147
+ }
148
+ tcg_gen_add_i32(addr, addr, tmp);
149
+ }
306
+ }
150
+ }
307
}
151
+ }
308
}
152
+ tcg_temp_free_i32(addr);
309
153
+ tcg_temp_free_i32(tmp);
310
- if (is_postidx) {
154
+ tcg_temp_free_i64(tmp64);
311
- if (rm == 31) {
155
+
312
+ if (a->p) {
156
+ gen_neon_ldst_base_update(s, a->rm, a->rn, nregs * interleave * 8);
313
+ if (a->rm == 31) {
314
tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
315
} else {
316
- tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, rm));
317
+ tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, a->rm));
318
}
319
}
157
+ return true;
320
+ return true;
158
+}
159
diff --git a/target/arm/translate.c b/target/arm/translate.c
160
index XXXXXXX..XXXXXXX 100644
161
--- a/target/arm/translate.c
162
+++ b/target/arm/translate.c
163
@@ -XXX,XX +XXX,XX @@ static void gen_neon_trn_u16(TCGv_i32 t0, TCGv_i32 t1)
164
}
321
}
165
322
166
323
/* AdvSIMD load/store single structure
167
-static struct {
324
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
168
- int nregs;
325
static void disas_ldst(DisasContext *s, uint32_t insn)
169
- int interleave;
170
- int spacing;
171
-} const neon_ls_element_type[11] = {
172
- {1, 4, 1},
173
- {1, 4, 2},
174
- {4, 1, 1},
175
- {2, 2, 2},
176
- {1, 3, 1},
177
- {1, 3, 2},
178
- {3, 1, 1},
179
- {1, 1, 1},
180
- {1, 2, 1},
181
- {1, 2, 2},
182
- {2, 1, 1}
183
-};
184
-
185
/* Translate a NEON load/store element instruction. Return nonzero if the
186
instruction is invalid. */
187
static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
188
{
326
{
189
int rd, rn, rm;
327
switch (extract32(insn, 24, 6)) {
190
- int op;
328
- case 0x0c: /* AdvSIMD load/store multiple structures */
191
int nregs;
329
- disas_ldst_multiple_struct(s, insn);
192
- int interleave;
330
- break;
193
- int spacing;
331
case 0x0d: /* AdvSIMD load/store single structure */
194
int stride;
332
disas_ldst_single_struct(s, insn);
195
int size;
333
break;
196
int reg;
197
int load;
198
- int n;
199
int vec_size;
200
- int mmu_idx;
201
- MemOp endian;
202
TCGv_i32 addr;
203
TCGv_i32 tmp;
204
- TCGv_i32 tmp2;
205
- TCGv_i64 tmp64;
206
207
if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
208
return 1;
209
@@ -XXX,XX +XXX,XX @@ static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
210
rn = (insn >> 16) & 0xf;
211
rm = insn & 0xf;
212
load = (insn & (1 << 21)) != 0;
213
- endian = s->be_data;
214
- mmu_idx = get_mem_index(s);
215
if ((insn & (1 << 23)) == 0) {
216
- /* Load store all elements. */
217
- op = (insn >> 8) & 0xf;
218
- size = (insn >> 6) & 3;
219
- if (op > 10)
220
- return 1;
221
- /* Catch UNDEF cases for bad values of align field */
222
- switch (op & 0xc) {
223
- case 4:
224
- if (((insn >> 5) & 1) == 1) {
225
- return 1;
226
- }
227
- break;
228
- case 8:
229
- if (((insn >> 4) & 3) == 3) {
230
- return 1;
231
- }
232
- break;
233
- default:
234
- break;
235
- }
236
- nregs = neon_ls_element_type[op].nregs;
237
- interleave = neon_ls_element_type[op].interleave;
238
- spacing = neon_ls_element_type[op].spacing;
239
- if (size == 3 && (interleave | spacing) != 1) {
240
- return 1;
241
- }
242
- /* For our purposes, bytes are always little-endian. */
243
- if (size == 0) {
244
- endian = MO_LE;
245
- }
246
- /* Consecutive little-endian elements from a single register
247
- * can be promoted to a larger little-endian operation.
248
- */
249
- if (interleave == 1 && endian == MO_LE) {
250
- size = 3;
251
- }
252
- tmp64 = tcg_temp_new_i64();
253
- addr = tcg_temp_new_i32();
254
- tmp2 = tcg_const_i32(1 << size);
255
- load_reg_var(s, addr, rn);
256
- for (reg = 0; reg < nregs; reg++) {
257
- for (n = 0; n < 8 >> size; n++) {
258
- int xs;
259
- for (xs = 0; xs < interleave; xs++) {
260
- int tt = rd + reg + spacing * xs;
261
-
262
- if (load) {
263
- gen_aa32_ld_i64(s, tmp64, addr, mmu_idx, endian | size);
264
- neon_store_element64(tt, n, size, tmp64);
265
- } else {
266
- neon_load_element64(tmp64, tt, n, size);
267
- gen_aa32_st_i64(s, tmp64, addr, mmu_idx, endian | size);
268
- }
269
- tcg_gen_add_i32(addr, addr, tmp2);
270
- }
271
- }
272
- }
273
- tcg_temp_free_i32(addr);
274
- tcg_temp_free_i32(tmp2);
275
- tcg_temp_free_i64(tmp64);
276
- stride = nregs * interleave * 8;
277
+ /* Load store all elements -- handled already by decodetree */
278
+ return 1;
279
} else {
280
size = (insn >> 10) & 3;
281
if (size == 3) {
282
--
334
--
283
2.20.1
335
2.34.1
284
285
diff view generated by jsdifflib
1
Convert the Neon VMUL, VMLA, VMLS and VSHL insns in the
1
Convert the ASIMD load/store single structure insns to decodetree.
2
3-reg-same grouping to decodetree.
3
2
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Message-id: 20230602155223.2040685-20-peter.maydell@linaro.org
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20200430181003.21682-20-peter.maydell@linaro.org
7
---
6
---
8
target/arm/neon-dp.decode | 9 +++++++
7
target/arm/tcg/a64.decode | 34 +++++
9
target/arm/translate-neon.inc.c | 44 +++++++++++++++++++++++++++++++++
8
target/arm/tcg/translate-a64.c | 219 +++++++++++++++------------------
10
target/arm/translate.c | 28 +++------------------
9
2 files changed, 136 insertions(+), 117 deletions(-)
11
3 files changed, 56 insertions(+), 25 deletions(-)
12
10
13
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
11
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
14
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/neon-dp.decode
13
--- a/target/arm/tcg/a64.decode
16
+++ b/target/arm/neon-dp.decode
14
+++ b/target/arm/tcg/a64.decode
17
@@ -XXX,XX +XXX,XX @@ VCGT_U_3s 1111 001 1 0 . .. .... .... 0011 . . . 0 .... @3same
15
@@ -XXX,XX +XXX,XX @@ LD_mult 0 . 001100 . 1 0 ..... 0110 .. ..... ..... @ldst_mult rpt=3 sele
18
VCGE_S_3s 1111 001 0 0 . .. .... .... 0011 . . . 1 .... @3same
16
LD_mult 0 . 001100 . 1 0 ..... 0111 .. ..... ..... @ldst_mult rpt=1 selem=1
19
VCGE_U_3s 1111 001 1 0 . .. .... .... 0011 . . . 1 .... @3same
17
LD_mult 0 . 001100 . 1 0 ..... 1000 .. ..... ..... @ldst_mult rpt=1 selem=2
20
18
LD_mult 0 . 001100 . 1 0 ..... 1010 .. ..... ..... @ldst_mult rpt=2 selem=1
21
+VSHL_S_3s 1111 001 0 0 . .. .... .... 0100 . . . 0 .... @3same
19
+
22
+VSHL_U_3s 1111 001 1 0 . .. .... .... 0100 . . . 0 .... @3same
20
+# Load/store single structure
23
+
21
+&ldst_single rm rn rt p selem index scale
24
VMAX_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 0 .... @3same
22
+
25
VMAX_U_3s 1111 001 1 0 . .. .... .... 0110 . . . 0 .... @3same
23
+%ldst_single_selem 13:1 21:1 !function=plus_1
26
VMIN_S_3s 1111 001 0 0 . .. .... .... 0110 . . . 1 .... @3same
24
+
27
@@ -XXX,XX +XXX,XX @@ VSUB_3s 1111 001 1 0 . .. .... .... 1000 . . . 0 .... @3same
25
+%ldst_single_index_b 30:1 10:3
28
26
+%ldst_single_index_h 30:1 11:2
29
VTST_3s 1111 001 0 0 . .. .... .... 1000 . . . 1 .... @3same
27
+%ldst_single_index_s 30:1 12:1
30
VCEQ_3s 1111 001 1 0 . .. .... .... 1000 . . . 1 .... @3same
28
+
31
+
29
+@ldst_single_b .. ...... p:1 .. rm:5 ...... rn:5 rt:5 \
32
+VMLA_3s 1111 001 0 0 . .. .... .... 1001 . . . 0 .... @3same
30
+ &ldst_single scale=0 selem=%ldst_single_selem \
33
+VMLS_3s 1111 001 1 0 . .. .... .... 1001 . . . 0 .... @3same
31
+ index=%ldst_single_index_b
34
+
32
+@ldst_single_h .. ...... p:1 .. rm:5 ...... rn:5 rt:5 \
35
+VMUL_3s 1111 001 0 0 . .. .... .... 1001 . . . 1 .... @3same
33
+ &ldst_single scale=1 selem=%ldst_single_selem \
36
+VMUL_p_3s 1111 001 1 0 . .. .... .... 1001 . . . 1 .... @3same
34
+ index=%ldst_single_index_h
37
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
35
+@ldst_single_s .. ...... p:1 .. rm:5 ...... rn:5 rt:5 \
36
+ &ldst_single scale=2 selem=%ldst_single_selem \
37
+ index=%ldst_single_index_s
38
+@ldst_single_d . index:1 ...... p:1 .. rm:5 ...... rn:5 rt:5 \
39
+ &ldst_single scale=3 selem=%ldst_single_selem
40
+
41
+ST_single 0 . 001101 . 0 . ..... 00 . ... ..... ..... @ldst_single_b
42
+ST_single 0 . 001101 . 0 . ..... 01 . ..0 ..... ..... @ldst_single_h
43
+ST_single 0 . 001101 . 0 . ..... 10 . .00 ..... ..... @ldst_single_s
44
+ST_single 0 . 001101 . 0 . ..... 10 . 001 ..... ..... @ldst_single_d
45
+
46
+LD_single 0 . 001101 . 1 . ..... 00 . ... ..... ..... @ldst_single_b
47
+LD_single 0 . 001101 . 1 . ..... 01 . ..0 ..... ..... @ldst_single_h
48
+LD_single 0 . 001101 . 1 . ..... 10 . .00 ..... ..... @ldst_single_s
49
+LD_single 0 . 001101 . 1 . ..... 10 . 001 ..... ..... @ldst_single_d
50
+
51
+# Replicating load case
52
+LD_single_repl 0 q:1 001101 p:1 1 . rm:5 11 . 0 scale:2 rn:5 rt:5 selem=%ldst_single_selem
53
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
38
index XXXXXXX..XXXXXXX 100644
54
index XXXXXXX..XXXXXXX 100644
39
--- a/target/arm/translate-neon.inc.c
55
--- a/target/arm/tcg/translate-a64.c
40
+++ b/target/arm/translate-neon.inc.c
56
+++ b/target/arm/tcg/translate-a64.c
41
@@ -XXX,XX +XXX,XX @@ DO_3SAME_NO_SZ_3(VMAX_S, tcg_gen_gvec_smax)
57
@@ -XXX,XX +XXX,XX @@ static bool trans_ST_mult(DisasContext *s, arg_ldst_mult *a)
42
DO_3SAME_NO_SZ_3(VMAX_U, tcg_gen_gvec_umax)
58
return true;
43
DO_3SAME_NO_SZ_3(VMIN_S, tcg_gen_gvec_smin)
59
}
44
DO_3SAME_NO_SZ_3(VMIN_U, tcg_gen_gvec_umin)
60
45
+DO_3SAME_NO_SZ_3(VMUL, tcg_gen_gvec_mul)
61
-/* AdvSIMD load/store single structure
46
62
- *
47
#define DO_3SAME_CMP(INSN, COND) \
63
- * 31 30 29 23 22 21 20 16 15 13 12 11 10 9 5 4 0
48
static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
64
- * +---+---+---------------+-----+-----------+-----+---+------+------+------+
49
@@ -XXX,XX +XXX,XX @@ DO_3SAME_GVEC4(VQADD_S, sqadd_op)
65
- * | 0 | Q | 0 0 1 1 0 1 0 | L R | 0 0 0 0 0 | opc | S | size | Rn | Rt |
50
DO_3SAME_GVEC4(VQADD_U, uqadd_op)
66
- * +---+---+---------------+-----+-----------+-----+---+------+------+------+
51
DO_3SAME_GVEC4(VQSUB_S, sqsub_op)
67
- *
52
DO_3SAME_GVEC4(VQSUB_U, uqsub_op)
68
- * AdvSIMD load/store single structure (post-indexed)
53
+
69
- *
54
+static void gen_VMUL_p_3s(unsigned vece, uint32_t rd_ofs, uint32_t rn_ofs,
70
- * 31 30 29 23 22 21 20 16 15 13 12 11 10 9 5 4 0
55
+ uint32_t rm_ofs, uint32_t oprsz, uint32_t maxsz)
71
- * +---+---+---------------+-----+-----------+-----+---+------+------+------+
72
- * | 0 | Q | 0 0 1 1 0 1 1 | L R | Rm | opc | S | size | Rn | Rt |
73
- * +---+---+---------------+-----+-----------+-----+---+------+------+------+
74
- *
75
- * Rt: first (or only) SIMD&FP register to be transferred
76
- * Rn: base address or SP
77
- * Rm (post-index only): post-index register (when !31) or size dependent #imm
78
- * index = encoded in Q:S:size dependent on size
79
- *
80
- * lane_size = encoded in R, opc
81
- * transfer width = encoded in opc, S, size
82
- */
83
-static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
84
+static bool trans_ST_single(DisasContext *s, arg_ldst_single *a)
85
{
86
- int rt = extract32(insn, 0, 5);
87
- int rn = extract32(insn, 5, 5);
88
- int rm = extract32(insn, 16, 5);
89
- int size = extract32(insn, 10, 2);
90
- int S = extract32(insn, 12, 1);
91
- int opc = extract32(insn, 13, 3);
92
- int R = extract32(insn, 21, 1);
93
- int is_load = extract32(insn, 22, 1);
94
- int is_postidx = extract32(insn, 23, 1);
95
- int is_q = extract32(insn, 30, 1);
96
-
97
- int scale = extract32(opc, 1, 2);
98
- int selem = (extract32(opc, 0, 1) << 1 | R) + 1;
99
- bool replicate = false;
100
- int index = is_q << 3 | S << 2 | size;
101
- int xs, total;
102
+ int xs, total, rt;
103
TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
104
MemOp mop;
105
106
- if (extract32(insn, 31, 1)) {
107
- unallocated_encoding(s);
108
- return;
109
+ if (!a->p && a->rm != 0) {
110
+ return false;
111
}
112
- if (!is_postidx && rm != 0) {
113
- unallocated_encoding(s);
114
- return;
115
- }
116
-
117
- switch (scale) {
118
- case 3:
119
- if (!is_load || S) {
120
- unallocated_encoding(s);
121
- return;
122
- }
123
- scale = size;
124
- replicate = true;
125
- break;
126
- case 0:
127
- break;
128
- case 1:
129
- if (extract32(size, 0, 1)) {
130
- unallocated_encoding(s);
131
- return;
132
- }
133
- index >>= 1;
134
- break;
135
- case 2:
136
- if (extract32(size, 1, 1)) {
137
- unallocated_encoding(s);
138
- return;
139
- }
140
- if (!extract32(size, 0, 1)) {
141
- index >>= 2;
142
- } else {
143
- if (S) {
144
- unallocated_encoding(s);
145
- return;
146
- }
147
- index >>= 3;
148
- scale = 3;
149
- }
150
- break;
151
- default:
152
- g_assert_not_reached();
153
- }
154
-
155
if (!fp_access_check(s)) {
156
- return;
157
+ return true;
158
}
159
160
- if (rn == 31) {
161
+ if (a->rn == 31) {
162
gen_check_sp_alignment(s);
163
}
164
165
- total = selem << scale;
166
- tcg_rn = cpu_reg_sp(s, rn);
167
+ total = a->selem << a->scale;
168
+ tcg_rn = cpu_reg_sp(s, a->rn);
169
170
- mop = finalize_memop_asimd(s, scale);
171
-
172
- clean_addr = gen_mte_checkN(s, tcg_rn, !is_load, is_postidx || rn != 31,
173
+ mop = finalize_memop_asimd(s, a->scale);
174
+ clean_addr = gen_mte_checkN(s, tcg_rn, true, a->p || a->rn != 31,
175
total, mop);
176
177
- tcg_ebytes = tcg_constant_i64(1 << scale);
178
- for (xs = 0; xs < selem; xs++) {
179
- if (replicate) {
180
- /* Load and replicate to all elements */
181
- TCGv_i64 tcg_tmp = tcg_temp_new_i64();
182
-
183
- tcg_gen_qemu_ld_i64(tcg_tmp, clean_addr, get_mem_index(s), mop);
184
- tcg_gen_gvec_dup_i64(scale, vec_full_reg_offset(s, rt),
185
- (is_q + 1) * 8, vec_full_reg_size(s),
186
- tcg_tmp);
187
- } else {
188
- /* Load/store one element per register */
189
- if (is_load) {
190
- do_vec_ld(s, rt, index, clean_addr, mop);
191
- } else {
192
- do_vec_st(s, rt, index, clean_addr, mop);
193
- }
194
- }
195
+ tcg_ebytes = tcg_constant_i64(1 << a->scale);
196
+ for (xs = 0, rt = a->rt; xs < a->selem; xs++, rt = (rt + 1) % 32) {
197
+ do_vec_st(s, rt, a->index, clean_addr, mop);
198
tcg_gen_add_i64(clean_addr, clean_addr, tcg_ebytes);
199
- rt = (rt + 1) % 32;
200
}
201
202
- if (is_postidx) {
203
- if (rm == 31) {
204
+ if (a->p) {
205
+ if (a->rm == 31) {
206
tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
207
} else {
208
- tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, rm));
209
+ tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, a->rm));
210
}
211
}
212
+ return true;
213
+}
214
+
215
+static bool trans_LD_single(DisasContext *s, arg_ldst_single *a)
56
+{
216
+{
57
+ tcg_gen_gvec_3_ool(rd_ofs, rn_ofs, rm_ofs, oprsz, maxsz,
217
+ int xs, total, rt;
58
+ 0, gen_helper_gvec_pmul_b);
218
+ TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
219
+ MemOp mop;
220
+
221
+ if (!a->p && a->rm != 0) {
222
+ return false;
223
+ }
224
+ if (!fp_access_check(s)) {
225
+ return true;
226
+ }
227
+
228
+ if (a->rn == 31) {
229
+ gen_check_sp_alignment(s);
230
+ }
231
+
232
+ total = a->selem << a->scale;
233
+ tcg_rn = cpu_reg_sp(s, a->rn);
234
+
235
+ mop = finalize_memop_asimd(s, a->scale);
236
+ clean_addr = gen_mte_checkN(s, tcg_rn, false, a->p || a->rn != 31,
237
+ total, mop);
238
+
239
+ tcg_ebytes = tcg_constant_i64(1 << a->scale);
240
+ for (xs = 0, rt = a->rt; xs < a->selem; xs++, rt = (rt + 1) % 32) {
241
+ do_vec_ld(s, rt, a->index, clean_addr, mop);
242
+ tcg_gen_add_i64(clean_addr, clean_addr, tcg_ebytes);
243
+ }
244
+
245
+ if (a->p) {
246
+ if (a->rm == 31) {
247
+ tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
248
+ } else {
249
+ tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, a->rm));
250
+ }
251
+ }
252
+ return true;
59
+}
253
+}
60
+
254
+
61
+static bool trans_VMUL_p_3s(DisasContext *s, arg_3same *a)
255
+static bool trans_LD_single_repl(DisasContext *s, arg_LD_single_repl *a)
62
+{
256
+{
63
+ if (a->size != 0) {
257
+ int xs, total, rt;
258
+ TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
259
+ MemOp mop;
260
+
261
+ if (!a->p && a->rm != 0) {
64
+ return false;
262
+ return false;
65
+ }
263
+ }
66
+ return do_3same(s, a, gen_VMUL_p_3s);
264
+ if (!fp_access_check(s)) {
67
+}
265
+ return true;
68
+
266
+ }
69
+#define DO_3SAME_GVEC3_NO_SZ_3(INSN, OPARRAY) \
267
+
70
+ static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
268
+ if (a->rn == 31) {
71
+ uint32_t rn_ofs, uint32_t rm_ofs, \
269
+ gen_check_sp_alignment(s);
72
+ uint32_t oprsz, uint32_t maxsz) \
270
+ }
73
+ { \
271
+
74
+ tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, \
272
+ total = a->selem << a->scale;
75
+ oprsz, maxsz, &OPARRAY[vece]); \
273
+ tcg_rn = cpu_reg_sp(s, a->rn);
76
+ } \
274
+
77
+ DO_3SAME_NO_SZ_3(INSN, gen_##INSN##_3s)
275
+ mop = finalize_memop_asimd(s, a->scale);
78
+
276
+ clean_addr = gen_mte_checkN(s, tcg_rn, false, a->p || a->rn != 31,
79
+
277
+ total, mop);
80
+DO_3SAME_GVEC3_NO_SZ_3(VMLA, mla_op)
278
+
81
+DO_3SAME_GVEC3_NO_SZ_3(VMLS, mls_op)
279
+ tcg_ebytes = tcg_constant_i64(1 << a->scale);
82
+
280
+ for (xs = 0, rt = a->rt; xs < a->selem; xs++, rt = (rt + 1) % 32) {
83
+#define DO_3SAME_GVEC3_SHIFT(INSN, OPARRAY) \
281
+ /* Load and replicate to all elements */
84
+ static void gen_##INSN##_3s(unsigned vece, uint32_t rd_ofs, \
282
+ TCGv_i64 tcg_tmp = tcg_temp_new_i64();
85
+ uint32_t rn_ofs, uint32_t rm_ofs, \
283
+
86
+ uint32_t oprsz, uint32_t maxsz) \
284
+ tcg_gen_qemu_ld_i64(tcg_tmp, clean_addr, get_mem_index(s), mop);
87
+ { \
285
+ tcg_gen_gvec_dup_i64(a->scale, vec_full_reg_offset(s, rt),
88
+ /* Note the operation is vshl vd,vm,vn */ \
286
+ (a->q + 1) * 8, vec_full_reg_size(s), tcg_tmp);
89
+ tcg_gen_gvec_3(rd_ofs, rm_ofs, rn_ofs, \
287
+ tcg_gen_add_i64(clean_addr, clean_addr, tcg_ebytes);
90
+ oprsz, maxsz, &OPARRAY[vece]); \
288
+ }
91
+ } \
289
+
92
+ DO_3SAME(INSN, gen_##INSN##_3s)
290
+ if (a->p) {
93
+
291
+ if (a->rm == 31) {
94
+DO_3SAME_GVEC3_SHIFT(VSHL_S, sshl_op)
292
+ tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
95
+DO_3SAME_GVEC3_SHIFT(VSHL_U, ushl_op)
293
+ } else {
96
diff --git a/target/arm/translate.c b/target/arm/translate.c
294
+ tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, a->rm));
97
index XXXXXXX..XXXXXXX 100644
295
+ }
98
--- a/target/arm/translate.c
296
+ }
99
+++ b/target/arm/translate.c
297
+ return true;
100
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
298
}
101
}
299
102
return 1;
300
/*
103
301
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
104
- case NEON_3R_VMUL: /* VMUL */
302
static void disas_ldst(DisasContext *s, uint32_t insn)
105
- if (u) {
303
{
106
- /* Polynomial case allows only P8. */
304
switch (extract32(insn, 24, 6)) {
107
- if (size != 0) {
305
- case 0x0d: /* AdvSIMD load/store single structure */
108
- return 1;
306
- disas_ldst_single_struct(s, insn);
109
- }
307
- break;
110
- tcg_gen_gvec_3_ool(rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size,
308
case 0x19:
111
- 0, gen_helper_gvec_pmul_b);
309
if (extract32(insn, 21, 1) != 0) {
112
- } else {
310
disas_ldst_tag(s, insn);
113
- tcg_gen_gvec_mul(size, rd_ofs, rn_ofs, rm_ofs,
114
- vec_size, vec_size);
115
- }
116
- return 0;
117
-
118
- case NEON_3R_VML: /* VMLA, VMLS */
119
- tcg_gen_gvec_3(rd_ofs, rn_ofs, rm_ofs, vec_size, vec_size,
120
- u ? &mls_op[size] : &mla_op[size]);
121
- return 0;
122
-
123
- case NEON_3R_VSHL:
124
- /* Note the operation is vshl vd,vm,vn */
125
- tcg_gen_gvec_3(rd_ofs, rm_ofs, rn_ofs, vec_size, vec_size,
126
- u ? &ushl_op[size] : &sshl_op[size]);
127
- return 0;
128
-
129
case NEON_3R_VADD_VSUB:
130
case NEON_3R_LOGIC:
131
case NEON_3R_VMAX:
132
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
133
case NEON_3R_VCGE:
134
case NEON_3R_VQADD:
135
case NEON_3R_VQSUB:
136
+ case NEON_3R_VMUL:
137
+ case NEON_3R_VML:
138
+ case NEON_3R_VSHL:
139
/* Already handled by decodetree */
140
return 1;
141
}
142
--
311
--
143
2.20.1
312
2.34.1
144
145
diff view generated by jsdifflib
1
The ARMv8.2-TTS2UXN feature extends the XN field in stage 2
1
Convert the instructions in the load/store memory tags instruction
2
translation table descriptors from just bit [54] to bits [54:53],
2
group to decodetree.
3
allowing stage 2 to control execution permissions separately for EL0
4
and EL1. Implement the new semantics of the XN field and enable
5
the feature for our 'max' CPU.
6
3
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Message-id: 20200330210400.11724-5-peter.maydell@linaro.org
6
Message-id: 20230602155223.2040685-21-peter.maydell@linaro.org
11
---
7
---
12
target/arm/cpu.h | 15 +++++++++++++++
8
target/arm/tcg/a64.decode | 25 +++
13
target/arm/cpu.c | 1 +
9
target/arm/tcg/translate-a64.c | 360 ++++++++++++++++-----------------
14
target/arm/cpu64.c | 2 ++
10
2 files changed, 199 insertions(+), 186 deletions(-)
15
target/arm/helper.c | 37 +++++++++++++++++++++++++++++++------
16
4 files changed, 49 insertions(+), 6 deletions(-)
17
11
18
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
12
diff --git a/target/arm/tcg/a64.decode b/target/arm/tcg/a64.decode
19
index XXXXXXX..XXXXXXX 100644
13
index XXXXXXX..XXXXXXX 100644
20
--- a/target/arm/cpu.h
14
--- a/target/arm/tcg/a64.decode
21
+++ b/target/arm/cpu.h
15
+++ b/target/arm/tcg/a64.decode
22
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa32_ccidx(const ARMISARegisters *id)
16
@@ -XXX,XX +XXX,XX @@ LD_single 0 . 001101 . 1 . ..... 10 . 001 ..... ..... @ldst_single_d
23
return FIELD_EX32(id->id_mmfr4, ID_MMFR4, CCIDX) != 0;
17
18
# Replicating load case
19
LD_single_repl 0 q:1 001101 p:1 1 . rm:5 11 . 0 scale:2 rn:5 rt:5 selem=%ldst_single_selem
20
+
21
+%tag_offset 12:s9 !function=scale_by_log2_tag_granule
22
+&ldst_tag rn rt imm p w
23
+@ldst_tag ........ .. . ......... .. rn:5 rt:5 &ldst_tag imm=%tag_offset
24
+@ldst_tag_mult ........ .. . 000000000 .. rn:5 rt:5 &ldst_tag imm=0
25
+
26
+STZGM 11011001 00 1 ......... 00 ..... ..... @ldst_tag_mult p=0 w=0
27
+STG 11011001 00 1 ......... 01 ..... ..... @ldst_tag p=1 w=1
28
+STG 11011001 00 1 ......... 10 ..... ..... @ldst_tag p=0 w=0
29
+STG 11011001 00 1 ......... 11 ..... ..... @ldst_tag p=0 w=1
30
+
31
+LDG 11011001 01 1 ......... 00 ..... ..... @ldst_tag p=0 w=0
32
+STZG 11011001 01 1 ......... 01 ..... ..... @ldst_tag p=1 w=1
33
+STZG 11011001 01 1 ......... 10 ..... ..... @ldst_tag p=0 w=0
34
+STZG 11011001 01 1 ......... 11 ..... ..... @ldst_tag p=0 w=1
35
+
36
+STGM 11011001 10 1 ......... 00 ..... ..... @ldst_tag_mult p=0 w=0
37
+ST2G 11011001 10 1 ......... 01 ..... ..... @ldst_tag p=1 w=1
38
+ST2G 11011001 10 1 ......... 10 ..... ..... @ldst_tag p=0 w=0
39
+ST2G 11011001 10 1 ......... 11 ..... ..... @ldst_tag p=0 w=1
40
+
41
+LDGM 11011001 11 1 ......... 00 ..... ..... @ldst_tag_mult p=0 w=0
42
+STZ2G 11011001 11 1 ......... 01 ..... ..... @ldst_tag p=1 w=1
43
+STZ2G 11011001 11 1 ......... 10 ..... ..... @ldst_tag p=0 w=0
44
+STZ2G 11011001 11 1 ......... 11 ..... ..... @ldst_tag p=0 w=1
45
diff --git a/target/arm/tcg/translate-a64.c b/target/arm/tcg/translate-a64.c
46
index XXXXXXX..XXXXXXX 100644
47
--- a/target/arm/tcg/translate-a64.c
48
+++ b/target/arm/tcg/translate-a64.c
49
@@ -XXX,XX +XXX,XX @@ static int uimm_scaled(DisasContext *s, int x)
50
return imm << scale;
24
}
51
}
25
52
26
+static inline bool isar_feature_aa32_tts2uxn(const ARMISARegisters *id)
53
+/* For load/store memory tags: scale offset by LOG2_TAG_GRANULE */
54
+static int scale_by_log2_tag_granule(DisasContext *s, int x)
27
+{
55
+{
28
+ return FIELD_EX32(id->id_mmfr4, ID_MMFR4, XNX) != 0;
56
+ return x << LOG2_TAG_GRANULE;
29
+}
57
+}
30
+
58
+
31
/*
59
/*
32
* 64-bit feature tests via id registers.
60
* Include the generated decoders.
33
*/
61
*/
34
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa64_ccidx(const ARMISARegisters *id)
62
@@ -XXX,XX +XXX,XX @@ static bool trans_LD_single_repl(DisasContext *s, arg_LD_single_repl *a)
35
return FIELD_EX64(id->id_aa64mmfr2, ID_AA64MMFR2, CCIDX) != 0;
63
return true;
36
}
64
}
37
65
38
+static inline bool isar_feature_aa64_tts2uxn(const ARMISARegisters *id)
66
-/*
67
- * Load/Store memory tags
68
- *
69
- * 31 30 29 24 22 21 12 10 5 0
70
- * +-----+-------------+-----+---+------+-----+------+------+
71
- * | 1 1 | 0 1 1 0 0 1 | op1 | 1 | imm9 | op2 | Rn | Rt |
72
- * +-----+-------------+-----+---+------+-----+------+------+
73
- */
74
-static void disas_ldst_tag(DisasContext *s, uint32_t insn)
75
+static bool trans_STZGM(DisasContext *s, arg_ldst_tag *a)
76
{
77
- int rt = extract32(insn, 0, 5);
78
- int rn = extract32(insn, 5, 5);
79
- uint64_t offset = sextract64(insn, 12, 9) << LOG2_TAG_GRANULE;
80
- int op2 = extract32(insn, 10, 2);
81
- int op1 = extract32(insn, 22, 2);
82
- bool is_load = false, is_pair = false, is_zero = false, is_mult = false;
83
- int index = 0;
84
TCGv_i64 addr, clean_addr, tcg_rt;
85
+ int size = 4 << s->dcz_blocksize;
86
87
- /* We checked insn bits [29:24,21] in the caller. */
88
- if (extract32(insn, 30, 2) != 3) {
89
- goto do_unallocated;
90
+ if (!dc_isar_feature(aa64_mte, s)) {
91
+ return false;
92
+ }
93
+ if (s->current_el == 0) {
94
+ return false;
95
}
96
97
- /*
98
- * @index is a tri-state variable which has 3 states:
99
- * < 0 : post-index, writeback
100
- * = 0 : signed offset
101
- * > 0 : pre-index, writeback
102
- */
103
- switch (op1) {
104
- case 0:
105
- if (op2 != 0) {
106
- /* STG */
107
- index = op2 - 2;
108
- } else {
109
- /* STZGM */
110
- if (s->current_el == 0 || offset != 0) {
111
- goto do_unallocated;
112
- }
113
- is_mult = is_zero = true;
114
- }
115
- break;
116
- case 1:
117
- if (op2 != 0) {
118
- /* STZG */
119
- is_zero = true;
120
- index = op2 - 2;
121
- } else {
122
- /* LDG */
123
- is_load = true;
124
- }
125
- break;
126
- case 2:
127
- if (op2 != 0) {
128
- /* ST2G */
129
- is_pair = true;
130
- index = op2 - 2;
131
- } else {
132
- /* STGM */
133
- if (s->current_el == 0 || offset != 0) {
134
- goto do_unallocated;
135
- }
136
- is_mult = true;
137
- }
138
- break;
139
- case 3:
140
- if (op2 != 0) {
141
- /* STZ2G */
142
- is_pair = is_zero = true;
143
- index = op2 - 2;
144
- } else {
145
- /* LDGM */
146
- if (s->current_el == 0 || offset != 0) {
147
- goto do_unallocated;
148
- }
149
- is_mult = is_load = true;
150
- }
151
- break;
152
-
153
- default:
154
- do_unallocated:
155
- unallocated_encoding(s);
156
- return;
157
- }
158
-
159
- if (is_mult
160
- ? !dc_isar_feature(aa64_mte, s)
161
- : !dc_isar_feature(aa64_mte_insn_reg, s)) {
162
- goto do_unallocated;
163
- }
164
-
165
- if (rn == 31) {
166
+ if (a->rn == 31) {
167
gen_check_sp_alignment(s);
168
}
169
170
- addr = read_cpu_reg_sp(s, rn, true);
171
- if (index >= 0) {
172
+ addr = read_cpu_reg_sp(s, a->rn, true);
173
+ tcg_gen_addi_i64(addr, addr, a->imm);
174
+ tcg_rt = cpu_reg(s, a->rt);
175
+
176
+ if (s->ata) {
177
+ gen_helper_stzgm_tags(cpu_env, addr, tcg_rt);
178
+ }
179
+ /*
180
+ * The non-tags portion of STZGM is mostly like DC_ZVA,
181
+ * except the alignment happens before the access.
182
+ */
183
+ clean_addr = clean_data_tbi(s, addr);
184
+ tcg_gen_andi_i64(clean_addr, clean_addr, -size);
185
+ gen_helper_dc_zva(cpu_env, clean_addr);
186
+ return true;
187
+}
188
+
189
+static bool trans_STGM(DisasContext *s, arg_ldst_tag *a)
39
+{
190
+{
40
+ return FIELD_EX64(id->id_aa64mmfr1, ID_AA64MMFR1, XNX) != 0;
191
+ TCGv_i64 addr, clean_addr, tcg_rt;
192
+
193
+ if (!dc_isar_feature(aa64_mte, s)) {
194
+ return false;
195
+ }
196
+ if (s->current_el == 0) {
197
+ return false;
198
+ }
199
+
200
+ if (a->rn == 31) {
201
+ gen_check_sp_alignment(s);
202
+ }
203
+
204
+ addr = read_cpu_reg_sp(s, a->rn, true);
205
+ tcg_gen_addi_i64(addr, addr, a->imm);
206
+ tcg_rt = cpu_reg(s, a->rt);
207
+
208
+ if (s->ata) {
209
+ gen_helper_stgm(cpu_env, addr, tcg_rt);
210
+ } else {
211
+ MMUAccessType acc = MMU_DATA_STORE;
212
+ int size = 4 << GMID_EL1_BS;
213
+
214
+ clean_addr = clean_data_tbi(s, addr);
215
+ tcg_gen_andi_i64(clean_addr, clean_addr, -size);
216
+ gen_probe_access(s, clean_addr, acc, size);
217
+ }
218
+ return true;
41
+}
219
+}
42
+
220
+
43
/*
221
+static bool trans_LDGM(DisasContext *s, arg_ldst_tag *a)
44
* Feature tests for "does this exist in either 32-bit or 64-bit?"
222
+{
45
*/
223
+ TCGv_i64 addr, clean_addr, tcg_rt;
46
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_any_ccidx(const ARMISARegisters *id)
224
+
47
return isar_feature_aa64_ccidx(id) || isar_feature_aa32_ccidx(id);
225
+ if (!dc_isar_feature(aa64_mte, s)) {
226
+ return false;
227
+ }
228
+ if (s->current_el == 0) {
229
+ return false;
230
+ }
231
+
232
+ if (a->rn == 31) {
233
+ gen_check_sp_alignment(s);
234
+ }
235
+
236
+ addr = read_cpu_reg_sp(s, a->rn, true);
237
+ tcg_gen_addi_i64(addr, addr, a->imm);
238
+ tcg_rt = cpu_reg(s, a->rt);
239
+
240
+ if (s->ata) {
241
+ gen_helper_ldgm(tcg_rt, cpu_env, addr);
242
+ } else {
243
+ MMUAccessType acc = MMU_DATA_LOAD;
244
+ int size = 4 << GMID_EL1_BS;
245
+
246
+ clean_addr = clean_data_tbi(s, addr);
247
+ tcg_gen_andi_i64(clean_addr, clean_addr, -size);
248
+ gen_probe_access(s, clean_addr, acc, size);
249
+ /* The result tags are zeros. */
250
+ tcg_gen_movi_i64(tcg_rt, 0);
251
+ }
252
+ return true;
253
+}
254
+
255
+static bool trans_LDG(DisasContext *s, arg_ldst_tag *a)
256
+{
257
+ TCGv_i64 addr, clean_addr, tcg_rt;
258
+
259
+ if (!dc_isar_feature(aa64_mte_insn_reg, s)) {
260
+ return false;
261
+ }
262
+
263
+ if (a->rn == 31) {
264
+ gen_check_sp_alignment(s);
265
+ }
266
+
267
+ addr = read_cpu_reg_sp(s, a->rn, true);
268
+ if (!a->p) {
269
/* pre-index or signed offset */
270
- tcg_gen_addi_i64(addr, addr, offset);
271
+ tcg_gen_addi_i64(addr, addr, a->imm);
272
}
273
274
- if (is_mult) {
275
- tcg_rt = cpu_reg(s, rt);
276
+ tcg_gen_andi_i64(addr, addr, -TAG_GRANULE);
277
+ tcg_rt = cpu_reg(s, a->rt);
278
+ if (s->ata) {
279
+ gen_helper_ldg(tcg_rt, cpu_env, addr, tcg_rt);
280
+ } else {
281
+ /*
282
+ * Tag access disabled: we must check for aborts on the load
283
+ * load from [rn+offset], and then insert a 0 tag into rt.
284
+ */
285
+ clean_addr = clean_data_tbi(s, addr);
286
+ gen_probe_access(s, clean_addr, MMU_DATA_LOAD, MO_8);
287
+ gen_address_with_allocation_tag0(tcg_rt, tcg_rt);
288
+ }
289
290
- if (is_zero) {
291
- int size = 4 << s->dcz_blocksize;
292
-
293
- if (s->ata) {
294
- gen_helper_stzgm_tags(cpu_env, addr, tcg_rt);
295
- }
296
- /*
297
- * The non-tags portion of STZGM is mostly like DC_ZVA,
298
- * except the alignment happens before the access.
299
- */
300
- clean_addr = clean_data_tbi(s, addr);
301
- tcg_gen_andi_i64(clean_addr, clean_addr, -size);
302
- gen_helper_dc_zva(cpu_env, clean_addr);
303
- } else if (s->ata) {
304
- if (is_load) {
305
- gen_helper_ldgm(tcg_rt, cpu_env, addr);
306
- } else {
307
- gen_helper_stgm(cpu_env, addr, tcg_rt);
308
- }
309
- } else {
310
- MMUAccessType acc = is_load ? MMU_DATA_LOAD : MMU_DATA_STORE;
311
- int size = 4 << GMID_EL1_BS;
312
-
313
- clean_addr = clean_data_tbi(s, addr);
314
- tcg_gen_andi_i64(clean_addr, clean_addr, -size);
315
- gen_probe_access(s, clean_addr, acc, size);
316
-
317
- if (is_load) {
318
- /* The result tags are zeros. */
319
- tcg_gen_movi_i64(tcg_rt, 0);
320
- }
321
+ if (a->w) {
322
+ /* pre-index or post-index */
323
+ if (a->p) {
324
+ /* post-index */
325
+ tcg_gen_addi_i64(addr, addr, a->imm);
326
}
327
- return;
328
+ tcg_gen_mov_i64(cpu_reg_sp(s, a->rn), addr);
329
+ }
330
+ return true;
331
+}
332
+
333
+static bool do_STG(DisasContext *s, arg_ldst_tag *a, bool is_zero, bool is_pair)
334
+{
335
+ TCGv_i64 addr, tcg_rt;
336
+
337
+ if (a->rn == 31) {
338
+ gen_check_sp_alignment(s);
339
}
340
341
- if (is_load) {
342
- tcg_gen_andi_i64(addr, addr, -TAG_GRANULE);
343
- tcg_rt = cpu_reg(s, rt);
344
- if (s->ata) {
345
- gen_helper_ldg(tcg_rt, cpu_env, addr, tcg_rt);
346
+ addr = read_cpu_reg_sp(s, a->rn, true);
347
+ if (!a->p) {
348
+ /* pre-index or signed offset */
349
+ tcg_gen_addi_i64(addr, addr, a->imm);
350
+ }
351
+ tcg_rt = cpu_reg_sp(s, a->rt);
352
+ if (!s->ata) {
353
+ /*
354
+ * For STG and ST2G, we need to check alignment and probe memory.
355
+ * TODO: For STZG and STZ2G, we could rely on the stores below,
356
+ * at least for system mode; user-only won't enforce alignment.
357
+ */
358
+ if (is_pair) {
359
+ gen_helper_st2g_stub(cpu_env, addr);
360
} else {
361
- /*
362
- * Tag access disabled: we must check for aborts on the load
363
- * load from [rn+offset], and then insert a 0 tag into rt.
364
- */
365
- clean_addr = clean_data_tbi(s, addr);
366
- gen_probe_access(s, clean_addr, MMU_DATA_LOAD, MO_8);
367
- gen_address_with_allocation_tag0(tcg_rt, tcg_rt);
368
+ gen_helper_stg_stub(cpu_env, addr);
369
+ }
370
+ } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
371
+ if (is_pair) {
372
+ gen_helper_st2g_parallel(cpu_env, addr, tcg_rt);
373
+ } else {
374
+ gen_helper_stg_parallel(cpu_env, addr, tcg_rt);
375
}
376
} else {
377
- tcg_rt = cpu_reg_sp(s, rt);
378
- if (!s->ata) {
379
- /*
380
- * For STG and ST2G, we need to check alignment and probe memory.
381
- * TODO: For STZG and STZ2G, we could rely on the stores below,
382
- * at least for system mode; user-only won't enforce alignment.
383
- */
384
- if (is_pair) {
385
- gen_helper_st2g_stub(cpu_env, addr);
386
- } else {
387
- gen_helper_stg_stub(cpu_env, addr);
388
- }
389
- } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
390
- if (is_pair) {
391
- gen_helper_st2g_parallel(cpu_env, addr, tcg_rt);
392
- } else {
393
- gen_helper_stg_parallel(cpu_env, addr, tcg_rt);
394
- }
395
+ if (is_pair) {
396
+ gen_helper_st2g(cpu_env, addr, tcg_rt);
397
} else {
398
- if (is_pair) {
399
- gen_helper_st2g(cpu_env, addr, tcg_rt);
400
- } else {
401
- gen_helper_stg(cpu_env, addr, tcg_rt);
402
- }
403
+ gen_helper_stg(cpu_env, addr, tcg_rt);
404
}
405
}
406
407
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
408
}
409
}
410
411
- if (index != 0) {
412
+ if (a->w) {
413
/* pre-index or post-index */
414
- if (index < 0) {
415
+ if (a->p) {
416
/* post-index */
417
- tcg_gen_addi_i64(addr, addr, offset);
418
+ tcg_gen_addi_i64(addr, addr, a->imm);
419
}
420
- tcg_gen_mov_i64(cpu_reg_sp(s, rn), addr);
421
+ tcg_gen_mov_i64(cpu_reg_sp(s, a->rn), addr);
422
}
423
+ return true;
48
}
424
}
49
425
50
+static inline bool isar_feature_any_tts2uxn(const ARMISARegisters *id)
426
-/* Loads and stores */
51
+{
427
-static void disas_ldst(DisasContext *s, uint32_t insn)
52
+ return isar_feature_aa64_tts2uxn(id) || isar_feature_aa32_tts2uxn(id);
428
-{
53
+}
429
- switch (extract32(insn, 24, 6)) {
54
+
430
- case 0x19:
55
/*
431
- if (extract32(insn, 21, 1) != 0) {
56
* Forward to the above feature tests given an ARMCPU pointer.
432
- disas_ldst_tag(s, insn);
57
*/
433
- } else {
58
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
434
- unallocated_encoding(s);
59
index XXXXXXX..XXXXXXX 100644
435
- }
60
--- a/target/arm/cpu.c
436
- break;
61
+++ b/target/arm/cpu.c
437
- default:
62
@@ -XXX,XX +XXX,XX @@ static void arm_max_initfn(Object *obj)
438
- unallocated_encoding(s);
63
t = FIELD_DP32(t, ID_MMFR4, HPDS, 1); /* AA32HPD */
439
- break;
64
t = FIELD_DP32(t, ID_MMFR4, AC2, 1); /* ACTLR2, HACTLR2 */
440
- }
65
t = FIELD_DP32(t, ID_MMFR4, CNP, 1); /* TTCNP */
441
-}
66
+ t = FIELD_DP32(t, ID_MMFR4, XNX, 1); /* TTS2UXN */
442
+TRANS_FEAT(STG, aa64_mte_insn_reg, do_STG, a, false, false)
67
cpu->isar.id_mmfr4 = t;
443
+TRANS_FEAT(STZG, aa64_mte_insn_reg, do_STG, a, true, false)
68
}
444
+TRANS_FEAT(ST2G, aa64_mte_insn_reg, do_STG, a, false, true)
69
#endif
445
+TRANS_FEAT(STZ2G, aa64_mte_insn_reg, do_STG, a, true, true)
70
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
446
71
index XXXXXXX..XXXXXXX 100644
447
typedef void ArithTwoOp(TCGv_i64, TCGv_i64, TCGv_i64);
72
--- a/target/arm/cpu64.c
448
73
+++ b/target/arm/cpu64.c
449
@@ -XXX,XX +XXX,XX @@ static bool btype_destination_ok(uint32_t insn, bool bt, int btype)
74
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
450
static void disas_a64_legacy(DisasContext *s, uint32_t insn)
75
t = FIELD_DP64(t, ID_AA64MMFR1, VH, 1);
76
t = FIELD_DP64(t, ID_AA64MMFR1, PAN, 2); /* ATS1E1 */
77
t = FIELD_DP64(t, ID_AA64MMFR1, VMIDBITS, 2); /* VMID16 */
78
+ t = FIELD_DP64(t, ID_AA64MMFR1, XNX, 1); /* TTS2UXN */
79
cpu->isar.id_aa64mmfr1 = t;
80
81
t = cpu->isar.id_aa64mmfr2;
82
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
83
u = FIELD_DP32(u, ID_MMFR4, HPDS, 1); /* AA32HPD */
84
u = FIELD_DP32(u, ID_MMFR4, AC2, 1); /* ACTLR2, HACTLR2 */
85
u = FIELD_DP32(u, ID_MMFR4, CNP, 1); /* TTCNP */
86
+ u = FIELD_DP32(u, ID_MMFR4, XNX, 1); /* TTS2UXN */
87
cpu->isar.id_mmfr4 = u;
88
89
u = cpu->isar.id_aa64dfr0;
90
diff --git a/target/arm/helper.c b/target/arm/helper.c
91
index XXXXXXX..XXXXXXX 100644
92
--- a/target/arm/helper.c
93
+++ b/target/arm/helper.c
94
@@ -XXX,XX +XXX,XX @@ simple_ap_to_rw_prot(CPUARMState *env, ARMMMUIdx mmu_idx, int ap)
95
*
96
* @env: CPUARMState
97
* @s2ap: The 2-bit stage2 access permissions (S2AP)
98
- * @xn: XN (execute-never) bit
99
+ * @xn: XN (execute-never) bits
100
+ * @s1_is_el0: true if this is S2 of an S1+2 walk for EL0
101
*/
102
-static int get_S2prot(CPUARMState *env, int s2ap, int xn)
103
+static int get_S2prot(CPUARMState *env, int s2ap, int xn, bool s1_is_el0)
104
{
451
{
105
int prot = 0;
452
switch (extract32(insn, 25, 4)) {
106
453
- case 0x4:
107
@@ -XXX,XX +XXX,XX @@ static int get_S2prot(CPUARMState *env, int s2ap, int xn)
454
- case 0x6:
108
if (s2ap & 2) {
455
- case 0xc:
109
prot |= PAGE_WRITE;
456
- case 0xe: /* Loads and stores */
110
}
457
- disas_ldst(s, insn);
111
- if (!xn) {
458
- break;
112
- if (arm_el_is_aa64(env, 2) || prot & PAGE_READ) {
459
case 0x5:
113
+
460
case 0xd: /* Data processing - register */
114
+ if (cpu_isar_feature(any_tts2uxn, env_archcpu(env))) {
461
disas_data_proc_reg(s, insn);
115
+ switch (xn) {
116
+ case 0:
117
prot |= PAGE_EXEC;
118
+ break;
119
+ case 1:
120
+ if (s1_is_el0) {
121
+ prot |= PAGE_EXEC;
122
+ }
123
+ break;
124
+ case 2:
125
+ break;
126
+ case 3:
127
+ if (!s1_is_el0) {
128
+ prot |= PAGE_EXEC;
129
+ }
130
+ break;
131
+ default:
132
+ g_assert_not_reached();
133
+ }
134
+ } else {
135
+ if (!extract32(xn, 1, 1)) {
136
+ if (arm_el_is_aa64(env, 2) || prot & PAGE_READ) {
137
+ prot |= PAGE_EXEC;
138
+ }
139
}
140
}
141
return prot;
142
@@ -XXX,XX +XXX,XX @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
143
}
144
145
ap = extract32(attrs, 4, 2);
146
- xn = extract32(attrs, 12, 1);
147
148
if (mmu_idx == ARMMMUIdx_Stage2) {
149
ns = true;
150
- *prot = get_S2prot(env, ap, xn);
151
+ xn = extract32(attrs, 11, 2);
152
+ *prot = get_S2prot(env, ap, xn, s1_is_el0);
153
} else {
154
ns = extract32(attrs, 3, 1);
155
+ xn = extract32(attrs, 12, 1);
156
pxn = extract32(attrs, 11, 1);
157
*prot = get_S1prot(env, mmu_idx, aarch64, ap, ns, xn, pxn);
158
}
159
--
462
--
160
2.20.1
463
2.34.1
161
162
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
In commit 2c5fa0778c3b430 we fixed an endianness bug in the Allwinner
2
A10 PIC model; however in the process we introduced a regression.
3
This is because the old code was robust against the incoming 'level'
4
argument being something other than 0 or 1, whereas the new code was
5
not.
2
6
3
hw/arm: versal: Add support for the RTC.
7
In particular, the allwinner-sdhost code treats its IRQ line
8
as 0-vs-non-0 rather than 0-vs-1, so when the SD controller
9
set its IRQ line for any reason other than transmit the
10
interrupt controller would ignore it. The observed effect
11
was a guest timeout when rebooting the guest kernel.
4
12
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
13
Handle level values other than 0 or 1, to restore the old
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
14
behaviour.
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
15
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
16
Fixes: 2c5fa0778c3b430 ("hw/intc/allwinner-a10-pic: Don't use set_bit()/clear_bit()")
9
Message-id: 20200427181649.26851-10-edgar.iglesias@gmail.com
17
Cc: qemu-stable@nongnu.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
18
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
19
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
20
Tested-by: Guenter Roeck <linux@roeck-us.net>
21
Message-id: 20230606104609.3692557-2-peter.maydell@linaro.org
11
---
22
---
12
include/hw/arm/xlnx-versal.h | 8 ++++++++
23
hw/intc/allwinner-a10-pic.c | 2 +-
13
hw/arm/xlnx-versal.c | 21 +++++++++++++++++++++
24
1 file changed, 1 insertion(+), 1 deletion(-)
14
2 files changed, 29 insertions(+)
15
25
16
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
26
diff --git a/hw/intc/allwinner-a10-pic.c b/hw/intc/allwinner-a10-pic.c
17
index XXXXXXX..XXXXXXX 100644
27
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/arm/xlnx-versal.h
28
--- a/hw/intc/allwinner-a10-pic.c
19
+++ b/include/hw/arm/xlnx-versal.h
29
+++ b/hw/intc/allwinner-a10-pic.c
20
@@ -XXX,XX +XXX,XX @@
30
@@ -XXX,XX +XXX,XX @@ static void aw_a10_pic_set_irq(void *opaque, int irq, int level)
21
#include "hw/char/pl011.h"
31
AwA10PICState *s = opaque;
22
#include "hw/dma/xlnx-zdma.h"
32
uint32_t *pending_reg = &s->irq_pending[irq / 32];
23
#include "hw/net/cadence_gem.h"
33
24
+#include "hw/rtc/xlnx-zynqmp-rtc.h"
34
- *pending_reg = deposit32(*pending_reg, irq % 32, 1, level);
25
35
+ *pending_reg = deposit32(*pending_reg, irq % 32, 1, !!level);
26
#define TYPE_XLNX_VERSAL "xlnx-versal"
36
aw_a10_pic_update(s);
27
#define XLNX_VERSAL(obj) OBJECT_CHECK(Versal, (obj), TYPE_XLNX_VERSAL)
28
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
29
struct {
30
SDHCIState sd[XLNX_VERSAL_NR_SDS];
31
} iou;
32
+
33
+ XlnxZynqMPRTC rtc;
34
} pmc;
35
36
struct {
37
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
38
#define VERSAL_GEM1_IRQ_0 58
39
#define VERSAL_GEM1_WAKE_IRQ_0 59
40
#define VERSAL_ADMA_IRQ_0 60
41
+#define VERSAL_RTC_APB_ERR_IRQ 121
42
#define VERSAL_SD0_IRQ_0 126
43
+#define VERSAL_RTC_ALARM_IRQ 142
44
+#define VERSAL_RTC_SECONDS_IRQ 143
45
46
/* Architecturally reserved IRQs suitable for virtualization. */
47
#define VERSAL_RSVD_IRQ_FIRST 111
48
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
49
#define MM_PMC_SD0_SIZE 0x10000
50
#define MM_PMC_CRP 0xf1260000U
51
#define MM_PMC_CRP_SIZE 0x10000
52
+#define MM_PMC_RTC 0xf12a0000
53
+#define MM_PMC_RTC_SIZE 0x10000
54
#endif
55
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
56
index XXXXXXX..XXXXXXX 100644
57
--- a/hw/arm/xlnx-versal.c
58
+++ b/hw/arm/xlnx-versal.c
59
@@ -XXX,XX +XXX,XX @@ static void versal_create_sds(Versal *s, qemu_irq *pic)
60
}
61
}
37
}
62
38
63
+static void versal_create_rtc(Versal *s, qemu_irq *pic)
64
+{
65
+ SysBusDevice *sbd;
66
+ MemoryRegion *mr;
67
+
68
+ sysbus_init_child_obj(OBJECT(s), "rtc", &s->pmc.rtc, sizeof(s->pmc.rtc),
69
+ TYPE_XLNX_ZYNQMP_RTC);
70
+ sbd = SYS_BUS_DEVICE(&s->pmc.rtc);
71
+ qdev_init_nofail(DEVICE(sbd));
72
+
73
+ mr = sysbus_mmio_get_region(sbd, 0);
74
+ memory_region_add_subregion(&s->mr_ps, MM_PMC_RTC, mr);
75
+
76
+ /*
77
+ * TODO: Connect the ALARM and SECONDS interrupts once our RTC model
78
+ * supports them.
79
+ */
80
+ sysbus_connect_irq(sbd, 1, pic[VERSAL_RTC_APB_ERR_IRQ]);
81
+}
82
+
83
/* This takes the board allocated linear DDR memory and creates aliases
84
* for each split DDR range/aperture on the Versal address map.
85
*/
86
@@ -XXX,XX +XXX,XX @@ static void versal_realize(DeviceState *dev, Error **errp)
87
versal_create_gems(s, pic);
88
versal_create_admas(s, pic);
89
versal_create_sds(s, pic);
90
+ versal_create_rtc(s, pic);
91
versal_map_ddr(s);
92
versal_unimp(s);
93
94
--
39
--
95
2.20.1
40
2.34.1
96
41
97
42
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
QEMU allows qemu_irq lines to transfer arbitrary integers. However
2
the convention is that for a simple IRQ line the values transferred
3
are always 0 and 1. The A10 SD controller device instead assumes a
4
0-vs-non-0 convention, which happens to work with the interrupt
5
controller it is wired up to.
2
6
3
Fix typo xlnx-ve -> xlnx-versal.
7
Coerce the value to boolean to follow our usual convention.
4
8
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20200427181649.26851-4-edgar.iglesias@gmail.com
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
11
Tested-by: Guenter Roeck <linux@roeck-us.net>
12
Message-id: 20230606104609.3692557-3-peter.maydell@linaro.org
11
---
13
---
12
hw/arm/xlnx-versal-virt.c | 2 +-
14
hw/sd/allwinner-sdhost.c | 2 +-
13
1 file changed, 1 insertion(+), 1 deletion(-)
15
1 file changed, 1 insertion(+), 1 deletion(-)
14
16
15
diff --git a/hw/arm/xlnx-versal-virt.c b/hw/arm/xlnx-versal-virt.c
17
diff --git a/hw/sd/allwinner-sdhost.c b/hw/sd/allwinner-sdhost.c
16
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/xlnx-versal-virt.c
19
--- a/hw/sd/allwinner-sdhost.c
18
+++ b/hw/arm/xlnx-versal-virt.c
20
+++ b/hw/sd/allwinner-sdhost.c
19
@@ -XXX,XX +XXX,XX @@ static void versal_virt_init(MachineState *machine)
21
@@ -XXX,XX +XXX,XX @@ static void allwinner_sdhost_update_irq(AwSdHostState *s)
20
psci_conduit = QEMU_PSCI_CONDUIT_SMC;
21
}
22
}
22
23
23
- sysbus_init_child_obj(OBJECT(machine), "xlnx-ve", &s->soc,
24
trace_allwinner_sdhost_update_irq(irq);
24
+ sysbus_init_child_obj(OBJECT(machine), "xlnx-versal", &s->soc,
25
- qemu_set_irq(s->irq, irq);
25
sizeof(s->soc), TYPE_XLNX_VERSAL);
26
+ qemu_set_irq(s->irq, !!irq);
26
object_property_set_link(OBJECT(&s->soc), OBJECT(machine->ram),
27
}
27
"ddr", &error_abort);
28
29
static void allwinner_sdhost_update_transfer_cnt(AwSdHostState *s,
28
--
30
--
29
2.20.1
31
2.34.1
30
32
31
33
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
The nrf51_timer has a free-running counter which we implement using
2
the pattern of using two fields (update_counter_ns, counter) to track
3
the last point at which we calculated the counter value, and the
4
counter value at that time. Then we can find the current counter
5
value by converting the difference in wall-clock time between then
6
and now to a tick count that we need to add to the counter value.
2
7
3
Add support for SD.
8
Unfortunately the nrf51_timer's implementation of this has a bug
9
which means it loses time every time update_counter() is called.
10
After updating s->counter it always sets s->update_counter_ns to
11
'now', even though the actual point when s->counter hit the new value
12
will be some point in the past (half a tick, say). In the worst case
13
(guest code in a tight loop reading the counter, icount mode) the
14
counter is continually queried less than a tick after it was last
15
read, so s->counter never advances but s->update_counter_ns does, and
16
the guest never makes forward progress.
4
17
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
18
The fix for this is to only advance update_counter_ns to the
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
19
timestamp of the last tick, not all the way to 'now'. (This is the
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
20
pattern used in hw/misc/mps2-fpgaio.c's counter.)
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
21
9
Message-id: 20200427181649.26851-9-edgar.iglesias@gmail.com
22
Cc: qemu-stable@nongnu.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
23
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
24
Reviewed-by: Joel Stanley <joel@jms.id.au>
25
Message-id: 20230606134917.3782215-1-peter.maydell@linaro.org
11
---
26
---
12
include/hw/arm/xlnx-versal.h | 12 ++++++++++++
27
hw/timer/nrf51_timer.c | 7 ++++++-
13
hw/arm/xlnx-versal.c | 31 +++++++++++++++++++++++++++++++
28
1 file changed, 6 insertions(+), 1 deletion(-)
14
2 files changed, 43 insertions(+)
15
29
16
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
30
diff --git a/hw/timer/nrf51_timer.c b/hw/timer/nrf51_timer.c
17
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/arm/xlnx-versal.h
32
--- a/hw/timer/nrf51_timer.c
19
+++ b/include/hw/arm/xlnx-versal.h
33
+++ b/hw/timer/nrf51_timer.c
20
@@ -XXX,XX +XXX,XX @@
34
@@ -XXX,XX +XXX,XX @@ static uint32_t update_counter(NRF51TimerState *s, int64_t now)
21
35
uint32_t ticks = ns_to_ticks(s, now - s->update_counter_ns);
22
#include "hw/sysbus.h"
36
23
#include "hw/arm/boot.h"
37
s->counter = (s->counter + ticks) % BIT(bitwidths[s->bitmode]);
24
+#include "hw/sd/sdhci.h"
38
- s->update_counter_ns = now;
25
#include "hw/intc/arm_gicv3.h"
39
+ /*
26
#include "hw/char/pl011.h"
40
+ * Only advance the sync time to the timestamp of the last tick,
27
#include "hw/dma/xlnx-zdma.h"
41
+ * not all the way to 'now', so we don't lose time if we do
28
@@ -XXX,XX +XXX,XX @@
42
+ * multiple resyncs in a single tick.
29
#define XLNX_VERSAL_NR_UARTS 2
43
+ */
30
#define XLNX_VERSAL_NR_GEMS 2
44
+ s->update_counter_ns += ticks_to_ns(s, ticks);
31
#define XLNX_VERSAL_NR_ADMAS 8
45
return ticks;
32
+#define XLNX_VERSAL_NR_SDS 2
33
#define XLNX_VERSAL_NR_IRQS 192
34
35
typedef struct Versal {
36
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
37
} iou;
38
} lpd;
39
40
+ /* The Platform Management Controller subsystem. */
41
+ struct {
42
+ struct {
43
+ SDHCIState sd[XLNX_VERSAL_NR_SDS];
44
+ } iou;
45
+ } pmc;
46
+
47
struct {
48
MemoryRegion *mr_ddr;
49
uint32_t psci_conduit;
50
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
51
#define VERSAL_GEM1_IRQ_0 58
52
#define VERSAL_GEM1_WAKE_IRQ_0 59
53
#define VERSAL_ADMA_IRQ_0 60
54
+#define VERSAL_SD0_IRQ_0 126
55
56
/* Architecturally reserved IRQs suitable for virtualization. */
57
#define VERSAL_RSVD_IRQ_FIRST 111
58
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
59
#define MM_FPD_CRF 0xfd1a0000U
60
#define MM_FPD_CRF_SIZE 0x140000
61
62
+#define MM_PMC_SD0 0xf1040000U
63
+#define MM_PMC_SD0_SIZE 0x10000
64
#define MM_PMC_CRP 0xf1260000U
65
#define MM_PMC_CRP_SIZE 0x10000
66
#endif
67
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
68
index XXXXXXX..XXXXXXX 100644
69
--- a/hw/arm/xlnx-versal.c
70
+++ b/hw/arm/xlnx-versal.c
71
@@ -XXX,XX +XXX,XX @@ static void versal_create_admas(Versal *s, qemu_irq *pic)
72
}
73
}
46
}
74
47
75
+#define SDHCI_CAPABILITIES 0x280737ec6481 /* Same as on ZynqMP. */
76
+static void versal_create_sds(Versal *s, qemu_irq *pic)
77
+{
78
+ int i;
79
+
80
+ for (i = 0; i < ARRAY_SIZE(s->pmc.iou.sd); i++) {
81
+ DeviceState *dev;
82
+ MemoryRegion *mr;
83
+
84
+ sysbus_init_child_obj(OBJECT(s), "sd[*]",
85
+ &s->pmc.iou.sd[i], sizeof(s->pmc.iou.sd[i]),
86
+ TYPE_SYSBUS_SDHCI);
87
+ dev = DEVICE(&s->pmc.iou.sd[i]);
88
+
89
+ object_property_set_uint(OBJECT(dev),
90
+ 3, "sd-spec-version", &error_fatal);
91
+ object_property_set_uint(OBJECT(dev), SDHCI_CAPABILITIES, "capareg",
92
+ &error_fatal);
93
+ object_property_set_uint(OBJECT(dev), UHS_I, "uhs", &error_fatal);
94
+ qdev_init_nofail(dev);
95
+
96
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
97
+ memory_region_add_subregion(&s->mr_ps,
98
+ MM_PMC_SD0 + i * MM_PMC_SD0_SIZE, mr);
99
+
100
+ sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0,
101
+ pic[VERSAL_SD0_IRQ_0 + i * 2]);
102
+ }
103
+}
104
+
105
/* This takes the board allocated linear DDR memory and creates aliases
106
* for each split DDR range/aperture on the Versal address map.
107
*/
108
@@ -XXX,XX +XXX,XX @@ static void versal_realize(DeviceState *dev, Error **errp)
109
versal_create_uarts(s, pic);
110
versal_create_gems(s, pic);
111
versal_create_admas(s, pic);
112
+ versal_create_sds(s, pic);
113
versal_map_ddr(s);
114
versal_unimp(s);
115
116
--
48
--
117
2.20.1
49
2.34.1
118
119
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
From: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
2
2
3
Embed the APUs into the SoC type.
3
Signed-off-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
4
4
Reviewed-by: Thomas Huth <thuth@redhat.com>
5
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
5
Message-id: 20230607092112.655098-1-marcin.juszkiewicz@linaro.org
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
10
Message-id: 20200427181649.26851-8-edgar.iglesias@gmail.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
7
---
13
include/hw/arm/xlnx-versal.h | 2 +-
8
hw/arm/Kconfig | 1 +
14
hw/arm/xlnx-versal-virt.c | 4 ++--
9
1 file changed, 1 insertion(+)
15
hw/arm/xlnx-versal.c | 19 +++++--------------
16
3 files changed, 8 insertions(+), 17 deletions(-)
17
10
18
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
11
diff --git a/hw/arm/Kconfig b/hw/arm/Kconfig
19
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
20
--- a/include/hw/arm/xlnx-versal.h
13
--- a/hw/arm/Kconfig
21
+++ b/include/hw/arm/xlnx-versal.h
14
+++ b/hw/arm/Kconfig
22
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
15
@@ -XXX,XX +XXX,XX @@ config SBSA_REF
23
struct {
16
select PL061 # GPIO
24
struct {
17
select USB_EHCI_SYSBUS
25
MemoryRegion mr;
18
select WDT_SBSA
26
- ARMCPU *cpu[XLNX_VERSAL_NR_ACPUS];
19
+ select BOCHS_DISPLAY
27
+ ARMCPU cpu[XLNX_VERSAL_NR_ACPUS];
20
28
GICv3State gic;
21
config SABRELITE
29
} apu;
22
bool
30
} fpd;
31
diff --git a/hw/arm/xlnx-versal-virt.c b/hw/arm/xlnx-versal-virt.c
32
index XXXXXXX..XXXXXXX 100644
33
--- a/hw/arm/xlnx-versal-virt.c
34
+++ b/hw/arm/xlnx-versal-virt.c
35
@@ -XXX,XX +XXX,XX @@ static void versal_virt_init(MachineState *machine)
36
s->binfo.get_dtb = versal_virt_get_dtb;
37
s->binfo.modify_dtb = versal_virt_modify_dtb;
38
if (machine->kernel_filename) {
39
- arm_load_kernel(s->soc.fpd.apu.cpu[0], machine, &s->binfo);
40
+ arm_load_kernel(&s->soc.fpd.apu.cpu[0], machine, &s->binfo);
41
} else {
42
- AddressSpace *as = arm_boot_address_space(s->soc.fpd.apu.cpu[0],
43
+ AddressSpace *as = arm_boot_address_space(&s->soc.fpd.apu.cpu[0],
44
&s->binfo);
45
/* Some boot-loaders (e.g u-boot) don't like blobs at address 0 (NULL).
46
* Offset things by 4K. */
47
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
48
index XXXXXXX..XXXXXXX 100644
49
--- a/hw/arm/xlnx-versal.c
50
+++ b/hw/arm/xlnx-versal.c
51
@@ -XXX,XX +XXX,XX @@ static void versal_create_apu_cpus(Versal *s)
52
53
for (i = 0; i < ARRAY_SIZE(s->fpd.apu.cpu); i++) {
54
Object *obj;
55
- char *name;
56
-
57
- obj = object_new(XLNX_VERSAL_ACPU_TYPE);
58
- if (!obj) {
59
- error_report("Unable to create apu.cpu[%d] of type %s",
60
- i, XLNX_VERSAL_ACPU_TYPE);
61
- exit(EXIT_FAILURE);
62
- }
63
-
64
- name = g_strdup_printf("apu-cpu[%d]", i);
65
- object_property_add_child(OBJECT(s), name, obj, &error_fatal);
66
- g_free(name);
67
68
+ object_initialize_child(OBJECT(s), "apu-cpu[*]",
69
+ &s->fpd.apu.cpu[i], sizeof(s->fpd.apu.cpu[i]),
70
+ XLNX_VERSAL_ACPU_TYPE, &error_abort, NULL);
71
+ obj = OBJECT(&s->fpd.apu.cpu[i]);
72
object_property_set_int(obj, s->cfg.psci_conduit,
73
"psci-conduit", &error_abort);
74
if (i) {
75
@@ -XXX,XX +XXX,XX @@ static void versal_create_apu_cpus(Versal *s)
76
object_property_set_link(obj, OBJECT(&s->fpd.apu.mr), "memory",
77
&error_abort);
78
object_property_set_bool(obj, true, "realized", &error_fatal);
79
- s->fpd.apu.cpu[i] = ARM_CPU(obj);
80
}
81
}
82
83
@@ -XXX,XX +XXX,XX @@ static void versal_create_apu_gic(Versal *s, qemu_irq *pic)
84
}
85
86
for (i = 0; i < nr_apu_cpus; i++) {
87
- DeviceState *cpudev = DEVICE(s->fpd.apu.cpu[i]);
88
+ DeviceState *cpudev = DEVICE(&s->fpd.apu.cpu[i]);
89
int ppibase = XLNX_VERSAL_NR_IRQS + i * GIC_INTERNAL + GIC_NR_SGIS;
90
qemu_irq maint_irq;
91
int ti;
92
--
23
--
93
2.20.1
24
2.34.1
94
95
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
From: Martin Kaiser <martin@kaiser.cx>
2
2
3
Remove inclusion of arm_gicv3_common.h, this already gets
3
The Linux kernel added a flood check for RX data recently in commit
4
included via xlnx-versal.h.
4
496a4471b7c3 ("serial: imx: work-around for hardware RX flood"). This
5
check uses the wake bit in the UART status register 2. The wake bit
6
indicates that the receiver detected a start bit on the RX line. If the
7
kernel sees a number of RX interrupts without the wake bit being set, it
8
treats this as spurious data and resets the UART port. imx_serial does
9
never set the wake bit and triggers the kernel's flood check.
5
10
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
11
This patch adds support for the wake bit. wake is set when we receive a
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
12
new character (it's not set for break events). It seems that wake is
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
13
cleared by the kernel driver, the hardware does not have to clear it
9
Message-id: 20200427181649.26851-2-edgar.iglesias@gmail.com
14
automatically after data was read.
15
16
The wake bit can be configured as an interrupt source. Support this
17
mechanism as well.
18
19
Co-developed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
20
Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org>
21
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
22
Signed-off-by: Martin Kaiser <martin@kaiser.cx>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
23
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
24
---
12
hw/arm/xlnx-versal.c | 1 -
25
include/hw/char/imx_serial.h | 1 +
13
1 file changed, 1 deletion(-)
26
hw/char/imx_serial.c | 5 ++++-
27
2 files changed, 5 insertions(+), 1 deletion(-)
14
28
15
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
29
diff --git a/include/hw/char/imx_serial.h b/include/hw/char/imx_serial.h
16
index XXXXXXX..XXXXXXX 100644
30
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/xlnx-versal.c
31
--- a/include/hw/char/imx_serial.h
18
+++ b/hw/arm/xlnx-versal.c
32
+++ b/include/hw/char/imx_serial.h
19
@@ -XXX,XX +XXX,XX @@
33
@@ -XXX,XX +XXX,XX @@ OBJECT_DECLARE_SIMPLE_TYPE(IMXSerialState, IMX_SERIAL)
20
#include "hw/arm/boot.h"
34
21
#include "kvm_arm.h"
35
#define UCR4_DREN BIT(0) /* Receive Data Ready interrupt enable */
22
#include "hw/misc/unimp.h"
36
#define UCR4_TCEN BIT(3) /* TX complete interrupt enable */
23
-#include "hw/intc/arm_gicv3_common.h"
37
+#define UCR4_WKEN BIT(7) /* WAKE interrupt enable */
24
#include "hw/arm/xlnx-versal.h"
38
25
#include "hw/char/pl011.h"
39
#define UTS1_TXEMPTY (1<<6)
40
#define UTS1_RXEMPTY (1<<5)
41
diff --git a/hw/char/imx_serial.c b/hw/char/imx_serial.c
42
index XXXXXXX..XXXXXXX 100644
43
--- a/hw/char/imx_serial.c
44
+++ b/hw/char/imx_serial.c
45
@@ -XXX,XX +XXX,XX @@ static void imx_update(IMXSerialState *s)
46
* TCEN and TXDC are both bit 3
47
* RDR and DREN are both bit 0
48
*/
49
- mask |= s->ucr4 & (UCR4_TCEN | UCR4_DREN);
50
+ mask |= s->ucr4 & (UCR4_WKEN | UCR4_TCEN | UCR4_DREN);
51
52
usr2 = s->usr2 & mask;
53
54
@@ -XXX,XX +XXX,XX @@ static void imx_put_data(void *opaque, uint32_t value)
55
56
static void imx_receive(void *opaque, const uint8_t *buf, int size)
57
{
58
+ IMXSerialState *s = (IMXSerialState *)opaque;
59
+
60
+ s->usr2 |= USR2_WAKE;
61
imx_put_data(opaque, *buf);
62
}
26
63
27
--
64
--
28
2.20.1
65
2.34.1
29
66
30
67
diff view generated by jsdifflib
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
1
From: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
2
2
3
By using the TYPE_* definitions for devices, we can:
3
We plan to add more hardware information into DeviceTree to limit amount
4
- quickly find where devices are used with 'git-grep'
4
of hardcoded values in firmware.
5
- easily rename a device (one-line change).
6
5
7
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Signed-off-by: Marcin Juszkiewicz <marcin.juszkiewicz@linaro.org>
8
Message-id: 20200428154650.21991-1-f4bug@amsat.org
7
Message-id: 20230531171834.236569-1-marcin.juszkiewicz@linaro.org
8
[PMM: fix format nits, add text about platform version fields from
9
a comment in the C source file]
9
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
---
12
hw/arm/mps2-tz.c | 2 +-
13
docs/system/arm/sbsa.rst | 38 +++++++++++++++++++++++++++++++-------
13
1 file changed, 1 insertion(+), 1 deletion(-)
14
1 file changed, 31 insertions(+), 7 deletions(-)
14
15
15
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
16
diff --git a/docs/system/arm/sbsa.rst b/docs/system/arm/sbsa.rst
16
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/mps2-tz.c
18
--- a/docs/system/arm/sbsa.rst
18
+++ b/hw/arm/mps2-tz.c
19
+++ b/docs/system/arm/sbsa.rst
19
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
20
@@ -XXX,XX +XXX,XX @@ any real hardware the ``sbsa-ref`` board intends to look like real
20
exit(EXIT_FAILURE);
21
hardware. The `Server Base System Architecture
21
}
22
<https://developer.arm.com/documentation/den0029/latest>`_ defines a
22
23
minimum base line of hardware support and importantly how the firmware
23
- sysbus_init_child_obj(OBJECT(machine), "iotkit", &mms->iotkit,
24
-reports that to any operating system. It is a static system that
24
+ sysbus_init_child_obj(OBJECT(machine), TYPE_IOTKIT, &mms->iotkit,
25
-reports a very minimal DT to the firmware for non-discoverable
25
sizeof(mms->iotkit), mmc->armsse_type);
26
-information about components affected by the qemu command line (i.e.
26
iotkitdev = DEVICE(&mms->iotkit);
27
-cpus and memory). As a result it must have a firmware specifically
27
object_property_set_link(OBJECT(&mms->iotkit), OBJECT(system_memory),
28
-built to expect a certain hardware layout (as you would in a real
29
-machine).
30
+reports that to any operating system.
31
32
It is intended to be a machine for developing firmware and testing
33
standards compliance with operating systems.
34
@@ -XXX,XX +XXX,XX @@ standards compliance with operating systems.
35
Supported devices
36
"""""""""""""""""
37
38
-The sbsa-ref board supports:
39
+The ``sbsa-ref`` board supports:
40
41
- A configurable number of AArch64 CPUs
42
- GIC version 3
43
@@ -XXX,XX +XXX,XX @@ The sbsa-ref board supports:
44
- Bochs display adapter on PCIe bus
45
- A generic SBSA watchdog device
46
47
+
48
+Board to firmware interface
49
+"""""""""""""""""""""""""""
50
+
51
+``sbsa-ref`` is a static system that reports a very minimal devicetree to the
52
+firmware for non-discoverable information about system components. This
53
+includes both internal hardware and parts affected by the qemu command line
54
+(i.e. CPUs and memory). As a result it must have a firmware specifically built
55
+to expect a certain hardware layout (as you would in a real machine).
56
+
57
+DeviceTree information
58
+''''''''''''''''''''''
59
+
60
+The devicetree provided by the board model to the firmware is not intended
61
+to be a complete compliant DT. It currently reports:
62
+
63
+ - CPUs
64
+ - memory
65
+ - platform version
66
+ - GIC addresses
67
+
68
+The platform version is only for informing platform firmware about
69
+what kind of ``sbsa-ref`` board it is running on. It is neither
70
+a QEMU versioned machine type nor a reflection of the level of the
71
+SBSA/SystemReady SR support provided.
72
+
73
+The ``machine-version-major`` value is updated when changes breaking
74
+fw compatibility are introduced. The ``machine-version-minor`` value
75
+is updated when features are added that don't break fw compatibility.
28
--
76
--
29
2.20.1
77
2.34.1
30
31
diff view generated by jsdifflib
Deleted patch
1
The access_type argument to get_phys_addr_lpae() is an MMUAccessType;
2
use the enum constant MMU_DATA_LOAD rather than a literal 0 when we
3
call it in S1_ptw_translate().
4
1
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200330210400.11724-3-peter.maydell@linaro.org
9
---
10
target/arm/helper.c | 5 +++--
11
1 file changed, 3 insertions(+), 2 deletions(-)
12
13
diff --git a/target/arm/helper.c b/target/arm/helper.c
14
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/helper.c
16
+++ b/target/arm/helper.c
17
@@ -XXX,XX +XXX,XX @@ static hwaddr S1_ptw_translate(CPUARMState *env, ARMMMUIdx mmu_idx,
18
pcacheattrs = &cacheattrs;
19
}
20
21
- ret = get_phys_addr_lpae(env, addr, 0, ARMMMUIdx_Stage2, &s2pa,
22
- &txattrs, &s2prot, &s2size, fi, pcacheattrs);
23
+ ret = get_phys_addr_lpae(env, addr, MMU_DATA_LOAD, ARMMMUIdx_Stage2,
24
+ &s2pa, &txattrs, &s2prot, &s2size, fi,
25
+ pcacheattrs);
26
if (ret) {
27
assert(fi->type != ARMFault_None);
28
fi->s2addr = addr;
29
--
30
2.20.1
31
32
diff view generated by jsdifflib
Deleted patch
1
In aarch64_max_initfn() we update both 32-bit and 64-bit ID
2
registers. The intended pattern is that for 64-bit ID registers we
3
use FIELD_DP64 and the uint64_t 't' register, while 32-bit ID
4
registers use FIELD_DP32 and the uint32_t 'u' register. For
5
ID_AA64DFR0 we accidentally used 'u', meaning that the top 32 bits of
6
this 64-bit ID register would end up always zero. Luckily at the
7
moment that's what they should be anyway, so this bug has no visible
8
effects.
9
1
10
Use the right-sized variable.
11
12
Fixes: 3bec78447a958d481991
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
14
Reviewed-by: Laurent Desnogues <laurent.desnogues@gmail.com>
15
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
16
Message-id: 20200423110915.10527-1-peter.maydell@linaro.org
17
---
18
target/arm/cpu64.c | 6 +++---
19
1 file changed, 3 insertions(+), 3 deletions(-)
20
21
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
22
index XXXXXXX..XXXXXXX 100644
23
--- a/target/arm/cpu64.c
24
+++ b/target/arm/cpu64.c
25
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
26
u = FIELD_DP32(u, ID_MMFR4, XNX, 1); /* TTS2UXN */
27
cpu->isar.id_mmfr4 = u;
28
29
- u = cpu->isar.id_aa64dfr0;
30
- u = FIELD_DP64(u, ID_AA64DFR0, PMUVER, 5); /* v8.4-PMU */
31
- cpu->isar.id_aa64dfr0 = u;
32
+ t = cpu->isar.id_aa64dfr0;
33
+ t = FIELD_DP64(t, ID_AA64DFR0, PMUVER, 5); /* v8.4-PMU */
34
+ cpu->isar.id_aa64dfr0 = t;
35
36
u = cpu->isar.id_dfr0;
37
u = FIELD_DP32(u, ID_DFR0, PERFMON, 5); /* v8.4-PMU */
38
--
39
2.20.1
40
41
diff view generated by jsdifflib
1
Add the infrastructure for building and invoking a decodetree decoder
1
From: Sergey Kambalin <sergey.kambalin@auriga.com>
2
for the AArch32 Neon encodings. At the moment the new decoder covers
3
nothing, so we always fall back to the existing hand-written decode.
4
2
5
We follow the same pattern we did for the VFP decodetree conversion
3
Signed-off-by: Sergey Kambalin <sergey.kambalin@auriga.com>
6
(commit 78e138bc1f672c145ef6ace74617d and following): code that deals
4
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
7
with Neon will be moving gradually out to translate-neon.vfp.inc,
5
Acked-by: Richard Henderson <richard.henderson@linaro.org>
8
which we #include into translate.c.
6
Message-id: 20230612223456.33824-2-philmd@linaro.org
7
Message-Id: <20230531155258.8361-1-sergey.kambalin@auriga.com>
8
[PMD: Split from bigger patch: 1/4]
9
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
include/hw/misc/raspberrypi-fw-defs.h | 163 ++++++++++++++++++++++++++
13
1 file changed, 163 insertions(+)
14
create mode 100644 include/hw/misc/raspberrypi-fw-defs.h
9
15
10
In order to share the decode files between A32 and T32, we
16
diff --git a/include/hw/misc/raspberrypi-fw-defs.h b/include/hw/misc/raspberrypi-fw-defs.h
11
split Neon into 3 parts:
12
* data-processing
13
* load-store
14
* 'shared' encodings
15
16
The first two groups of instructions have similar but not identical
17
A32 and T32 encodings, so we need to manually transform the T32
18
encoding into the A32 one before calling the decoder; the third group
19
covers the Neon instructions which are identical in A32 and T32.
20
21
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
22
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
23
Message-id: 20200430181003.21682-4-peter.maydell@linaro.org
24
---
25
target/arm/neon-dp.decode | 29 ++++++++++++++++++++++++++
26
target/arm/neon-ls.decode | 29 ++++++++++++++++++++++++++
27
target/arm/neon-shared.decode | 27 +++++++++++++++++++++++++
28
target/arm/translate-neon.inc.c | 32 +++++++++++++++++++++++++++++
29
target/arm/translate.c | 36 +++++++++++++++++++++++++++++++--
30
target/arm/Makefile.objs | 18 +++++++++++++++++
31
6 files changed, 169 insertions(+), 2 deletions(-)
32
create mode 100644 target/arm/neon-dp.decode
33
create mode 100644 target/arm/neon-ls.decode
34
create mode 100644 target/arm/neon-shared.decode
35
create mode 100644 target/arm/translate-neon.inc.c
36
37
diff --git a/target/arm/neon-dp.decode b/target/arm/neon-dp.decode
38
new file mode 100644
17
new file mode 100644
39
index XXXXXXX..XXXXXXX
18
index XXXXXXX..XXXXXXX
40
--- /dev/null
19
--- /dev/null
41
+++ b/target/arm/neon-dp.decode
20
+++ b/include/hw/misc/raspberrypi-fw-defs.h
42
@@ -XXX,XX +XXX,XX @@
43
+# AArch32 Neon data-processing instruction descriptions
44
+#
45
+# Copyright (c) 2020 Linaro, Ltd
46
+#
47
+# This library is free software; you can redistribute it and/or
48
+# modify it under the terms of the GNU Lesser General Public
49
+# License as published by the Free Software Foundation; either
50
+# version 2 of the License, or (at your option) any later version.
51
+#
52
+# This library is distributed in the hope that it will be useful,
53
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
54
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
55
+# Lesser General Public License for more details.
56
+#
57
+# You should have received a copy of the GNU Lesser General Public
58
+# License along with this library; if not, see <http://www.gnu.org/licenses/>.
59
+
60
+#
61
+# This file is processed by scripts/decodetree.py
62
+#
63
+
64
+# Encodings for Neon data processing instructions where the T32 encoding
65
+# is a simple transformation of the A32 encoding.
66
+# More specifically, this file covers instructions where the A32 encoding is
67
+# 0b1111_001p_qqqq_qqqq_qqqq_qqqq_qqqq_qqqq
68
+# and the T32 encoding is
69
+# 0b111p_1111_qqqq_qqqq_qqqq_qqqq_qqqq_qqqq
70
+# This file works on the A32 encoding only; calling code for T32 has to
71
+# transform the insn into the A32 version first.
72
diff --git a/target/arm/neon-ls.decode b/target/arm/neon-ls.decode
73
new file mode 100644
74
index XXXXXXX..XXXXXXX
75
--- /dev/null
76
+++ b/target/arm/neon-ls.decode
77
@@ -XXX,XX +XXX,XX @@
78
+# AArch32 Neon load/store instruction descriptions
79
+#
80
+# Copyright (c) 2020 Linaro, Ltd
81
+#
82
+# This library is free software; you can redistribute it and/or
83
+# modify it under the terms of the GNU Lesser General Public
84
+# License as published by the Free Software Foundation; either
85
+# version 2 of the License, or (at your option) any later version.
86
+#
87
+# This library is distributed in the hope that it will be useful,
88
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
89
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
90
+# Lesser General Public License for more details.
91
+#
92
+# You should have received a copy of the GNU Lesser General Public
93
+# License along with this library; if not, see <http://www.gnu.org/licenses/>.
94
+
95
+#
96
+# This file is processed by scripts/decodetree.py
97
+#
98
+
99
+# Encodings for Neon load/store instructions where the T32 encoding
100
+# is a simple transformation of the A32 encoding.
101
+# More specifically, this file covers instructions where the A32 encoding is
102
+# 0b1111_0100_xxx0_xxxx_xxxx_xxxx_xxxx_xxxx
103
+# and the T32 encoding is
104
+# 0b1111_1001_xxx0_xxxx_xxxx_xxxx_xxxx_xxxx
105
+# This file works on the A32 encoding only; calling code for T32 has to
106
+# transform the insn into the A32 version first.
107
diff --git a/target/arm/neon-shared.decode b/target/arm/neon-shared.decode
108
new file mode 100644
109
index XXXXXXX..XXXXXXX
110
--- /dev/null
111
+++ b/target/arm/neon-shared.decode
112
@@ -XXX,XX +XXX,XX @@
113
+# AArch32 Neon instruction descriptions
114
+#
115
+# Copyright (c) 2020 Linaro, Ltd
116
+#
117
+# This library is free software; you can redistribute it and/or
118
+# modify it under the terms of the GNU Lesser General Public
119
+# License as published by the Free Software Foundation; either
120
+# version 2 of the License, or (at your option) any later version.
121
+#
122
+# This library is distributed in the hope that it will be useful,
123
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
124
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
125
+# Lesser General Public License for more details.
126
+#
127
+# You should have received a copy of the GNU Lesser General Public
128
+# License along with this library; if not, see <http://www.gnu.org/licenses/>.
129
+
130
+#
131
+# This file is processed by scripts/decodetree.py
132
+#
133
+
134
+# Encodings for Neon instructions whose encoding is the same for
135
+# both A32 and T32.
136
+
137
+# More specifically, this covers:
138
+# 2reg scalar ext: 0b1111_1110_xxxx_xxxx_xxxx_1x0x_xxxx_xxxx
139
+# 3same ext: 0b1111_110x_xxxx_xxxx_xxxx_1x0x_xxxx_xxxx
140
diff --git a/target/arm/translate-neon.inc.c b/target/arm/translate-neon.inc.c
141
new file mode 100644
142
index XXXXXXX..XXXXXXX
143
--- /dev/null
144
+++ b/target/arm/translate-neon.inc.c
145
@@ -XXX,XX +XXX,XX @@
21
@@ -XXX,XX +XXX,XX @@
146
+/*
22
+/*
147
+ * ARM translation: AArch32 Neon instructions
23
+ * Raspberry Pi firmware definitions
148
+ *
24
+ *
149
+ * Copyright (c) 2003 Fabrice Bellard
25
+ * Copyright (C) 2022 Auriga LLC, based on Linux kernel
150
+ * Copyright (c) 2005-2007 CodeSourcery
26
+ * `include/soc/bcm2835/raspberrypi-firmware.h` (Copyright © 2015 Broadcom)
151
+ * Copyright (c) 2007 OpenedHand, Ltd.
152
+ * Copyright (c) 2020 Linaro, Ltd.
153
+ *
27
+ *
154
+ * This library is free software; you can redistribute it and/or
28
+ * SPDX-License-Identifier: GPL-2.0-or-later
155
+ * modify it under the terms of the GNU Lesser General Public
156
+ * License as published by the Free Software Foundation; either
157
+ * version 2 of the License, or (at your option) any later version.
158
+ *
159
+ * This library is distributed in the hope that it will be useful,
160
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
161
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
162
+ * Lesser General Public License for more details.
163
+ *
164
+ * You should have received a copy of the GNU Lesser General Public
165
+ * License along with this library; if not, see <http://www.gnu.org/licenses/>.
166
+ */
29
+ */
167
+
30
+
168
+/*
31
+#ifndef INCLUDE_HW_MISC_RASPBERRYPI_FW_DEFS_H_
169
+ * This file is intended to be included from translate.c; it uses
32
+#define INCLUDE_HW_MISC_RASPBERRYPI_FW_DEFS_H_
170
+ * some macros and definitions provided by that file.
171
+ * It might be possible to convert it to a standalone .c file eventually.
172
+ */
173
+
33
+
174
+/* Include the generated Neon decoder */
34
+#include "qemu/osdep.h"
175
+#include "decode-neon-dp.inc.c"
176
+#include "decode-neon-ls.inc.c"
177
+#include "decode-neon-shared.inc.c"
178
diff --git a/target/arm/translate.c b/target/arm/translate.c
179
index XXXXXXX..XXXXXXX 100644
180
--- a/target/arm/translate.c
181
+++ b/target/arm/translate.c
182
@@ -XXX,XX +XXX,XX @@ static TCGv_ptr vfp_reg_ptr(bool dp, int reg)
183
184
#define ARM_CP_RW_BIT (1 << 20)
185
186
-/* Include the VFP decoder */
187
+/* Include the VFP and Neon decoders */
188
#include "translate-vfp.inc.c"
189
+#include "translate-neon.inc.c"
190
191
static inline void iwmmxt_load_reg(TCGv_i64 var, int reg)
192
{
193
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
194
/* Unconditional instructions. */
195
/* TODO: Perhaps merge these into one decodetree output file. */
196
if (disas_a32_uncond(s, insn) ||
197
- disas_vfp_uncond(s, insn)) {
198
+ disas_vfp_uncond(s, insn) ||
199
+ disas_neon_dp(s, insn) ||
200
+ disas_neon_ls(s, insn) ||
201
+ disas_neon_shared(s, insn)) {
202
return;
203
}
204
/* fall back to legacy decoder */
205
@@ -XXX,XX +XXX,XX @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn)
206
ARCH(6T2);
207
}
208
209
+ if ((insn & 0xef000000) == 0xef000000) {
210
+ /*
211
+ * T32 encodings 0b111p_1111_qqqq_qqqq_qqqq_qqqq_qqqq_qqqq
212
+ * transform into
213
+ * A32 encodings 0b1111_001p_qqqq_qqqq_qqqq_qqqq_qqqq_qqqq
214
+ */
215
+ uint32_t a32_insn = (insn & 0xe2ffffff) |
216
+ ((insn & (1 << 28)) >> 4) | (1 << 28);
217
+
35
+
218
+ if (disas_neon_dp(s, a32_insn)) {
36
+enum rpi_firmware_property_tag {
219
+ return;
37
+ RPI_FWREQ_PROPERTY_END = 0,
220
+ }
38
+ RPI_FWREQ_GET_FIRMWARE_REVISION = 0x00000001,
221
+ }
39
+ RPI_FWREQ_GET_FIRMWARE_VARIANT = 0x00000002,
40
+ RPI_FWREQ_GET_FIRMWARE_HASH = 0x00000003,
222
+
41
+
223
+ if ((insn & 0xff100000) == 0xf9000000) {
42
+ RPI_FWREQ_SET_CURSOR_INFO = 0x00008010,
224
+ /*
43
+ RPI_FWREQ_SET_CURSOR_STATE = 0x00008011,
225
+ * T32 encodings 0b1111_1001_ppp0_qqqq_qqqq_qqqq_qqqq_qqqq
226
+ * transform into
227
+ * A32 encodings 0b1111_0100_ppp0_qqqq_qqqq_qqqq_qqqq_qqqq
228
+ */
229
+ uint32_t a32_insn = (insn & 0x00ffffff) | 0xf4000000;
230
+
44
+
231
+ if (disas_neon_ls(s, a32_insn)) {
45
+ RPI_FWREQ_GET_BOARD_MODEL = 0x00010001,
232
+ return;
46
+ RPI_FWREQ_GET_BOARD_REVISION = 0x00010002,
233
+ }
47
+ RPI_FWREQ_GET_BOARD_MAC_ADDRESS = 0x00010003,
234
+ }
48
+ RPI_FWREQ_GET_BOARD_SERIAL = 0x00010004,
49
+ RPI_FWREQ_GET_ARM_MEMORY = 0x00010005,
50
+ RPI_FWREQ_GET_VC_MEMORY = 0x00010006,
51
+ RPI_FWREQ_GET_CLOCKS = 0x00010007,
52
+ RPI_FWREQ_GET_POWER_STATE = 0x00020001,
53
+ RPI_FWREQ_GET_TIMING = 0x00020002,
54
+ RPI_FWREQ_SET_POWER_STATE = 0x00028001,
55
+ RPI_FWREQ_GET_CLOCK_STATE = 0x00030001,
56
+ RPI_FWREQ_GET_CLOCK_RATE = 0x00030002,
57
+ RPI_FWREQ_GET_VOLTAGE = 0x00030003,
58
+ RPI_FWREQ_GET_MAX_CLOCK_RATE = 0x00030004,
59
+ RPI_FWREQ_GET_MAX_VOLTAGE = 0x00030005,
60
+ RPI_FWREQ_GET_TEMPERATURE = 0x00030006,
61
+ RPI_FWREQ_GET_MIN_CLOCK_RATE = 0x00030007,
62
+ RPI_FWREQ_GET_MIN_VOLTAGE = 0x00030008,
63
+ RPI_FWREQ_GET_TURBO = 0x00030009,
64
+ RPI_FWREQ_GET_MAX_TEMPERATURE = 0x0003000a,
65
+ RPI_FWREQ_GET_STC = 0x0003000b,
66
+ RPI_FWREQ_ALLOCATE_MEMORY = 0x0003000c,
67
+ RPI_FWREQ_LOCK_MEMORY = 0x0003000d,
68
+ RPI_FWREQ_UNLOCK_MEMORY = 0x0003000e,
69
+ RPI_FWREQ_RELEASE_MEMORY = 0x0003000f,
70
+ RPI_FWREQ_EXECUTE_CODE = 0x00030010,
71
+ RPI_FWREQ_EXECUTE_QPU = 0x00030011,
72
+ RPI_FWREQ_SET_ENABLE_QPU = 0x00030012,
73
+ RPI_FWREQ_GET_DISPMANX_RESOURCE_MEM_HANDLE = 0x00030014,
74
+ RPI_FWREQ_GET_EDID_BLOCK = 0x00030020,
75
+ RPI_FWREQ_GET_CUSTOMER_OTP = 0x00030021,
76
+ RPI_FWREQ_GET_EDID_BLOCK_DISPLAY = 0x00030023,
77
+ RPI_FWREQ_GET_DOMAIN_STATE = 0x00030030,
78
+ RPI_FWREQ_GET_THROTTLED = 0x00030046,
79
+ RPI_FWREQ_GET_CLOCK_MEASURED = 0x00030047,
80
+ RPI_FWREQ_NOTIFY_REBOOT = 0x00030048,
81
+ RPI_FWREQ_SET_CLOCK_STATE = 0x00038001,
82
+ RPI_FWREQ_SET_CLOCK_RATE = 0x00038002,
83
+ RPI_FWREQ_SET_VOLTAGE = 0x00038003,
84
+ RPI_FWREQ_SET_MAX_CLOCK_RATE = 0x00038004,
85
+ RPI_FWREQ_SET_MIN_CLOCK_RATE = 0x00038007,
86
+ RPI_FWREQ_SET_TURBO = 0x00038009,
87
+ RPI_FWREQ_SET_CUSTOMER_OTP = 0x00038021,
88
+ RPI_FWREQ_SET_DOMAIN_STATE = 0x00038030,
89
+ RPI_FWREQ_GET_GPIO_STATE = 0x00030041,
90
+ RPI_FWREQ_SET_GPIO_STATE = 0x00038041,
91
+ RPI_FWREQ_SET_SDHOST_CLOCK = 0x00038042,
92
+ RPI_FWREQ_GET_GPIO_CONFIG = 0x00030043,
93
+ RPI_FWREQ_SET_GPIO_CONFIG = 0x00038043,
94
+ RPI_FWREQ_GET_PERIPH_REG = 0x00030045,
95
+ RPI_FWREQ_SET_PERIPH_REG = 0x00038045,
96
+ RPI_FWREQ_GET_POE_HAT_VAL = 0x00030049,
97
+ RPI_FWREQ_SET_POE_HAT_VAL = 0x00038049,
98
+ RPI_FWREQ_SET_POE_HAT_VAL_OLD = 0x00030050,
99
+ RPI_FWREQ_NOTIFY_XHCI_RESET = 0x00030058,
100
+ RPI_FWREQ_GET_REBOOT_FLAGS = 0x00030064,
101
+ RPI_FWREQ_SET_REBOOT_FLAGS = 0x00038064,
102
+ RPI_FWREQ_NOTIFY_DISPLAY_DONE = 0x00030066,
235
+
103
+
236
/*
104
+ /* Dispmanx TAGS */
237
* TODO: Perhaps merge these into one decodetree output file.
105
+ RPI_FWREQ_FRAMEBUFFER_ALLOCATE = 0x00040001,
238
* Note disas_vfp is written for a32 with cond field in the
106
+ RPI_FWREQ_FRAMEBUFFER_BLANK = 0x00040002,
239
@@ -XXX,XX +XXX,XX @@ static void disas_thumb2_insn(DisasContext *s, uint32_t insn)
107
+ RPI_FWREQ_FRAMEBUFFER_GET_PHYSICAL_WIDTH_HEIGHT = 0x00040003,
240
*/
108
+ RPI_FWREQ_FRAMEBUFFER_GET_VIRTUAL_WIDTH_HEIGHT = 0x00040004,
241
if (disas_t32(s, insn) ||
109
+ RPI_FWREQ_FRAMEBUFFER_GET_DEPTH = 0x00040005,
242
disas_vfp_uncond(s, insn) ||
110
+ RPI_FWREQ_FRAMEBUFFER_GET_PIXEL_ORDER = 0x00040006,
243
+ disas_neon_shared(s, insn) ||
111
+ RPI_FWREQ_FRAMEBUFFER_GET_ALPHA_MODE = 0x00040007,
244
((insn >> 28) == 0xe && disas_vfp(s, insn))) {
112
+ RPI_FWREQ_FRAMEBUFFER_GET_PITCH = 0x00040008,
245
return;
113
+ RPI_FWREQ_FRAMEBUFFER_GET_VIRTUAL_OFFSET = 0x00040009,
246
}
114
+ RPI_FWREQ_FRAMEBUFFER_GET_OVERSCAN = 0x0004000a,
247
diff --git a/target/arm/Makefile.objs b/target/arm/Makefile.objs
115
+ RPI_FWREQ_FRAMEBUFFER_GET_PALETTE = 0x0004000b,
248
index XXXXXXX..XXXXXXX 100644
116
+ RPI_FWREQ_FRAMEBUFFER_GET_LAYER = 0x0004000c,
249
--- a/target/arm/Makefile.objs
117
+ RPI_FWREQ_FRAMEBUFFER_GET_TRANSFORM = 0x0004000d,
250
+++ b/target/arm/Makefile.objs
118
+ RPI_FWREQ_FRAMEBUFFER_GET_VSYNC = 0x0004000e,
251
@@ -XXX,XX +XXX,XX @@ target/arm/decode-sve.inc.c: $(SRC_PATH)/target/arm/sve.decode $(DECODETREE)
119
+ RPI_FWREQ_FRAMEBUFFER_GET_TOUCHBUF = 0x0004000f,
252
     $(PYTHON) $(DECODETREE) --decode disas_sve -o $@ $<,\
120
+ RPI_FWREQ_FRAMEBUFFER_GET_GPIOVIRTBUF = 0x00040010,
253
     "GEN", $(TARGET_DIR)$@)
121
+ RPI_FWREQ_FRAMEBUFFER_RELEASE = 0x00048001,
254
122
+ RPI_FWREQ_FRAMEBUFFER_GET_DISPLAY_ID = 0x00040016,
255
+target/arm/decode-neon-shared.inc.c: $(SRC_PATH)/target/arm/neon-shared.decode $(DECODETREE)
123
+ RPI_FWREQ_FRAMEBUFFER_SET_DISPLAY_NUM = 0x00048013,
256
+    $(call quiet-command,\
124
+ RPI_FWREQ_FRAMEBUFFER_GET_NUM_DISPLAYS = 0x00040013,
257
+     $(PYTHON) $(DECODETREE) --static-decode disas_neon_shared -o $@ $<,\
125
+ RPI_FWREQ_FRAMEBUFFER_GET_DISPLAY_SETTINGS = 0x00040014,
258
+     "GEN", $(TARGET_DIR)$@)
126
+ RPI_FWREQ_FRAMEBUFFER_TEST_PHYSICAL_WIDTH_HEIGHT = 0x00044003,
127
+ RPI_FWREQ_FRAMEBUFFER_TEST_VIRTUAL_WIDTH_HEIGHT = 0x00044004,
128
+ RPI_FWREQ_FRAMEBUFFER_TEST_DEPTH = 0x00044005,
129
+ RPI_FWREQ_FRAMEBUFFER_TEST_PIXEL_ORDER = 0x00044006,
130
+ RPI_FWREQ_FRAMEBUFFER_TEST_ALPHA_MODE = 0x00044007,
131
+ RPI_FWREQ_FRAMEBUFFER_TEST_VIRTUAL_OFFSET = 0x00044009,
132
+ RPI_FWREQ_FRAMEBUFFER_TEST_OVERSCAN = 0x0004400a,
133
+ RPI_FWREQ_FRAMEBUFFER_TEST_PALETTE = 0x0004400b,
134
+ RPI_FWREQ_FRAMEBUFFER_TEST_LAYER = 0x0004400c,
135
+ RPI_FWREQ_FRAMEBUFFER_TEST_TRANSFORM = 0x0004400d,
136
+ RPI_FWREQ_FRAMEBUFFER_TEST_VSYNC = 0x0004400e,
137
+ RPI_FWREQ_FRAMEBUFFER_SET_PHYSICAL_WIDTH_HEIGHT = 0x00048003,
138
+ RPI_FWREQ_FRAMEBUFFER_SET_VIRTUAL_WIDTH_HEIGHT = 0x00048004,
139
+ RPI_FWREQ_FRAMEBUFFER_SET_DEPTH = 0x00048005,
140
+ RPI_FWREQ_FRAMEBUFFER_SET_PIXEL_ORDER = 0x00048006,
141
+ RPI_FWREQ_FRAMEBUFFER_SET_ALPHA_MODE = 0x00048007,
142
+ RPI_FWREQ_FRAMEBUFFER_SET_PITCH = 0x00048008,
143
+ RPI_FWREQ_FRAMEBUFFER_SET_VIRTUAL_OFFSET = 0x00048009,
144
+ RPI_FWREQ_FRAMEBUFFER_SET_OVERSCAN = 0x0004800a,
145
+ RPI_FWREQ_FRAMEBUFFER_SET_PALETTE = 0x0004800b,
259
+
146
+
260
+target/arm/decode-neon-dp.inc.c: $(SRC_PATH)/target/arm/neon-dp.decode $(DECODETREE)
147
+ RPI_FWREQ_FRAMEBUFFER_SET_TOUCHBUF = 0x0004801f,
261
+    $(call quiet-command,\
148
+ RPI_FWREQ_FRAMEBUFFER_SET_GPIOVIRTBUF = 0x00048020,
262
+     $(PYTHON) $(DECODETREE) --static-decode disas_neon_dp -o $@ $<,\
149
+ RPI_FWREQ_FRAMEBUFFER_SET_VSYNC = 0x0004800e,
263
+     "GEN", $(TARGET_DIR)$@)
150
+ RPI_FWREQ_FRAMEBUFFER_SET_LAYER = 0x0004800c,
151
+ RPI_FWREQ_FRAMEBUFFER_SET_TRANSFORM = 0x0004800d,
152
+ RPI_FWREQ_FRAMEBUFFER_SET_BACKLIGHT = 0x0004800f,
264
+
153
+
265
+target/arm/decode-neon-ls.inc.c: $(SRC_PATH)/target/arm/neon-ls.decode $(DECODETREE)
154
+ RPI_FWREQ_VCHIQ_INIT = 0x00048010,
266
+    $(call quiet-command,\
267
+     $(PYTHON) $(DECODETREE) --static-decode disas_neon_ls -o $@ $<,\
268
+     "GEN", $(TARGET_DIR)$@)
269
+
155
+
270
target/arm/decode-vfp.inc.c: $(SRC_PATH)/target/arm/vfp.decode $(DECODETREE)
156
+ RPI_FWREQ_SET_PLANE = 0x00048015,
271
    $(call quiet-command,\
157
+ RPI_FWREQ_GET_DISPLAY_TIMING = 0x00040017,
272
     $(PYTHON) $(DECODETREE) --static-decode disas_vfp -o $@ $<,\
158
+ RPI_FWREQ_SET_TIMING = 0x00048017,
273
@@ -XXX,XX +XXX,XX @@ target/arm/decode-t16.inc.c: $(SRC_PATH)/target/arm/t16.decode $(DECODETREE)
159
+ RPI_FWREQ_GET_DISPLAY_CFG = 0x00040018,
274
     "GEN", $(TARGET_DIR)$@)
160
+ RPI_FWREQ_SET_DISPLAY_POWER = 0x00048019,
275
161
+ RPI_FWREQ_GET_COMMAND_LINE = 0x00050001,
276
target/arm/translate-sve.o: target/arm/decode-sve.inc.c
162
+ RPI_FWREQ_GET_DMA_CHANNELS = 0x00060001,
277
+target/arm/translate.o: target/arm/decode-neon-shared.inc.c
163
+};
278
+target/arm/translate.o: target/arm/decode-neon-dp.inc.c
164
+
279
+target/arm/translate.o: target/arm/decode-neon-ls.inc.c
165
+enum rpi_firmware_clk_id {
280
target/arm/translate.o: target/arm/decode-vfp.inc.c
166
+ RPI_FIRMWARE_EMMC_CLK_ID = 1,
281
target/arm/translate.o: target/arm/decode-vfp-uncond.inc.c
167
+ RPI_FIRMWARE_UART_CLK_ID,
282
target/arm/translate.o: target/arm/decode-a32.inc.c
168
+ RPI_FIRMWARE_ARM_CLK_ID,
169
+ RPI_FIRMWARE_CORE_CLK_ID,
170
+ RPI_FIRMWARE_V3D_CLK_ID,
171
+ RPI_FIRMWARE_H264_CLK_ID,
172
+ RPI_FIRMWARE_ISP_CLK_ID,
173
+ RPI_FIRMWARE_SDRAM_CLK_ID,
174
+ RPI_FIRMWARE_PIXEL_CLK_ID,
175
+ RPI_FIRMWARE_PWM_CLK_ID,
176
+ RPI_FIRMWARE_HEVC_CLK_ID,
177
+ RPI_FIRMWARE_EMMC2_CLK_ID,
178
+ RPI_FIRMWARE_M2MC_CLK_ID,
179
+ RPI_FIRMWARE_PIXEL_BVB_CLK_ID,
180
+ RPI_FIRMWARE_VEC_CLK_ID,
181
+ RPI_FIRMWARE_NUM_CLK_ID,
182
+};
183
+
184
+#endif /* INCLUDE_HW_MISC_RASPBERRYPI_FW_DEFS_H_ */
283
--
185
--
284
2.20.1
186
2.34.1
285
187
286
188
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
From: Sergey Kambalin <sergey.kambalin@auriga.com>
2
2
3
Embed the GEMs into the SoC type.
3
Replace magic property values by a proper definition,
4
4
removing redundant comments.
5
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
5
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
6
Signed-off-by: Sergey Kambalin <sergey.kambalin@auriga.com>
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20230612223456.33824-3-philmd@linaro.org
10
Message-id: 20200427181649.26851-6-edgar.iglesias@gmail.com
10
Message-Id: <20230531155258.8361-1-sergey.kambalin@auriga.com>
11
[PMD: Split from bigger patch: 2/4]
12
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
14
---
13
include/hw/arm/xlnx-versal.h | 3 ++-
15
hw/misc/bcm2835_property.c | 101 +++++++++++++++++++------------------
14
hw/arm/xlnx-versal.c | 15 ++++++++-------
16
1 file changed, 51 insertions(+), 50 deletions(-)
15
2 files changed, 10 insertions(+), 8 deletions(-)
17
16
18
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
17
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
18
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
19
--- a/include/hw/arm/xlnx-versal.h
20
--- a/hw/misc/bcm2835_property.c
20
+++ b/include/hw/arm/xlnx-versal.h
21
+++ b/hw/misc/bcm2835_property.c
21
@@ -XXX,XX +XXX,XX @@
22
@@ -XXX,XX +XXX,XX @@
22
#include "hw/arm/boot.h"
23
#include "migration/vmstate.h"
23
#include "hw/intc/arm_gicv3.h"
24
#include "hw/irq.h"
24
#include "hw/char/pl011.h"
25
#include "hw/misc/bcm2835_mbox_defs.h"
25
+#include "hw/net/cadence_gem.h"
26
+#include "hw/misc/raspberrypi-fw-defs.h"
26
27
#include "sysemu/dma.h"
27
#define TYPE_XLNX_VERSAL "xlnx-versal"
28
#include "qemu/log.h"
28
#define XLNX_VERSAL(obj) OBJECT_CHECK(Versal, (obj), TYPE_XLNX_VERSAL)
29
#include "qemu/module.h"
29
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
30
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
30
31
/* @(value + 8) : Request/response indicator */
31
struct {
32
resplen = 0;
32
PL011State uart[XLNX_VERSAL_NR_UARTS];
33
switch (tag) {
33
- SysBusDevice *gem[XLNX_VERSAL_NR_GEMS];
34
- case 0x00000000: /* End tag */
34
+ CadenceGEMState gem[XLNX_VERSAL_NR_GEMS];
35
+ case RPI_FWREQ_PROPERTY_END:
35
SysBusDevice *adma[XLNX_VERSAL_NR_ADMAS];
36
break;
36
} iou;
37
- case 0x00000001: /* Get firmware revision */
37
} lpd;
38
+ case RPI_FWREQ_GET_FIRMWARE_REVISION:
38
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
39
stl_le_phys(&s->dma_as, value + 12, 346337);
39
index XXXXXXX..XXXXXXX 100644
40
resplen = 4;
40
--- a/hw/arm/xlnx-versal.c
41
break;
41
+++ b/hw/arm/xlnx-versal.c
42
- case 0x00010001: /* Get board model */
42
@@ -XXX,XX +XXX,XX @@ static void versal_create_gems(Versal *s, qemu_irq *pic)
43
+ case RPI_FWREQ_GET_BOARD_MODEL:
43
DeviceState *dev;
44
qemu_log_mask(LOG_UNIMP,
44
MemoryRegion *mr;
45
"bcm2835_property: 0x%08x get board model NYI\n",
45
46
tag);
46
- dev = qdev_create(NULL, "cadence_gem");
47
resplen = 4;
47
- s->lpd.iou.gem[i] = SYS_BUS_DEVICE(dev);
48
break;
48
- object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
49
- case 0x00010002: /* Get board revision */
49
+ sysbus_init_child_obj(OBJECT(s), name,
50
+ case RPI_FWREQ_GET_BOARD_REVISION:
50
+ &s->lpd.iou.gem[i], sizeof(s->lpd.iou.gem[i]),
51
stl_le_phys(&s->dma_as, value + 12, s->board_rev);
51
+ TYPE_CADENCE_GEM);
52
resplen = 4;
52
+ dev = DEVICE(&s->lpd.iou.gem[i]);
53
break;
53
if (nd->used) {
54
- case 0x00010003: /* Get board MAC address */
54
qemu_check_nic_model(nd, "cadence_gem");
55
+ case RPI_FWREQ_GET_BOARD_MAC_ADDRESS:
55
qdev_set_nic_properties(dev, nd);
56
resplen = sizeof(s->macaddr.a);
56
}
57
dma_memory_write(&s->dma_as, value + 12, s->macaddr.a, resplen,
57
- object_property_set_int(OBJECT(s->lpd.iou.gem[i]),
58
MEMTXATTRS_UNSPECIFIED);
58
+ object_property_set_int(OBJECT(dev),
59
break;
59
2, "num-priority-queues",
60
- case 0x00010004: /* Get board serial */
60
&error_abort);
61
+ case RPI_FWREQ_GET_BOARD_SERIAL:
61
- object_property_set_link(OBJECT(s->lpd.iou.gem[i]),
62
qemu_log_mask(LOG_UNIMP,
62
+ object_property_set_link(OBJECT(dev),
63
"bcm2835_property: 0x%08x get board serial NYI\n",
63
OBJECT(&s->mr_ps), "dma",
64
tag);
64
&error_abort);
65
resplen = 8;
65
qdev_init_nofail(dev);
66
break;
66
67
- case 0x00010005: /* Get ARM memory */
67
- mr = sysbus_mmio_get_region(s->lpd.iou.gem[i], 0);
68
+ case RPI_FWREQ_GET_ARM_MEMORY:
68
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
69
/* base */
69
memory_region_add_subregion(&s->mr_ps, addrs[i], mr);
70
stl_le_phys(&s->dma_as, value + 12, 0);
70
71
/* size */
71
- sysbus_connect_irq(s->lpd.iou.gem[i], 0, pic[irqs[i]]);
72
stl_le_phys(&s->dma_as, value + 16, s->fbdev->vcram_base);
72
+ sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0, pic[irqs[i]]);
73
resplen = 8;
73
g_free(name);
74
break;
74
}
75
- case 0x00010006: /* Get VC memory */
75
}
76
+ case RPI_FWREQ_GET_VC_MEMORY:
77
/* base */
78
stl_le_phys(&s->dma_as, value + 12, s->fbdev->vcram_base);
79
/* size */
80
stl_le_phys(&s->dma_as, value + 16, s->fbdev->vcram_size);
81
resplen = 8;
82
break;
83
- case 0x00028001: /* Set power state */
84
+ case RPI_FWREQ_SET_POWER_STATE:
85
/* Assume that whatever device they asked for exists,
86
* and we'll just claim we set it to the desired state
87
*/
88
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
89
90
/* Clocks */
91
92
- case 0x00030001: /* Get clock state */
93
+ case RPI_FWREQ_GET_CLOCK_STATE:
94
stl_le_phys(&s->dma_as, value + 16, 0x1);
95
resplen = 8;
96
break;
97
98
- case 0x00038001: /* Set clock state */
99
+ case RPI_FWREQ_SET_CLOCK_STATE:
100
qemu_log_mask(LOG_UNIMP,
101
"bcm2835_property: 0x%08x set clock state NYI\n",
102
tag);
103
resplen = 8;
104
break;
105
106
- case 0x00030002: /* Get clock rate */
107
- case 0x00030004: /* Get max clock rate */
108
- case 0x00030007: /* Get min clock rate */
109
+ case RPI_FWREQ_GET_CLOCK_RATE:
110
+ case RPI_FWREQ_GET_MAX_CLOCK_RATE:
111
+ case RPI_FWREQ_GET_MIN_CLOCK_RATE:
112
switch (ldl_le_phys(&s->dma_as, value + 12)) {
113
- case 1: /* EMMC */
114
+ case RPI_FIRMWARE_EMMC_CLK_ID:
115
stl_le_phys(&s->dma_as, value + 16, 50000000);
116
break;
117
- case 2: /* UART */
118
+ case RPI_FIRMWARE_UART_CLK_ID:
119
stl_le_phys(&s->dma_as, value + 16, 3000000);
120
break;
121
default:
122
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
123
resplen = 8;
124
break;
125
126
- case 0x00038002: /* Set clock rate */
127
- case 0x00038004: /* Set max clock rate */
128
- case 0x00038007: /* Set min clock rate */
129
+ case RPI_FWREQ_SET_CLOCK_RATE:
130
+ case RPI_FWREQ_SET_MAX_CLOCK_RATE:
131
+ case RPI_FWREQ_SET_MIN_CLOCK_RATE:
132
qemu_log_mask(LOG_UNIMP,
133
"bcm2835_property: 0x%08x set clock rate NYI\n",
134
tag);
135
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
136
137
/* Temperature */
138
139
- case 0x00030006: /* Get temperature */
140
+ case RPI_FWREQ_GET_TEMPERATURE:
141
stl_le_phys(&s->dma_as, value + 16, 25000);
142
resplen = 8;
143
break;
144
145
- case 0x0003000A: /* Get max temperature */
146
+ case RPI_FWREQ_GET_MAX_TEMPERATURE:
147
stl_le_phys(&s->dma_as, value + 16, 99000);
148
resplen = 8;
149
break;
150
151
/* Frame buffer */
152
153
- case 0x00040001: /* Allocate buffer */
154
+ case RPI_FWREQ_FRAMEBUFFER_ALLOCATE:
155
stl_le_phys(&s->dma_as, value + 12, fbconfig.base);
156
stl_le_phys(&s->dma_as, value + 16,
157
bcm2835_fb_get_size(&fbconfig));
158
resplen = 8;
159
break;
160
- case 0x00048001: /* Release buffer */
161
+ case RPI_FWREQ_FRAMEBUFFER_RELEASE:
162
resplen = 0;
163
break;
164
- case 0x00040002: /* Blank screen */
165
+ case RPI_FWREQ_FRAMEBUFFER_BLANK:
166
resplen = 4;
167
break;
168
- case 0x00044003: /* Test physical display width/height */
169
- case 0x00044004: /* Test virtual display width/height */
170
+ case RPI_FWREQ_FRAMEBUFFER_TEST_PHYSICAL_WIDTH_HEIGHT:
171
+ case RPI_FWREQ_FRAMEBUFFER_TEST_VIRTUAL_WIDTH_HEIGHT:
172
resplen = 8;
173
break;
174
- case 0x00048003: /* Set physical display width/height */
175
+ case RPI_FWREQ_FRAMEBUFFER_SET_PHYSICAL_WIDTH_HEIGHT:
176
fbconfig.xres = ldl_le_phys(&s->dma_as, value + 12);
177
fbconfig.yres = ldl_le_phys(&s->dma_as, value + 16);
178
bcm2835_fb_validate_config(&fbconfig);
179
fbconfig_updated = true;
180
/* fall through */
181
- case 0x00040003: /* Get physical display width/height */
182
+ case RPI_FWREQ_FRAMEBUFFER_GET_PHYSICAL_WIDTH_HEIGHT:
183
stl_le_phys(&s->dma_as, value + 12, fbconfig.xres);
184
stl_le_phys(&s->dma_as, value + 16, fbconfig.yres);
185
resplen = 8;
186
break;
187
- case 0x00048004: /* Set virtual display width/height */
188
+ case RPI_FWREQ_FRAMEBUFFER_SET_VIRTUAL_WIDTH_HEIGHT:
189
fbconfig.xres_virtual = ldl_le_phys(&s->dma_as, value + 12);
190
fbconfig.yres_virtual = ldl_le_phys(&s->dma_as, value + 16);
191
bcm2835_fb_validate_config(&fbconfig);
192
fbconfig_updated = true;
193
/* fall through */
194
- case 0x00040004: /* Get virtual display width/height */
195
+ case RPI_FWREQ_FRAMEBUFFER_GET_VIRTUAL_WIDTH_HEIGHT:
196
stl_le_phys(&s->dma_as, value + 12, fbconfig.xres_virtual);
197
stl_le_phys(&s->dma_as, value + 16, fbconfig.yres_virtual);
198
resplen = 8;
199
break;
200
- case 0x00044005: /* Test depth */
201
+ case RPI_FWREQ_FRAMEBUFFER_TEST_DEPTH:
202
resplen = 4;
203
break;
204
- case 0x00048005: /* Set depth */
205
+ case RPI_FWREQ_FRAMEBUFFER_SET_DEPTH:
206
fbconfig.bpp = ldl_le_phys(&s->dma_as, value + 12);
207
bcm2835_fb_validate_config(&fbconfig);
208
fbconfig_updated = true;
209
/* fall through */
210
- case 0x00040005: /* Get depth */
211
+ case RPI_FWREQ_FRAMEBUFFER_GET_DEPTH:
212
stl_le_phys(&s->dma_as, value + 12, fbconfig.bpp);
213
resplen = 4;
214
break;
215
- case 0x00044006: /* Test pixel order */
216
+ case RPI_FWREQ_FRAMEBUFFER_TEST_PIXEL_ORDER:
217
resplen = 4;
218
break;
219
- case 0x00048006: /* Set pixel order */
220
+ case RPI_FWREQ_FRAMEBUFFER_SET_PIXEL_ORDER:
221
fbconfig.pixo = ldl_le_phys(&s->dma_as, value + 12);
222
bcm2835_fb_validate_config(&fbconfig);
223
fbconfig_updated = true;
224
/* fall through */
225
- case 0x00040006: /* Get pixel order */
226
+ case RPI_FWREQ_FRAMEBUFFER_GET_PIXEL_ORDER:
227
stl_le_phys(&s->dma_as, value + 12, fbconfig.pixo);
228
resplen = 4;
229
break;
230
- case 0x00044007: /* Test pixel alpha */
231
+ case RPI_FWREQ_FRAMEBUFFER_TEST_ALPHA_MODE:
232
resplen = 4;
233
break;
234
- case 0x00048007: /* Set alpha */
235
+ case RPI_FWREQ_FRAMEBUFFER_SET_ALPHA_MODE:
236
fbconfig.alpha = ldl_le_phys(&s->dma_as, value + 12);
237
bcm2835_fb_validate_config(&fbconfig);
238
fbconfig_updated = true;
239
/* fall through */
240
- case 0x00040007: /* Get alpha */
241
+ case RPI_FWREQ_FRAMEBUFFER_GET_ALPHA_MODE:
242
stl_le_phys(&s->dma_as, value + 12, fbconfig.alpha);
243
resplen = 4;
244
break;
245
- case 0x00040008: /* Get pitch */
246
+ case RPI_FWREQ_FRAMEBUFFER_GET_PITCH:
247
stl_le_phys(&s->dma_as, value + 12,
248
bcm2835_fb_get_pitch(&fbconfig));
249
resplen = 4;
250
break;
251
- case 0x00044009: /* Test virtual offset */
252
+ case RPI_FWREQ_FRAMEBUFFER_TEST_VIRTUAL_OFFSET:
253
resplen = 8;
254
break;
255
- case 0x00048009: /* Set virtual offset */
256
+ case RPI_FWREQ_FRAMEBUFFER_SET_VIRTUAL_OFFSET:
257
fbconfig.xoffset = ldl_le_phys(&s->dma_as, value + 12);
258
fbconfig.yoffset = ldl_le_phys(&s->dma_as, value + 16);
259
bcm2835_fb_validate_config(&fbconfig);
260
fbconfig_updated = true;
261
/* fall through */
262
- case 0x00040009: /* Get virtual offset */
263
+ case RPI_FWREQ_FRAMEBUFFER_GET_VIRTUAL_OFFSET:
264
stl_le_phys(&s->dma_as, value + 12, fbconfig.xoffset);
265
stl_le_phys(&s->dma_as, value + 16, fbconfig.yoffset);
266
resplen = 8;
267
break;
268
- case 0x0004000a: /* Get/Test/Set overscan */
269
- case 0x0004400a:
270
- case 0x0004800a:
271
+ case RPI_FWREQ_FRAMEBUFFER_GET_OVERSCAN:
272
+ case RPI_FWREQ_FRAMEBUFFER_TEST_OVERSCAN:
273
+ case RPI_FWREQ_FRAMEBUFFER_SET_OVERSCAN:
274
stl_le_phys(&s->dma_as, value + 12, 0);
275
stl_le_phys(&s->dma_as, value + 16, 0);
276
stl_le_phys(&s->dma_as, value + 20, 0);
277
stl_le_phys(&s->dma_as, value + 24, 0);
278
resplen = 16;
279
break;
280
- case 0x0004800b: /* Set palette */
281
+ case RPI_FWREQ_FRAMEBUFFER_SET_PALETTE:
282
offset = ldl_le_phys(&s->dma_as, value + 12);
283
length = ldl_le_phys(&s->dma_as, value + 16);
284
n = 0;
285
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
286
stl_le_phys(&s->dma_as, value + 12, 0);
287
resplen = 4;
288
break;
289
- case 0x00040013: /* Get number of displays */
290
+ case RPI_FWREQ_FRAMEBUFFER_GET_NUM_DISPLAYS:
291
stl_le_phys(&s->dma_as, value + 12, 1);
292
resplen = 4;
293
break;
294
295
- case 0x00060001: /* Get DMA channels */
296
+ case RPI_FWREQ_GET_DMA_CHANNELS:
297
/* channels 2-5 */
298
stl_le_phys(&s->dma_as, value + 12, 0x003C);
299
resplen = 4;
300
break;
301
302
- case 0x00050001: /* Get command line */
303
+ case RPI_FWREQ_GET_COMMAND_LINE:
304
/*
305
* We follow the firmware behaviour: no NUL terminator is
306
* written to the buffer, and if the buffer is too short
76
--
307
--
77
2.20.1
308
2.34.1
78
309
79
310
diff view generated by jsdifflib
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
1
From: Sergey Kambalin <sergey.kambalin@auriga.com>
2
2
3
Embed the ADMAs into the SoC type.
3
Signed-off-by: Sergey Kambalin <sergey.kambalin@auriga.com>
4
4
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
5
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
6
Message-id: 20230612223456.33824-4-philmd@linaro.org
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Message-Id: <20230531155258.8361-1-sergey.kambalin@auriga.com>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
[PMD: Split from bigger patch: 4/4]
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
10
Message-id: 20200427181649.26851-7-edgar.iglesias@gmail.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
11
---
13
include/hw/arm/xlnx-versal.h | 3 ++-
12
include/hw/arm/raspi_platform.h | 5 +++++
14
hw/arm/xlnx-versal.c | 14 +++++++-------
13
hw/misc/bcm2835_property.c | 8 +++++---
15
2 files changed, 9 insertions(+), 8 deletions(-)
14
2 files changed, 10 insertions(+), 3 deletions(-)
16
15
17
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
16
diff --git a/include/hw/arm/raspi_platform.h b/include/hw/arm/raspi_platform.h
18
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
19
--- a/include/hw/arm/xlnx-versal.h
18
--- a/include/hw/arm/raspi_platform.h
20
+++ b/include/hw/arm/xlnx-versal.h
19
+++ b/include/hw/arm/raspi_platform.h
21
@@ -XXX,XX +XXX,XX @@
20
@@ -XXX,XX +XXX,XX @@
22
#include "hw/arm/boot.h"
21
#define INTERRUPT_ILLEGAL_TYPE0 6
23
#include "hw/intc/arm_gicv3.h"
22
#define INTERRUPT_ILLEGAL_TYPE1 7
24
#include "hw/char/pl011.h"
23
25
+#include "hw/dma/xlnx-zdma.h"
24
+/* Clock rates */
26
#include "hw/net/cadence_gem.h"
25
+#define RPI_FIRMWARE_EMMC_CLK_RATE 50000000
27
26
+#define RPI_FIRMWARE_UART_CLK_RATE 3000000
28
#define TYPE_XLNX_VERSAL "xlnx-versal"
27
+#define RPI_FIRMWARE_DEFAULT_CLK_RATE 700000000
29
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
28
+
30
struct {
29
#endif
31
PL011State uart[XLNX_VERSAL_NR_UARTS];
30
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
32
CadenceGEMState gem[XLNX_VERSAL_NR_GEMS];
33
- SysBusDevice *adma[XLNX_VERSAL_NR_ADMAS];
34
+ XlnxZDMA adma[XLNX_VERSAL_NR_ADMAS];
35
} iou;
36
} lpd;
37
38
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
39
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
40
--- a/hw/arm/xlnx-versal.c
32
--- a/hw/misc/bcm2835_property.c
41
+++ b/hw/arm/xlnx-versal.c
33
+++ b/hw/misc/bcm2835_property.c
42
@@ -XXX,XX +XXX,XX @@ static void versal_create_admas(Versal *s, qemu_irq *pic)
34
@@ -XXX,XX +XXX,XX @@
43
DeviceState *dev;
35
#include "qemu/log.h"
44
MemoryRegion *mr;
36
#include "qemu/module.h"
45
37
#include "trace.h"
46
- dev = qdev_create(NULL, "xlnx.zdma");
38
+#include "hw/arm/raspi_platform.h"
47
- s->lpd.iou.adma[i] = SYS_BUS_DEVICE(dev);
39
48
- object_property_set_int(OBJECT(s->lpd.iou.adma[i]), 128, "bus-width",
40
/* https://github.com/raspberrypi/firmware/wiki/Mailbox-property-interface */
49
- &error_abort);
41
50
- object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
42
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
51
+ sysbus_init_child_obj(OBJECT(s), name,
43
case RPI_FWREQ_GET_MIN_CLOCK_RATE:
52
+ &s->lpd.iou.adma[i], sizeof(s->lpd.iou.adma[i]),
44
switch (ldl_le_phys(&s->dma_as, value + 12)) {
53
+ TYPE_XLNX_ZDMA);
45
case RPI_FIRMWARE_EMMC_CLK_ID:
54
+ dev = DEVICE(&s->lpd.iou.adma[i]);
46
- stl_le_phys(&s->dma_as, value + 16, 50000000);
55
+ object_property_set_int(OBJECT(dev), 128, "bus-width", &error_abort);
47
+ stl_le_phys(&s->dma_as, value + 16, RPI_FIRMWARE_EMMC_CLK_RATE);
56
qdev_init_nofail(dev);
48
break;
57
49
case RPI_FIRMWARE_UART_CLK_ID:
58
- mr = sysbus_mmio_get_region(s->lpd.iou.adma[i], 0);
50
- stl_le_phys(&s->dma_as, value + 16, 3000000);
59
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
51
+ stl_le_phys(&s->dma_as, value + 16, RPI_FIRMWARE_UART_CLK_RATE);
60
memory_region_add_subregion(&s->mr_ps,
52
break;
61
MM_ADMA_CH0 + i * MM_ADMA_CH0_SIZE, mr);
53
default:
62
54
- stl_le_phys(&s->dma_as, value + 16, 700000000);
63
- sysbus_connect_irq(s->lpd.iou.adma[i], 0, pic[VERSAL_ADMA_IRQ_0 + i]);
55
+ stl_le_phys(&s->dma_as, value + 16,
64
+ sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0, pic[VERSAL_ADMA_IRQ_0 + i]);
56
+ RPI_FIRMWARE_DEFAULT_CLK_RATE);
65
g_free(name);
57
break;
66
}
58
}
67
}
59
resplen = 8;
68
--
60
--
69
2.20.1
61
2.34.1
70
62
71
63
diff view generated by jsdifflib
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
1
From: Sergey Kambalin <sergey.kambalin@auriga.com>
2
2
3
MIDR_EL1 is a 64-bit system register with the top 32-bit being RES0.
3
Signed-off-by: Sergey Kambalin <sergey.kambalin@auriga.com>
4
Represent it in QEMU's ARMCPU struct with a uint64_t, not a
4
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
5
uint32_t.
5
Message-id: 20230612223456.33824-5-philmd@linaro.org
6
6
Message-Id: <20230531155258.8361-1-sergey.kambalin@auriga.com>
7
This fixes an error when compiling with -Werror=conversion
7
[PMD: Split from bigger patch: 3/4]
8
because we were manipulating the register value using a
8
Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
9
local uint64_t variable:
9
[PMM: added a comment about RPI_FIRMWARE_CORE_CLK_RATE
10
10
really being SoC-specific]
11
target/arm/cpu64.c: In function ‘aarch64_max_initfn’:
12
target/arm/cpu64.c:628:21: error: conversion from ‘uint64_t’ {aka ‘long unsigned int’} to ‘uint32_t’ {aka ‘unsigned int’} may change value [-Werror=conversion]
13
628 | cpu->midr = t;
14
| ^
15
16
and future-proofs us against a possible future architecture
17
change using some of the top 32 bits.
18
19
Suggested-by: Laurent Desnogues <laurent.desnogues@gmail.com>
20
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
21
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
22
Reviewed-by: Laurent Desnogues <laurent.desnogues@gmail.com>
23
Message-id: 20200428172634.29707-1-f4bug@amsat.org
24
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
25
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
26
---
13
---
27
target/arm/cpu.h | 2 +-
14
include/hw/arm/raspi_platform.h | 5 +++++
28
target/arm/cpu.c | 2 +-
15
hw/misc/bcm2835_property.c | 3 +++
29
2 files changed, 2 insertions(+), 2 deletions(-)
16
2 files changed, 8 insertions(+)
30
17
31
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
diff --git a/include/hw/arm/raspi_platform.h b/include/hw/arm/raspi_platform.h
32
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
33
--- a/target/arm/cpu.h
20
--- a/include/hw/arm/raspi_platform.h
34
+++ b/target/arm/cpu.h
21
+++ b/include/hw/arm/raspi_platform.h
35
@@ -XXX,XX +XXX,XX @@ struct ARMCPU {
22
@@ -XXX,XX +XXX,XX @@
36
uint64_t id_aa64dfr0;
23
/* Clock rates */
37
uint64_t id_aa64dfr1;
24
#define RPI_FIRMWARE_EMMC_CLK_RATE 50000000
38
} isar;
25
#define RPI_FIRMWARE_UART_CLK_RATE 3000000
39
- uint32_t midr;
26
+/*
40
+ uint64_t midr;
27
+ * TODO: this is really SoC-specific; we might want to
41
uint32_t revidr;
28
+ * set it per-SoC if it turns out any guests care.
42
uint32_t reset_fpsid;
29
+ */
43
uint32_t ctr;
30
+#define RPI_FIRMWARE_CORE_CLK_RATE 350000000
44
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
31
#define RPI_FIRMWARE_DEFAULT_CLK_RATE 700000000
32
33
#endif
34
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
45
index XXXXXXX..XXXXXXX 100644
35
index XXXXXXX..XXXXXXX 100644
46
--- a/target/arm/cpu.c
36
--- a/hw/misc/bcm2835_property.c
47
+++ b/target/arm/cpu.c
37
+++ b/hw/misc/bcm2835_property.c
48
@@ -XXX,XX +XXX,XX @@ static const ARMCPUInfo arm_cpus[] = {
38
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
49
static Property arm_cpu_properties[] = {
39
case RPI_FIRMWARE_UART_CLK_ID:
50
DEFINE_PROP_BOOL("start-powered-off", ARMCPU, start_powered_off, false),
40
stl_le_phys(&s->dma_as, value + 16, RPI_FIRMWARE_UART_CLK_RATE);
51
DEFINE_PROP_UINT32("psci-conduit", ARMCPU, psci_conduit, 0),
41
break;
52
- DEFINE_PROP_UINT32("midr", ARMCPU, midr, 0),
42
+ case RPI_FIRMWARE_CORE_CLK_ID:
53
+ DEFINE_PROP_UINT64("midr", ARMCPU, midr, 0),
43
+ stl_le_phys(&s->dma_as, value + 16, RPI_FIRMWARE_CORE_CLK_RATE);
54
DEFINE_PROP_UINT64("mp-affinity", ARMCPU,
44
+ break;
55
mp_affinity, ARM64_AFFINITY_INVALID),
45
default:
56
DEFINE_PROP_INT32("node-id", ARMCPU, node_id, CPU_UNSET_NUMA_NODE_ID),
46
stl_le_phys(&s->dma_as, value + 16,
47
RPI_FIRMWARE_DEFAULT_CLK_RATE);
57
--
48
--
58
2.20.1
49
2.34.1
59
50
60
51
diff view generated by jsdifflib
Deleted patch
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
1
3
Move misplaced comment.
4
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20200427181649.26851-3-edgar.iglesias@gmail.com
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
hw/arm/xlnx-versal.c | 2 +-
13
1 file changed, 1 insertion(+), 1 deletion(-)
14
15
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/xlnx-versal.c
18
+++ b/hw/arm/xlnx-versal.c
19
@@ -XXX,XX +XXX,XX @@ static void versal_create_apu_cpus(Versal *s)
20
21
obj = object_new(XLNX_VERSAL_ACPU_TYPE);
22
if (!obj) {
23
- /* Secondary CPUs start in PSCI powered-down state */
24
error_report("Unable to create apu.cpu[%d] of type %s",
25
i, XLNX_VERSAL_ACPU_TYPE);
26
exit(EXIT_FAILURE);
27
@@ -XXX,XX +XXX,XX @@ static void versal_create_apu_cpus(Versal *s)
28
object_property_set_int(obj, s->cfg.psci_conduit,
29
"psci-conduit", &error_abort);
30
if (i) {
31
+ /* Secondary CPUs start in PSCI powered-down state */
32
object_property_set_bool(obj, true,
33
"start-powered-off", &error_abort);
34
}
35
--
36
2.20.1
37
38
diff view generated by jsdifflib
Deleted patch
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
1
3
Embed the UARTs into the SoC type.
4
5
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
10
Message-id: 20200427181649.26851-5-edgar.iglesias@gmail.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
13
include/hw/arm/xlnx-versal.h | 3 ++-
14
hw/arm/xlnx-versal.c | 12 ++++++------
15
2 files changed, 8 insertions(+), 7 deletions(-)
16
17
diff --git a/include/hw/arm/xlnx-versal.h b/include/hw/arm/xlnx-versal.h
18
index XXXXXXX..XXXXXXX 100644
19
--- a/include/hw/arm/xlnx-versal.h
20
+++ b/include/hw/arm/xlnx-versal.h
21
@@ -XXX,XX +XXX,XX @@
22
#include "hw/sysbus.h"
23
#include "hw/arm/boot.h"
24
#include "hw/intc/arm_gicv3.h"
25
+#include "hw/char/pl011.h"
26
27
#define TYPE_XLNX_VERSAL "xlnx-versal"
28
#define XLNX_VERSAL(obj) OBJECT_CHECK(Versal, (obj), TYPE_XLNX_VERSAL)
29
@@ -XXX,XX +XXX,XX @@ typedef struct Versal {
30
MemoryRegion mr_ocm;
31
32
struct {
33
- SysBusDevice *uart[XLNX_VERSAL_NR_UARTS];
34
+ PL011State uart[XLNX_VERSAL_NR_UARTS];
35
SysBusDevice *gem[XLNX_VERSAL_NR_GEMS];
36
SysBusDevice *adma[XLNX_VERSAL_NR_ADMAS];
37
} iou;
38
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
39
index XXXXXXX..XXXXXXX 100644
40
--- a/hw/arm/xlnx-versal.c
41
+++ b/hw/arm/xlnx-versal.c
42
@@ -XXX,XX +XXX,XX @@
43
#include "kvm_arm.h"
44
#include "hw/misc/unimp.h"
45
#include "hw/arm/xlnx-versal.h"
46
-#include "hw/char/pl011.h"
47
48
#define XLNX_VERSAL_ACPU_TYPE ARM_CPU_TYPE_NAME("cortex-a72")
49
#define GEM_REVISION 0x40070106
50
@@ -XXX,XX +XXX,XX @@ static void versal_create_uarts(Versal *s, qemu_irq *pic)
51
DeviceState *dev;
52
MemoryRegion *mr;
53
54
- dev = qdev_create(NULL, TYPE_PL011);
55
- s->lpd.iou.uart[i] = SYS_BUS_DEVICE(dev);
56
+ sysbus_init_child_obj(OBJECT(s), name,
57
+ &s->lpd.iou.uart[i], sizeof(s->lpd.iou.uart[i]),
58
+ TYPE_PL011);
59
+ dev = DEVICE(&s->lpd.iou.uart[i]);
60
qdev_prop_set_chr(dev, "chardev", serial_hd(i));
61
- object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
62
qdev_init_nofail(dev);
63
64
- mr = sysbus_mmio_get_region(s->lpd.iou.uart[i], 0);
65
+ mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(dev), 0);
66
memory_region_add_subregion(&s->mr_ps, addrs[i], mr);
67
68
- sysbus_connect_irq(s->lpd.iou.uart[i], 0, pic[irqs[i]]);
69
+ sysbus_connect_irq(SYS_BUS_DEVICE(dev), 0, pic[irqs[i]]);
70
g_free(name);
71
}
72
}
73
--
74
2.20.1
75
76
diff view generated by jsdifflib
Deleted patch
1
We were accidentally permitting decode of Thumb Neon insns even if
2
the CPU didn't have the FEATURE_NEON bit set, because the feature
3
check was being done before the call to disas_neon_data_insn() and
4
disas_neon_ls_insn() in the Arm decoder but was omitted from the
5
Thumb decoder. Push the feature bit check down into the called
6
functions so it is done for both Arm and Thumb encodings.
7
1
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
11
Message-id: 20200430181003.21682-3-peter.maydell@linaro.org
12
---
13
target/arm/translate.c | 16 ++++++++--------
14
1 file changed, 8 insertions(+), 8 deletions(-)
15
16
diff --git a/target/arm/translate.c b/target/arm/translate.c
17
index XXXXXXX..XXXXXXX 100644
18
--- a/target/arm/translate.c
19
+++ b/target/arm/translate.c
20
@@ -XXX,XX +XXX,XX @@ static int disas_neon_ls_insn(DisasContext *s, uint32_t insn)
21
TCGv_i32 tmp2;
22
TCGv_i64 tmp64;
23
24
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
25
+ return 1;
26
+ }
27
+
28
/* FIXME: this access check should not take precedence over UNDEF
29
* for invalid encodings; we will generate incorrect syndrome information
30
* for attempts to execute invalid vfp/neon encodings with FP disabled.
31
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
32
TCGv_ptr ptr1, ptr2, ptr3;
33
TCGv_i64 tmp64;
34
35
+ if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
36
+ return 1;
37
+ }
38
+
39
/* FIXME: this access check should not take precedence over UNDEF
40
* for invalid encodings; we will generate incorrect syndrome information
41
* for attempts to execute invalid vfp/neon encodings with FP disabled.
42
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
43
44
if (((insn >> 25) & 7) == 1) {
45
/* NEON Data processing. */
46
- if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
47
- goto illegal_op;
48
- }
49
-
50
if (disas_neon_data_insn(s, insn)) {
51
goto illegal_op;
52
}
53
@@ -XXX,XX +XXX,XX @@ static void disas_arm_insn(DisasContext *s, unsigned int insn)
54
}
55
if ((insn & 0x0f100000) == 0x04000000) {
56
/* NEON load/store. */
57
- if (!arm_dc_feature(s, ARM_FEATURE_NEON)) {
58
- goto illegal_op;
59
- }
60
-
61
if (disas_neon_ls_insn(s, insn)) {
62
goto illegal_op;
63
}
64
--
65
2.20.1
66
67
diff view generated by jsdifflib