1
target-arm queue: this clears out a bunch of patches I'd sent over
1
Mostly this is RTH's memtag series, but there are also some cleanups
2
the last coupled of weeks that have now got reviewed. Mostly
2
from Philippe.
3
this is MPS2 device support improvements, put there is also
4
more of the incremental work towards supporting AArch32 Hyp mode,
5
a floating point bugfix, and the raspi framebuffer viewport support.
6
3
7
thanks
4
thanks
8
-- PMM
5
-- PMM
9
6
10
The following changes since commit 5ccac548faf041ff5229a8e8342e3be14a34c8af:
7
The following changes since commit 10f7ffabf9c507fc02382b89912003b1c43c3231:
11
8
12
Merge remote-tracking branch 'remotes/cody/tags/block-pull-request' into staging (2018-08-23 17:35:48 +0100)
9
Merge remote-tracking branch 'remotes/mcayland/tags/qemu-macppc-20200626' into staging (2020-06-26 12:14:18 +0100)
13
10
14
are available in the Git repository at:
11
are available in the Git repository at:
15
12
16
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20180824
13
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20200626
17
14
18
for you to fetch changes up to 30a719e3cb5c5367f3651eba8fa935634bfee286:
15
for you to fetch changes up to c7459633baa71d1781fde4a245d6ec9ce2f008cf:
19
16
20
hw/arm/mps2: Fix ID register errors on AN511 and AN385 (2018-08-24 10:22:44 +0100)
17
target/arm: Enable MTE (2020-06-26 14:32:24 +0100)
21
18
22
----------------------------------------------------------------
19
----------------------------------------------------------------
23
target-arm queue:
20
target-arm queue:
24
* Fix rounding errors in scaling float-to-int and int-to-float operations
21
* hw/arm/aspeed: improve QOM usage
25
* Connect virtualization-related IRQs and memory regions of GICv2
22
* hw/misc/pca9552: trace GPIO change events
26
in boards that use Cortex-A7 or Cortex-A15
23
* target/arm: Implement ARMv8.5-MemTag for system emulation
27
* Support taking exceptions to AArch32 Hyp mode
28
* Clear CPSR.IL and CPSR.J on 32-bit exception entry
29
(a minor bug fix that won't affect non-buggy guest code)
30
* mps2-an505: Implement various missing devices:
31
dual timer, watchdogs, counters in the FPGAIO registers,
32
some missing ID/control registers, TrustZone Master Security
33
Controllers, PL081 DMA controllers, PL022 SPI controllers
34
* correct ID register values for mps2-an385, -an511, -an505
35
* fix some hardcoded tabs in untouched backwaters of the
36
target/arm codebase
37
* raspi: Refactor framebuffer property handling code and implement
38
support for the virtual framebuffer/viewport
39
24
40
----------------------------------------------------------------
25
----------------------------------------------------------------
41
Peter Maydell (48):
26
Philippe Mathieu-Daudé (12):
42
hw/intc/arm_gic: Make per-cpu GICH memory regions 0x200 bytes large
27
hw/arm/aspeed: Remove extraneous MemoryRegion object owner
43
hw/arm/vexpress: Connect VIRQ and VFIQ
28
hw/arm/aspeed: Rename AspeedBoardState as AspeedMachineState
44
hw/arm/highbank: Connect VIRQ and VFIQ
29
hw/arm/aspeed: QOM'ify AspeedMachineState
45
hw/arm/fsl-imx6ul: Connect VIRQ and VFIQ
30
hw/i2c/core: Add i2c_try_create_slave() and i2c_realize_and_unref()
46
hw/arm/fsl-imx6ul: Connect VIRQ and VFIQ
31
hw/misc/pca9552: Rename 'nr_leds' as 'pin_count'
47
hw/cpu/a15mpcore: If CPU has EL2, enable it on the GIC and wire it up
32
hw/misc/pca9552: Rename generic code as pca955x
48
hw/arm/vexpress: Don't set info->secure_boot if CPU doesn't have EL3
33
hw/misc/pca9552: Add generic PCA955xClass, parent of TYPE_PCA9552
49
hw/arm/vexpress: Add "virtualization" property controlling presence of EL2
34
hw/misc/pca9552: Add a 'description' property for debugging purpose
50
target/arm: Implement RAZ/WI HACTLR2
35
hw/misc/pca9552: Trace GPIO High/Low events
51
target/arm: Implement AArch32 HCR and HCR2
36
hw/arm/aspeed: Describe each PCA9552 device
52
target/arm: Factor out code for taking an AArch32 exception
37
hw/misc/pca9552: Trace GPIO change events
53
target/arm: Implement support for taking exceptions to Hyp mode
38
hw/misc/pca9552: Model qdev output GPIOs
54
target/arm: Clear CPSR.IL and CPSR.J on 32-bit exception entry
55
hw/arm/boot: AArch32 kernels should be started in Hyp mode if available
56
hw/misc/mps2-fpgaio: Implement 1Hz and 100Hz counters
57
hw/misc/mps2-fpgaio: Implement PSCNTR and COUNTER
58
hw/timer/cmsdk-apb-dualtimer: Implement CMSDK dual timer module
59
hw/arm/iotkit: Wire up the dualtimer
60
hw/arm/mps2: Wire up dual-timer in mps2-an385 and mps2-an511
61
hw/arm/iotkit: Wire up the watchdogs
62
hw/arm/iotkit: Wire up the S32KTIMER
63
hw/misc/iotkit-sysctl: Implement IoTKit system control element
64
hw/misc/iotkit-sysinfo: Implement IoTKit system information block
65
hw/misc/iotkit: Wire up the sysctl and sysinfo register blocks
66
hw/misc/tz-msc: Model TrustZone Master Security Controller
67
hw/misc/iotkit-secctl: Wire up registers for controlling MSCs
68
hw/arm/iotkit: Wire up the lines for MSCs
69
hw/arm/mps2-tz: Create PL081s and MSCs
70
hw/ssi/pl022: Allow use as embedded-struct device
71
hw/ssi/pl022: Set up reset function in class init
72
hw/ssi/pl022: Don't directly call vmstate_register()
73
hw/ssi/pl022: Use DeviceState::realize rather than SysBusDevice::init
74
hw/ssi/pl022: Correct wrong value for PL022_INT_RT
75
hw/ssi/pl022: Correct wrong DMACR and ICR handling
76
hw/arm/mps2-tz: Instantiate SPI controllers
77
hw/arm/mps2-tz: Fix MPS2 SCC config register values
78
target/arm: Untabify translate.c
79
target/arm: Untabify iwmmxt_helper.c
80
target/arm: Remove a handful of stray tabs
81
hw/misc/bcm2835_fb: Move config fields to their own struct
82
hw/misc/bcm2835_property: Track fb settings using BCM2835FBConfig
83
hw/display/bcm2835_fb: Drop unused size and pitch fields
84
hw/display/bcm2835_fb: Reset resolution, etc correctly
85
hw/display/bcm2835_fb: Abstract out calculation of pitch, size
86
hw/display/bcm2835_fb: Fix handling of virtual framebuffer
87
hw/display/bcm2835_fb: Validate config settings
88
hw/display/bcm2835_fb: Validate bcm2835_fb_mbox_push() config
89
hw/arm/mps2: Fix ID register errors on AN511 and AN385
90
39
91
Richard Henderson (4):
40
Richard Henderson (45):
92
softfloat: Add scaling int-to-float routines
41
target/arm: Add isar tests for mte
93
softfloat: Add scaling float-to-int routines
42
target/arm: Improve masking of SCR RES0 bits
94
target/arm: Use the int-to-float-scale softfloat routines
43
target/arm: Add support for MTE to SCTLR_ELx
95
target/arm: Use the float-to-int-scale softfloat routines
44
target/arm: Add support for MTE to HCR_EL2 and SCR_EL3
45
target/arm: Rename DISAS_UPDATE to DISAS_UPDATE_EXIT
46
target/arm: Add DISAS_UPDATE_NOCHAIN
47
target/arm: Add MTE system registers
48
target/arm: Add MTE bits to tb_flags
49
target/arm: Implement the IRG instruction
50
target/arm: Revise decoding for disas_add_sub_imm
51
target/arm: Implement the ADDG, SUBG instructions
52
target/arm: Implement the GMI instruction
53
target/arm: Implement the SUBP instruction
54
target/arm: Define arm_cpu_do_unaligned_access for user-only
55
target/arm: Implement LDG, STG, ST2G instructions
56
target/arm: Implement the STGP instruction
57
target/arm: Restrict the values of DCZID.BS under TCG
58
target/arm: Simplify DC_ZVA
59
target/arm: Implement the LDGM, STGM, STZGM instructions
60
target/arm: Implement the access tag cache flushes
61
target/arm: Move regime_el to internals.h
62
target/arm: Move regime_tcr to internals.h
63
target/arm: Add gen_mte_check1
64
target/arm: Add gen_mte_checkN
65
target/arm: Implement helper_mte_check1
66
target/arm: Implement helper_mte_checkN
67
target/arm: Add helper_mte_check_zva
68
target/arm: Use mte_checkN for sve unpredicated loads
69
target/arm: Use mte_checkN for sve unpredicated stores
70
target/arm: Use mte_check1 for sve LD1R
71
target/arm: Tidy trans_LD1R_zpri
72
target/arm: Add arm_tlb_bti_gp
73
target/arm: Add mte helpers for sve scalar + int loads
74
target/arm: Add mte helpers for sve scalar + int stores
75
target/arm: Add mte helpers for sve scalar + int ff/nf loads
76
target/arm: Handle TBI for sve scalar + int memory ops
77
target/arm: Add mte helpers for sve scatter/gather memory ops
78
target/arm: Complete TBI clearing for user-only for SVE
79
target/arm: Implement data cache set allocation tags
80
target/arm: Set PSTATE.TCO on exception entry
81
target/arm: Always pass cacheattr to get_phys_addr
82
target/arm: Cache the Tagged bit for a page in MemTxAttrs
83
target/arm: Create tagged ram when MTE is enabled
84
target/arm: Add allocation tag storage for system mode
85
target/arm: Enable MTE
96
86
97
hw/misc/Makefile.objs | 3 +
87
include/hw/arm/aspeed.h | 12 +-
98
hw/timer/Makefile.objs | 1 +
88
include/hw/i2c/i2c.h | 2 +
99
include/fpu/softfloat.h | 169 +++++++---
89
include/hw/misc/pca9552.h | 16 +-
100
include/hw/arm/iotkit.h | 25 +-
90
target/arm/cpu.h | 50 +-
101
include/hw/display/bcm2835_fb.h | 59 +++-
91
target/arm/helper-a64.h | 16 +
102
include/hw/misc/iotkit-secctl.h | 14 +
92
target/arm/helper-sve.h | 488 ++++++++++++++
103
include/hw/misc/iotkit-sysctl.h | 49 +++
93
target/arm/helper.h | 2 +
104
include/hw/misc/iotkit-sysinfo.h | 37 +++
94
target/arm/internals.h | 153 ++++-
105
include/hw/misc/mps2-fpgaio.h | 10 +
95
target/arm/translate-a64.h | 5 +
106
include/hw/misc/tz-msc.h | 79 +++++
96
target/arm/translate.h | 23 +-
107
include/hw/ssi/pl022.h | 51 +++
97
hw/arm/aspeed.c | 46 +-
108
include/hw/timer/cmsdk-apb-dualtimer.h | 72 ++++
98
hw/arm/virt.c | 55 +-
109
target/arm/cpu.h | 16 +-
99
hw/i2c/core.c | 18 +-
110
fpu/softfloat.c | 579 ++++++++++++++++++++++++++-------
100
hw/misc/pca9552.c | 216 +++++--
111
hw/arm/boot.c | 11 +
101
target/arm/cpu.c | 81 ++-
112
hw/arm/fsl-imx6ul.c | 4 +
102
target/arm/cpu64.c | 5 +
113
hw/arm/fsl-imx7.c | 4 +
103
target/arm/helper-a64.c | 94 +--
114
hw/arm/highbank.c | 6 +
104
target/arm/helper.c | 423 ++++++++++---
115
hw/arm/iotkit.c | 114 ++++++-
105
target/arm/m_helper.c | 11 +-
116
hw/arm/mps2-tz.c | 142 +++++++-
106
target/arm/mte_helper.c | 906 ++++++++++++++++++++++++++
117
hw/arm/mps2.c | 17 +-
107
target/arm/op_helper.c | 16 +
118
hw/arm/vexpress.c | 64 +++-
108
target/arm/sve_helper.c | 616 ++++++++++++++----
119
hw/cpu/a15mpcore.c | 31 +-
109
target/arm/tlb_helper.c | 13 +-
120
hw/display/bcm2835_fb.c | 218 ++++++++-----
110
target/arm/translate-a64.c | 657 ++++++++++++++++---
121
hw/intc/arm_gic.c | 2 +-
111
target/arm/translate-sve.c | 1366 ++++++++++++++++++++++++++--------------
122
hw/misc/bcm2835_property.c | 123 ++++---
112
target/arm/translate-vfp.inc.c | 4 +-
123
hw/misc/iotkit-secctl.c | 73 ++++-
113
target/arm/translate.c | 16 +-
124
hw/misc/iotkit-sysctl.c | 261 +++++++++++++++
114
hw/misc/trace-events | 4 +
125
hw/misc/iotkit-sysinfo.c | 128 ++++++++
115
target/arm/Makefile.objs | 1 +
126
hw/misc/mps2-fpgaio.c | 146 ++++++++-
116
29 files changed, 4391 insertions(+), 924 deletions(-)
127
hw/misc/tz-msc.c | 308 ++++++++++++++++++
117
create mode 100644 target/arm/mte_helper.c
128
hw/ssi/pl022.c | 57 ++--
129
hw/timer/cmsdk-apb-dualtimer.c | 515 +++++++++++++++++++++++++++++
130
target/arm/arm-semi.c | 2 +-
131
target/arm/helper.c | 342 +++++++++++++------
132
target/arm/iwmmxt_helper.c | 234 ++++++-------
133
target/arm/translate.c | 122 +++----
134
MAINTAINERS | 10 +
135
default-configs/arm-softmmu.mak | 4 +
136
hw/misc/trace-events | 16 +
137
hw/timer/trace-events | 5 +
138
41 files changed, 3405 insertions(+), 718 deletions(-)
139
create mode 100644 include/hw/misc/iotkit-sysctl.h
140
create mode 100644 include/hw/misc/iotkit-sysinfo.h
141
create mode 100644 include/hw/misc/tz-msc.h
142
create mode 100644 include/hw/ssi/pl022.h
143
create mode 100644 include/hw/timer/cmsdk-apb-dualtimer.h
144
create mode 100644 hw/misc/iotkit-sysctl.c
145
create mode 100644 hw/misc/iotkit-sysinfo.c
146
create mode 100644 hw/misc/tz-msc.c
147
create mode 100644 hw/timer/cmsdk-apb-dualtimer.c
148
118
diff view generated by jsdifflib
1
Fix MPS2 SCC config register values for the mps2-an511
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
and mps2-an385 boards:
3
* the SCC_AID bits [23:20] specify the FPGA build target board revision,
4
and the SCC_CFG4 register specifies the actual board revision, so
5
these should have matching values. Claim to be board revision C,
6
consistently -- we had the revision in the wrong part of SCC_AID.
7
* SCC_ID bits [15:4] should be the board number in hex, not decimal
8
2
3
I'm confused by this code, 'bmc' is created as:
4
5
bmc = g_new0(AspeedBoardState, 1);
6
7
Then we use it as QOM owner for different MemoryRegion objects.
8
But looking at memory_region_init_ram (similarly for ROM):
9
10
void memory_region_init_ram(MemoryRegion *mr,
11
struct Object *owner,
12
const char *name,
13
uint64_t size,
14
Error **errp)
15
{
16
DeviceState *owner_dev;
17
Error *err = NULL;
18
19
memory_region_init_ram_nomigrate(mr, owner, name, size, &err);
20
if (err) {
21
error_propagate(errp, err);
22
return;
23
}
24
/* This will assert if owner is neither NULL nor a DeviceState.
25
* We only want the owner here for the purposes of defining a
26
* unique name for migration. TODO: Ideally we should implement
27
* a naming scheme for Objects which are not DeviceStates, in
28
* which case we can relax this restriction.
29
*/
30
owner_dev = DEVICE(owner);
31
vmstate_register_ram(mr, owner_dev);
32
}
33
34
The expected assertion is not triggered ('bmc' is not NULL neither
35
a DeviceState).
36
37
'bmc' structure is defined as:
38
39
struct AspeedBoardState {
40
AspeedSoCState soc;
41
MemoryRegion ram_container;
42
MemoryRegion max_ram;
43
};
44
45
What happens is when using 'OBJECT(bmc)', the QOM macros cast the
46
memory pointed by bmc, which first member is 'soc', which is
47
initialized ...:
48
49
object_initialize_child(OBJECT(machine), "soc",
50
&bmc->soc, amc->soc_name);
51
52
The 'soc' object is indeed a DeviceState, so the assertion passes.
53
54
Since this is fragile and only happens to work by luck, remove the
55
dangerous OBJECT(bmc) owner argument.
56
57
Note, this probably breaks migration for this machine.
58
59
Reviewed-by: Cédric Le Goater <clg@kaod.org>
60
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
61
Message-id: 20200623072132.2868-2-f4bug@amsat.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
62
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Message-id: 20180823175225.22612-1-peter.maydell@linaro.org
12
---
63
---
13
hw/arm/mps2.c | 6 +++---
64
hw/arm/aspeed.c | 6 +++---
14
1 file changed, 3 insertions(+), 3 deletions(-)
65
1 file changed, 3 insertions(+), 3 deletions(-)
15
66
16
diff --git a/hw/arm/mps2.c b/hw/arm/mps2.c
67
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
17
index XXXXXXX..XXXXXXX 100644
68
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/mps2.c
69
--- a/hw/arm/aspeed.c
19
+++ b/hw/arm/mps2.c
70
+++ b/hw/arm/aspeed.c
20
@@ -XXX,XX +XXX,XX @@ static void mps2_common_init(MachineState *machine)
71
@@ -XXX,XX +XXX,XX @@ static void aspeed_machine_init(MachineState *machine)
21
sccdev = DEVICE(&mms->scc);
72
* needed by the flash modules of the Aspeed machines.
22
qdev_set_parent_bus(sccdev, sysbus_get_default());
73
*/
23
qdev_prop_set_uint32(sccdev, "scc-cfg4", 0x2);
74
if (ASPEED_MACHINE(machine)->mmio_exec) {
24
- qdev_prop_set_uint32(sccdev, "scc-aid", 0x02000008);
75
- memory_region_init_alias(boot_rom, OBJECT(bmc), "aspeed.boot_rom",
25
+ qdev_prop_set_uint32(sccdev, "scc-aid", 0x00200008);
76
+ memory_region_init_alias(boot_rom, NULL, "aspeed.boot_rom",
26
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
77
&fl->mmio, 0, fl->size);
27
object_property_set_bool(OBJECT(&mms->scc), true, "realized",
78
memory_region_add_subregion(get_system_memory(), FIRMWARE_ADDR,
28
&error_fatal);
79
boot_rom);
29
@@ -XXX,XX +XXX,XX @@ static void mps2_an385_class_init(ObjectClass *oc, void *data)
80
} else {
30
mc->desc = "ARM MPS2 with AN385 FPGA image for Cortex-M3";
81
- memory_region_init_rom(boot_rom, OBJECT(bmc), "aspeed.boot_rom",
31
mmc->fpga_type = FPGA_AN385;
82
+ memory_region_init_rom(boot_rom, NULL, "aspeed.boot_rom",
32
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m3");
83
fl->size, &error_abort);
33
- mmc->scc_id = 0x41040000 | (385 << 4);
84
memory_region_add_subregion(get_system_memory(), FIRMWARE_ADDR,
34
+ mmc->scc_id = 0x41043850;
85
boot_rom);
35
}
86
@@ -XXX,XX +XXX,XX @@ static void aspeed_machine_init(MachineState *machine)
36
87
if (machine->kernel_filename && sc->num_cpus > 1) {
37
static void mps2_an511_class_init(ObjectClass *oc, void *data)
88
/* With no u-boot we must set up a boot stub for the secondary CPU */
38
@@ -XXX,XX +XXX,XX @@ static void mps2_an511_class_init(ObjectClass *oc, void *data)
89
MemoryRegion *smpboot = g_new(MemoryRegion, 1);
39
mc->desc = "ARM MPS2 with AN511 DesignStart FPGA image for Cortex-M3";
90
- memory_region_init_ram(smpboot, OBJECT(bmc), "aspeed.smpboot",
40
mmc->fpga_type = FPGA_AN511;
91
+ memory_region_init_ram(smpboot, NULL, "aspeed.smpboot",
41
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m3");
92
0x80, &error_abort);
42
- mmc->scc_id = 0x4104000 | (511 << 4);
93
memory_region_add_subregion(get_system_memory(),
43
+ mmc->scc_id = 0x41045110;
94
AST_SMP_MAILBOX_BASE, smpboot);
44
}
45
46
static const TypeInfo mps2_info = {
47
--
95
--
48
2.18.0
96
2.20.1
49
97
50
98
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
2
3
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
3
To have a more consistent naming, rename AspeedBoardState
4
Message-id: 20180814002653.12828-2-richard.henderson@linaro.org
4
as AspeedMachineState.
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
6
Suggested-by: Cédric Le Goater <clg@kaod.org>
7
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Reviewed-by: Cédric Le Goater <clg@kaod.org>
9
Message-id: 20200623072132.2868-3-f4bug@amsat.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
11
---
8
include/fpu/softfloat.h | 56 ++++++++----
12
include/hw/arm/aspeed.h | 4 ++--
9
fpu/softfloat.c | 188 +++++++++++++++++++++++++++++-----------
13
hw/arm/aspeed.c | 20 ++++++++++----------
10
2 files changed, 179 insertions(+), 65 deletions(-)
14
2 files changed, 12 insertions(+), 12 deletions(-)
11
15
12
diff --git a/include/fpu/softfloat.h b/include/fpu/softfloat.h
16
diff --git a/include/hw/arm/aspeed.h b/include/hw/arm/aspeed.h
13
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
14
--- a/include/fpu/softfloat.h
18
--- a/include/hw/arm/aspeed.h
15
+++ b/include/fpu/softfloat.h
19
+++ b/include/hw/arm/aspeed.h
16
@@ -XXX,XX +XXX,XX @@ enum {
20
@@ -XXX,XX +XXX,XX @@
17
/*----------------------------------------------------------------------------
21
18
| Software IEC/IEEE integer-to-floating-point conversion routines.
22
#include "hw/boards.h"
19
*----------------------------------------------------------------------------*/
23
20
+
24
-typedef struct AspeedBoardState AspeedBoardState;
21
+float16 int16_to_float16_scalbn(int16_t a, int, float_status *status);
25
+typedef struct AspeedMachineState AspeedMachineState;
22
+float16 int32_to_float16_scalbn(int32_t a, int, float_status *status);
26
23
+float16 int64_to_float16_scalbn(int64_t a, int, float_status *status);
27
#define TYPE_ASPEED_MACHINE MACHINE_TYPE_NAME("aspeed")
24
+float16 uint16_to_float16_scalbn(uint16_t a, int, float_status *status);
28
#define ASPEED_MACHINE(obj) \
25
+float16 uint32_to_float16_scalbn(uint32_t a, int, float_status *status);
29
@@ -XXX,XX +XXX,XX @@ typedef struct AspeedMachineClass {
26
+float16 uint64_to_float16_scalbn(uint64_t a, int, float_status *status);
30
const char *spi_model;
27
+
31
uint32_t num_cs;
28
+float16 int16_to_float16(int16_t a, float_status *status);
32
uint32_t macs_mask;
29
+float16 int32_to_float16(int32_t a, float_status *status);
33
- void (*i2c_init)(AspeedBoardState *bmc);
30
+float16 int64_to_float16(int64_t a, float_status *status);
34
+ void (*i2c_init)(AspeedMachineState *bmc);
31
+float16 uint16_to_float16(uint16_t a, float_status *status);
35
} AspeedMachineClass;
32
+float16 uint32_to_float16(uint32_t a, float_status *status);
36
33
+float16 uint64_to_float16(uint64_t a, float_status *status);
37
34
+
38
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
35
+float32 int16_to_float32_scalbn(int16_t, int, float_status *status);
36
+float32 int32_to_float32_scalbn(int32_t, int, float_status *status);
37
+float32 int64_to_float32_scalbn(int64_t, int, float_status *status);
38
+float32 uint16_to_float32_scalbn(uint16_t, int, float_status *status);
39
+float32 uint32_to_float32_scalbn(uint32_t, int, float_status *status);
40
+float32 uint64_to_float32_scalbn(uint64_t, int, float_status *status);
41
+
42
float32 int16_to_float32(int16_t, float_status *status);
43
float32 int32_to_float32(int32_t, float_status *status);
44
-float64 int16_to_float64(int16_t, float_status *status);
45
-float64 int32_to_float64(int32_t, float_status *status);
46
+float32 int64_to_float32(int64_t, float_status *status);
47
float32 uint16_to_float32(uint16_t, float_status *status);
48
float32 uint32_to_float32(uint32_t, float_status *status);
49
+float32 uint64_to_float32(uint64_t, float_status *status);
50
+
51
+float64 int16_to_float64_scalbn(int16_t, int, float_status *status);
52
+float64 int32_to_float64_scalbn(int32_t, int, float_status *status);
53
+float64 int64_to_float64_scalbn(int64_t, int, float_status *status);
54
+float64 uint16_to_float64_scalbn(uint16_t, int, float_status *status);
55
+float64 uint32_to_float64_scalbn(uint32_t, int, float_status *status);
56
+float64 uint64_to_float64_scalbn(uint64_t, int, float_status *status);
57
+
58
+float64 int16_to_float64(int16_t, float_status *status);
59
+float64 int32_to_float64(int32_t, float_status *status);
60
+float64 int64_to_float64(int64_t, float_status *status);
61
float64 uint16_to_float64(uint16_t, float_status *status);
62
float64 uint32_to_float64(uint32_t, float_status *status);
63
-floatx80 int32_to_floatx80(int32_t, float_status *status);
64
-float128 int32_to_float128(int32_t, float_status *status);
65
-float32 int64_to_float32(int64_t, float_status *status);
66
-float64 int64_to_float64(int64_t, float_status *status);
67
-floatx80 int64_to_floatx80(int64_t, float_status *status);
68
-float128 int64_to_float128(int64_t, float_status *status);
69
-float32 uint64_to_float32(uint64_t, float_status *status);
70
float64 uint64_to_float64(uint64_t, float_status *status);
71
+
72
+floatx80 int32_to_floatx80(int32_t, float_status *status);
73
+floatx80 int64_to_floatx80(int64_t, float_status *status);
74
+
75
+float128 int32_to_float128(int32_t, float_status *status);
76
+float128 int64_to_float128(int64_t, float_status *status);
77
float128 uint64_to_float128(uint64_t, float_status *status);
78
79
/*----------------------------------------------------------------------------
80
@@ -XXX,XX +XXX,XX @@ int64_t float16_to_int64(float16, float_status *status);
81
uint64_t float16_to_uint64(float16 a, float_status *status);
82
int64_t float16_to_int64_round_to_zero(float16, float_status *status);
83
uint64_t float16_to_uint64_round_to_zero(float16 a, float_status *status);
84
-float16 int16_to_float16(int16_t a, float_status *status);
85
-float16 int32_to_float16(int32_t a, float_status *status);
86
-float16 int64_to_float16(int64_t a, float_status *status);
87
-float16 uint16_to_float16(uint16_t a, float_status *status);
88
-float16 uint32_to_float16(uint32_t a, float_status *status);
89
-float16 uint64_to_float16(uint64_t a, float_status *status);
90
91
/*----------------------------------------------------------------------------
92
| Software half-precision operations.
93
diff --git a/fpu/softfloat.c b/fpu/softfloat.c
94
index XXXXXXX..XXXXXXX 100644
39
index XXXXXXX..XXXXXXX 100644
95
--- a/fpu/softfloat.c
40
--- a/hw/arm/aspeed.c
96
+++ b/fpu/softfloat.c
41
+++ b/hw/arm/aspeed.c
97
@@ -XXX,XX +XXX,XX @@ FLOAT_TO_UINT(64, 64)
42
@@ -XXX,XX +XXX,XX @@ static struct arm_boot_info aspeed_board_binfo = {
98
* to the IEC/IEEE Standard for Binary Floating-Point Arithmetic.
43
.board_id = -1, /* device-tree-only board */
99
*/
44
};
100
45
101
-static FloatParts int_to_float(int64_t a, float_status *status)
46
-struct AspeedBoardState {
102
+static FloatParts int_to_float(int64_t a, int scale, float_status *status)
47
+struct AspeedMachineState {
48
AspeedSoCState soc;
49
MemoryRegion ram_container;
50
MemoryRegion max_ram;
51
@@ -XXX,XX +XXX,XX @@ static void sdhci_attach_drive(SDHCIState *sdhci, DriveInfo *dinfo)
52
53
static void aspeed_machine_init(MachineState *machine)
103
{
54
{
104
- FloatParts r = {};
55
- AspeedBoardState *bmc;
105
+ FloatParts r = { .sign = false };
56
+ AspeedMachineState *bmc;
106
+
57
AspeedMachineClass *amc = ASPEED_MACHINE_GET_CLASS(machine);
107
if (a == 0) {
58
AspeedSoCClass *sc;
108
r.cls = float_class_zero;
59
DriveInfo *drive0 = drive_get(IF_MTD, 0, 0);
109
- r.sign = false;
60
@@ -XXX,XX +XXX,XX @@ static void aspeed_machine_init(MachineState *machine)
110
- } else if (a == (1ULL << 63)) {
61
int i;
111
- r.cls = float_class_normal;
62
NICInfo *nd = &nd_table[0];
112
- r.sign = true;
63
113
- r.frac = DECOMPOSED_IMPLICIT_BIT;
64
- bmc = g_new0(AspeedBoardState, 1);
114
- r.exp = 63;
65
+ bmc = g_new0(AspeedMachineState, 1);
115
} else {
66
116
- uint64_t f;
67
memory_region_init(&bmc->ram_container, NULL, "aspeed-ram-container",
117
- if (a < 0) {
68
4 * GiB);
118
- f = -a;
69
@@ -XXX,XX +XXX,XX @@ static void aspeed_machine_init(MachineState *machine)
119
- r.sign = true;
70
arm_load_kernel(ARM_CPU(first_cpu), machine, &aspeed_board_binfo);
120
- } else {
121
- f = a;
122
- r.sign = false;
123
- }
124
- int shift = clz64(f) - 1;
125
+ uint64_t f = a;
126
+ int shift;
127
+
128
r.cls = float_class_normal;
129
- r.exp = (DECOMPOSED_BINARY_POINT - shift);
130
- r.frac = f << shift;
131
+ if (a < 0) {
132
+ f = -f;
133
+ r.sign = true;
134
+ }
135
+ shift = clz64(f) - 1;
136
+ scale = MIN(MAX(scale, -0x10000), 0x10000);
137
+
138
+ r.exp = DECOMPOSED_BINARY_POINT - shift + scale;
139
+ r.frac = (shift < 0 ? DECOMPOSED_IMPLICIT_BIT : f << shift);
140
}
141
142
return r;
143
}
71
}
144
72
145
+float16 int64_to_float16_scalbn(int64_t a, int scale, float_status *status)
73
-static void palmetto_bmc_i2c_init(AspeedBoardState *bmc)
146
+{
74
+static void palmetto_bmc_i2c_init(AspeedMachineState *bmc)
147
+ FloatParts pa = int_to_float(a, scale, status);
148
+ return float16_round_pack_canonical(pa, status);
149
+}
150
+
151
+float16 int32_to_float16_scalbn(int32_t a, int scale, float_status *status)
152
+{
153
+ return int64_to_float16_scalbn(a, scale, status);
154
+}
155
+
156
+float16 int16_to_float16_scalbn(int16_t a, int scale, float_status *status)
157
+{
158
+ return int64_to_float16_scalbn(a, scale, status);
159
+}
160
+
161
float16 int64_to_float16(int64_t a, float_status *status)
162
{
75
{
163
- FloatParts pa = int_to_float(a, status);
76
AspeedSoCState *soc = &bmc->soc;
164
- return float16_round_pack_canonical(pa, status);
77
DeviceState *dev;
165
+ return int64_to_float16_scalbn(a, 0, status);
78
@@ -XXX,XX +XXX,XX @@ static void palmetto_bmc_i2c_init(AspeedBoardState *bmc)
79
object_property_set_int(OBJECT(dev), 110000, "temperature3", &error_abort);
166
}
80
}
167
81
168
float16 int32_to_float16(int32_t a, float_status *status)
82
-static void ast2500_evb_i2c_init(AspeedBoardState *bmc)
83
+static void ast2500_evb_i2c_init(AspeedMachineState *bmc)
169
{
84
{
170
- return int64_to_float16(a, status);
85
AspeedSoCState *soc = &bmc->soc;
171
+ return int64_to_float16_scalbn(a, 0, status);
86
uint8_t *eeprom_buf = g_malloc0(8 * 1024);
87
@@ -XXX,XX +XXX,XX @@ static void ast2500_evb_i2c_init(AspeedBoardState *bmc)
88
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), "ds1338", 0x32);
172
}
89
}
173
90
174
float16 int16_to_float16(int16_t a, float_status *status)
91
-static void ast2600_evb_i2c_init(AspeedBoardState *bmc)
92
+static void ast2600_evb_i2c_init(AspeedMachineState *bmc)
175
{
93
{
176
- return int64_to_float16(a, status);
94
/* Start with some devices on our I2C busses */
177
+ return int64_to_float16_scalbn(a, 0, status);
95
ast2500_evb_i2c_init(bmc);
178
+}
179
+
180
+float32 int64_to_float32_scalbn(int64_t a, int scale, float_status *status)
181
+{
182
+ FloatParts pa = int_to_float(a, scale, status);
183
+ return float32_round_pack_canonical(pa, status);
184
+}
185
+
186
+float32 int32_to_float32_scalbn(int32_t a, int scale, float_status *status)
187
+{
188
+ return int64_to_float32_scalbn(a, scale, status);
189
+}
190
+
191
+float32 int16_to_float32_scalbn(int16_t a, int scale, float_status *status)
192
+{
193
+ return int64_to_float32_scalbn(a, scale, status);
194
}
96
}
195
97
196
float32 int64_to_float32(int64_t a, float_status *status)
98
-static void romulus_bmc_i2c_init(AspeedBoardState *bmc)
99
+static void romulus_bmc_i2c_init(AspeedMachineState *bmc)
197
{
100
{
198
- FloatParts pa = int_to_float(a, status);
101
AspeedSoCState *soc = &bmc->soc;
199
- return float32_round_pack_canonical(pa, status);
102
200
+ return int64_to_float32_scalbn(a, 0, status);
103
@@ -XXX,XX +XXX,XX @@ static void romulus_bmc_i2c_init(AspeedBoardState *bmc)
104
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), "ds1338", 0x32);
201
}
105
}
202
106
203
float32 int32_to_float32(int32_t a, float_status *status)
107
-static void swift_bmc_i2c_init(AspeedBoardState *bmc)
108
+static void swift_bmc_i2c_init(AspeedMachineState *bmc)
204
{
109
{
205
- return int64_to_float32(a, status);
110
AspeedSoCState *soc = &bmc->soc;
206
+ return int64_to_float32_scalbn(a, 0, status);
111
112
@@ -XXX,XX +XXX,XX @@ static void swift_bmc_i2c_init(AspeedBoardState *bmc)
113
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 12), "tmp105", 0x4a);
207
}
114
}
208
115
209
float32 int16_to_float32(int16_t a, float_status *status)
116
-static void sonorapass_bmc_i2c_init(AspeedBoardState *bmc)
117
+static void sonorapass_bmc_i2c_init(AspeedMachineState *bmc)
210
{
118
{
211
- return int64_to_float32(a, status);
119
AspeedSoCState *soc = &bmc->soc;
212
+ return int64_to_float32_scalbn(a, 0, status);
120
213
+}
121
@@ -XXX,XX +XXX,XX @@ static void sonorapass_bmc_i2c_init(AspeedBoardState *bmc)
214
+
122
215
+float64 int64_to_float64_scalbn(int64_t a, int scale, float_status *status)
216
+{
217
+ FloatParts pa = int_to_float(a, scale, status);
218
+ return float64_round_pack_canonical(pa, status);
219
+}
220
+
221
+float64 int32_to_float64_scalbn(int32_t a, int scale, float_status *status)
222
+{
223
+ return int64_to_float64_scalbn(a, scale, status);
224
+}
225
+
226
+float64 int16_to_float64_scalbn(int16_t a, int scale, float_status *status)
227
+{
228
+ return int64_to_float64_scalbn(a, scale, status);
229
}
123
}
230
124
231
float64 int64_to_float64(int64_t a, float_status *status)
125
-static void witherspoon_bmc_i2c_init(AspeedBoardState *bmc)
126
+static void witherspoon_bmc_i2c_init(AspeedMachineState *bmc)
232
{
127
{
233
- FloatParts pa = int_to_float(a, status);
128
AspeedSoCState *soc = &bmc->soc;
234
- return float64_round_pack_canonical(pa, status);
129
uint8_t *eeprom_buf = g_malloc0(8 * 1024);
235
+ return int64_to_float64_scalbn(a, 0, status);
236
}
237
238
float64 int32_to_float64(int32_t a, float_status *status)
239
{
240
- return int64_to_float64(a, status);
241
+ return int64_to_float64_scalbn(a, 0, status);
242
}
243
244
float64 int16_to_float64(int16_t a, float_status *status)
245
{
246
- return int64_to_float64(a, status);
247
+ return int64_to_float64_scalbn(a, 0, status);
248
}
249
250
251
@@ -XXX,XX +XXX,XX @@ float64 int16_to_float64(int16_t a, float_status *status)
252
* IEC/IEEE Standard for Binary Floating-Point Arithmetic.
253
*/
254
255
-static FloatParts uint_to_float(uint64_t a, float_status *status)
256
+static FloatParts uint_to_float(uint64_t a, int scale, float_status *status)
257
{
258
- FloatParts r = { .sign = false};
259
+ FloatParts r = { .sign = false };
260
261
if (a == 0) {
262
r.cls = float_class_zero;
263
} else {
264
- int spare_bits = clz64(a) - 1;
265
+ scale = MIN(MAX(scale, -0x10000), 0x10000);
266
r.cls = float_class_normal;
267
- r.exp = DECOMPOSED_BINARY_POINT - spare_bits;
268
- if (spare_bits < 0) {
269
- shift64RightJamming(a, -spare_bits, &a);
270
+ if ((int64_t)a < 0) {
271
+ r.exp = DECOMPOSED_BINARY_POINT + 1 + scale;
272
+ shift64RightJamming(a, 1, &a);
273
r.frac = a;
274
} else {
275
- r.frac = a << spare_bits;
276
+ int shift = clz64(a) - 1;
277
+ r.exp = DECOMPOSED_BINARY_POINT - shift + scale;
278
+ r.frac = a << shift;
279
}
280
}
281
282
return r;
283
}
284
285
+float16 uint64_to_float16_scalbn(uint64_t a, int scale, float_status *status)
286
+{
287
+ FloatParts pa = uint_to_float(a, scale, status);
288
+ return float16_round_pack_canonical(pa, status);
289
+}
290
+
291
+float16 uint32_to_float16_scalbn(uint32_t a, int scale, float_status *status)
292
+{
293
+ return uint64_to_float16_scalbn(a, scale, status);
294
+}
295
+
296
+float16 uint16_to_float16_scalbn(uint16_t a, int scale, float_status *status)
297
+{
298
+ return uint64_to_float16_scalbn(a, scale, status);
299
+}
300
+
301
float16 uint64_to_float16(uint64_t a, float_status *status)
302
{
303
- FloatParts pa = uint_to_float(a, status);
304
- return float16_round_pack_canonical(pa, status);
305
+ return uint64_to_float16_scalbn(a, 0, status);
306
}
307
308
float16 uint32_to_float16(uint32_t a, float_status *status)
309
{
310
- return uint64_to_float16(a, status);
311
+ return uint64_to_float16_scalbn(a, 0, status);
312
}
313
314
float16 uint16_to_float16(uint16_t a, float_status *status)
315
{
316
- return uint64_to_float16(a, status);
317
+ return uint64_to_float16_scalbn(a, 0, status);
318
+}
319
+
320
+float32 uint64_to_float32_scalbn(uint64_t a, int scale, float_status *status)
321
+{
322
+ FloatParts pa = uint_to_float(a, scale, status);
323
+ return float32_round_pack_canonical(pa, status);
324
+}
325
+
326
+float32 uint32_to_float32_scalbn(uint32_t a, int scale, float_status *status)
327
+{
328
+ return uint64_to_float32_scalbn(a, scale, status);
329
+}
330
+
331
+float32 uint16_to_float32_scalbn(uint16_t a, int scale, float_status *status)
332
+{
333
+ return uint64_to_float32_scalbn(a, scale, status);
334
}
335
336
float32 uint64_to_float32(uint64_t a, float_status *status)
337
{
338
- FloatParts pa = uint_to_float(a, status);
339
- return float32_round_pack_canonical(pa, status);
340
+ return uint64_to_float32_scalbn(a, 0, status);
341
}
342
343
float32 uint32_to_float32(uint32_t a, float_status *status)
344
{
345
- return uint64_to_float32(a, status);
346
+ return uint64_to_float32_scalbn(a, 0, status);
347
}
348
349
float32 uint16_to_float32(uint16_t a, float_status *status)
350
{
351
- return uint64_to_float32(a, status);
352
+ return uint64_to_float32_scalbn(a, 0, status);
353
+}
354
+
355
+float64 uint64_to_float64_scalbn(uint64_t a, int scale, float_status *status)
356
+{
357
+ FloatParts pa = uint_to_float(a, scale, status);
358
+ return float64_round_pack_canonical(pa, status);
359
+}
360
+
361
+float64 uint32_to_float64_scalbn(uint32_t a, int scale, float_status *status)
362
+{
363
+ return uint64_to_float64_scalbn(a, scale, status);
364
+}
365
+
366
+float64 uint16_to_float64_scalbn(uint16_t a, int scale, float_status *status)
367
+{
368
+ return uint64_to_float64_scalbn(a, scale, status);
369
}
370
371
float64 uint64_to_float64(uint64_t a, float_status *status)
372
{
373
- FloatParts pa = uint_to_float(a, status);
374
- return float64_round_pack_canonical(pa, status);
375
+ return uint64_to_float64_scalbn(a, 0, status);
376
}
377
378
float64 uint32_to_float64(uint32_t a, float_status *status)
379
{
380
- return uint64_to_float64(a, status);
381
+ return uint64_to_float64_scalbn(a, 0, status);
382
}
383
384
float64 uint16_to_float64(uint16_t a, float_status *status)
385
{
386
- return uint64_to_float64(a, status);
387
+ return uint64_to_float64_scalbn(a, 0, status);
388
}
389
390
/* Float Min/Max */
391
--
130
--
392
2.18.0
131
2.20.1
393
132
394
133
diff view generated by jsdifflib
1
The BCM2835FBState struct has a 'pitch' field which is a
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
cached copy of xres * (bpp >> 3), and a 'size' field which is
3
a cached copy of pitch * yres. However we don't actually do
4
anything with these fields; delete them. We retain the
5
now-unused slots in the VMState struct for migration
6
compatibility.
7
2
3
AspeedMachineState seems crippled. We use incorrectly 2
4
different structures to do the same thing. Merge them
5
altogether:
6
- Move AspeedMachine fields to AspeedMachineState
7
- AspeedMachineState is now QOM
8
- Remove unused AspeedMachine structure
9
10
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
11
Reviewed-by: Cédric Le Goater <clg@kaod.org>
12
Message-id: 20200623072132.2868-4-f4bug@amsat.org
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Message-id: 20180814144436.679-4-peter.maydell@linaro.org
11
---
14
---
12
include/hw/display/bcm2835_fb.h | 4 ----
15
include/hw/arm/aspeed.h | 8 +-------
13
hw/display/bcm2835_fb.c | 19 ++++++++-----------
16
hw/arm/aspeed.c | 11 +++++++----
14
2 files changed, 8 insertions(+), 15 deletions(-)
17
2 files changed, 8 insertions(+), 11 deletions(-)
15
18
16
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
19
diff --git a/include/hw/arm/aspeed.h b/include/hw/arm/aspeed.h
17
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/display/bcm2835_fb.h
21
--- a/include/hw/arm/aspeed.h
19
+++ b/include/hw/display/bcm2835_fb.h
22
+++ b/include/hw/arm/aspeed.h
20
@@ -XXX,XX +XXX,XX @@ typedef struct {
23
@@ -XXX,XX +XXX,XX @@ typedef struct AspeedMachineState AspeedMachineState;
21
bool lock, invalidate, pending;
24
22
25
#define TYPE_ASPEED_MACHINE MACHINE_TYPE_NAME("aspeed")
23
BCM2835FBConfig config;
26
#define ASPEED_MACHINE(obj) \
27
- OBJECT_CHECK(AspeedMachine, (obj), TYPE_ASPEED_MACHINE)
24
-
28
-
25
- /* These are just cached values calculated from the config settings */
29
-typedef struct AspeedMachine {
26
- uint32_t size;
30
- MachineState parent_obj;
27
- uint32_t pitch;
31
-
28
} BCM2835FBState;
32
- bool mmio_exec;
29
33
-} AspeedMachine;
30
void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig);
34
+ OBJECT_CHECK(AspeedMachineState, (obj), TYPE_ASPEED_MACHINE)
31
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
35
36
#define ASPEED_MAC0_ON (1 << 0)
37
#define ASPEED_MAC1_ON (1 << 1)
38
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
32
index XXXXXXX..XXXXXXX 100644
39
index XXXXXXX..XXXXXXX 100644
33
--- a/hw/display/bcm2835_fb.c
40
--- a/hw/arm/aspeed.c
34
+++ b/hw/display/bcm2835_fb.c
41
+++ b/hw/arm/aspeed.c
35
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
42
@@ -XXX,XX +XXX,XX @@ static struct arm_boot_info aspeed_board_binfo = {
36
43
};
37
static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
44
45
struct AspeedMachineState {
46
+ /* Private */
47
+ MachineState parent_obj;
48
+ /* Public */
49
+
50
AspeedSoCState soc;
51
MemoryRegion ram_container;
52
MemoryRegion max_ram;
53
+ bool mmio_exec;
54
};
55
56
/* Palmetto hardware value: 0x120CE416 */
57
@@ -XXX,XX +XXX,XX @@ static void sdhci_attach_drive(SDHCIState *sdhci, DriveInfo *dinfo)
58
59
static void aspeed_machine_init(MachineState *machine)
38
{
60
{
39
+ uint32_t pitch;
61
- AspeedMachineState *bmc;
40
+ uint32_t size;
62
+ AspeedMachineState *bmc = ASPEED_MACHINE(machine);
41
+
63
AspeedMachineClass *amc = ASPEED_MACHINE_GET_CLASS(machine);
42
value &= ~0xf;
64
AspeedSoCClass *sc;
43
65
DriveInfo *drive0 = drive_get(IF_MTD, 0, 0);
44
s->lock = true;
66
@@ -XXX,XX +XXX,XX @@ static void aspeed_machine_init(MachineState *machine)
45
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
67
int i;
46
68
NICInfo *nd = &nd_table[0];
47
/* TODO - Manage properly virtual resolution */
69
48
70
- bmc = g_new0(AspeedMachineState, 1);
49
- s->pitch = s->config.xres * (s->config.bpp >> 3);
50
- s->size = s->config.yres * s->pitch;
51
+ pitch = s->config.xres * (s->config.bpp >> 3);
52
+ size = s->config.yres * pitch;
53
54
- stl_le_phys(&s->dma_as, value + 16, s->pitch);
55
+ stl_le_phys(&s->dma_as, value + 16, pitch);
56
stl_le_phys(&s->dma_as, value + 32, s->config.base);
57
- stl_le_phys(&s->dma_as, value + 36, s->size);
58
+ stl_le_phys(&s->dma_as, value + 36, size);
59
60
s->invalidate = true;
61
qemu_console_resize(s->con, s->config.xres, s->config.yres);
62
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
63
64
/* TODO - Manage properly virtual resolution */
65
66
- s->pitch = s->config.xres * (s->config.bpp >> 3);
67
- s->size = s->config.yres * s->pitch;
68
-
71
-
69
s->invalidate = true;
72
memory_region_init(&bmc->ram_container, NULL, "aspeed-ram-container",
70
qemu_console_resize(s->con, s->config.xres, s->config.yres);
73
4 * GiB);
71
s->lock = false;
74
memory_region_add_subregion(&bmc->ram_container, 0, machine->ram);
72
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_bcm2835_fb = {
75
@@ -XXX,XX +XXX,XX @@ static const TypeInfo aspeed_machine_types[] = {
73
VMSTATE_UINT32(config.yoffset, BCM2835FBState),
76
}, {
74
VMSTATE_UINT32(config.bpp, BCM2835FBState),
77
.name = TYPE_ASPEED_MACHINE,
75
VMSTATE_UINT32(config.base, BCM2835FBState),
78
.parent = TYPE_MACHINE,
76
- VMSTATE_UINT32(pitch, BCM2835FBState),
79
- .instance_size = sizeof(AspeedMachine),
77
- VMSTATE_UINT32(size, BCM2835FBState),
80
+ .instance_size = sizeof(AspeedMachineState),
78
+ VMSTATE_UNUSED(8), /* Was pitch and size */
81
.instance_init = aspeed_machine_instance_init,
79
VMSTATE_UINT32(config.pixo, BCM2835FBState),
82
.class_size = sizeof(AspeedMachineClass),
80
VMSTATE_UINT32(config.alpha, BCM2835FBState),
83
.class_init = aspeed_machine_class_init,
81
VMSTATE_END_OF_LIST()
82
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_reset(DeviceState *dev)
83
s->config.xoffset = 0;
84
s->config.yoffset = 0;
85
s->config.base = s->vcram_base + BCM2835_FB_OFFSET;
86
- s->pitch = s->config.xres * (s->config.bpp >> 3);
87
- s->size = s->config.yres * s->pitch;
88
89
s->invalidate = true;
90
s->lock = false;
91
--
84
--
92
2.18.0
85
2.20.1
93
86
94
87
diff view generated by jsdifflib
1
Add a "virtualization" property to the vexpress-a15 board,
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
controlling presence of EL2. As with EL3, we default to
3
enabling it, but the user can disable it if they have an
4
older guest which can't cope with it being present.
5
2
3
Extract i2c_try_create_slave() and i2c_realize_and_unref()
4
from i2c_create_slave().
5
We can now set properties on a I2CSlave before it is realized.
6
7
This is in line with the recent qdev/QOM changes merged
8
in commit 6675a653d2e.
9
10
Reviewed-by: Corey Minyard <cminyard@mvista.com>
11
Reviewed-by: Cédric Le Goater <clg@kaod.org>
12
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
13
Reviewed-by: Markus Armbruster <armbru@redhat.com>
14
Tested-by: Cédric Le Goater <clg@kaod.org>
15
Message-id: 20200623072723.6324-2-f4bug@amsat.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180821132811.17675-10-peter.maydell@linaro.org
9
---
17
---
10
hw/arm/vexpress.c | 56 ++++++++++++++++++++++++++++++++++++++++++++---
18
include/hw/i2c/i2c.h | 2 ++
11
1 file changed, 53 insertions(+), 3 deletions(-)
19
hw/i2c/core.c | 18 ++++++++++++++++--
20
2 files changed, 18 insertions(+), 2 deletions(-)
12
21
13
diff --git a/hw/arm/vexpress.c b/hw/arm/vexpress.c
22
diff --git a/include/hw/i2c/i2c.h b/include/hw/i2c/i2c.h
14
index XXXXXXX..XXXXXXX 100644
23
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/vexpress.c
24
--- a/include/hw/i2c/i2c.h
16
+++ b/hw/arm/vexpress.c
25
+++ b/include/hw/i2c/i2c.h
17
@@ -XXX,XX +XXX,XX @@ typedef struct {
26
@@ -XXX,XX +XXX,XX @@ int i2c_send(I2CBus *bus, uint8_t data);
18
typedef struct {
27
uint8_t i2c_recv(I2CBus *bus);
19
MachineState parent;
28
20
bool secure;
29
DeviceState *i2c_create_slave(I2CBus *bus, const char *name, uint8_t addr);
21
+ bool virt;
30
+DeviceState *i2c_try_create_slave(const char *name, uint8_t addr);
22
} VexpressMachineState;
31
+bool i2c_realize_and_unref(DeviceState *dev, I2CBus *bus, Error **errp);
23
32
24
#define TYPE_VEXPRESS_MACHINE "vexpress"
33
/* lm832x.c */
25
@@ -XXX,XX +XXX,XX @@ struct VEDBoardInfo {
34
void lm832x_key_event(DeviceState *dev, int key, int state);
35
diff --git a/hw/i2c/core.c b/hw/i2c/core.c
36
index XXXXXXX..XXXXXXX 100644
37
--- a/hw/i2c/core.c
38
+++ b/hw/i2c/core.c
39
@@ -XXX,XX +XXX,XX @@ const VMStateDescription vmstate_i2c_slave = {
40
}
26
};
41
};
27
42
28
static void init_cpus(const char *cpu_type, const char *privdev,
43
-DeviceState *i2c_create_slave(I2CBus *bus, const char *name, uint8_t addr)
29
- hwaddr periphbase, qemu_irq *pic, bool secure)
44
+DeviceState *i2c_try_create_slave(const char *name, uint8_t addr)
30
+ hwaddr periphbase, qemu_irq *pic, bool secure, bool virt)
31
{
45
{
32
DeviceState *dev;
46
DeviceState *dev;
33
SysBusDevice *busdev;
47
34
@@ -XXX,XX +XXX,XX @@ static void init_cpus(const char *cpu_type, const char *privdev,
48
dev = qdev_new(name);
35
if (!secure) {
49
qdev_prop_set_uint8(dev, "address", addr);
36
object_property_set_bool(cpuobj, false, "has_el3", NULL);
50
- qdev_realize_and_unref(dev, &bus->qbus, &error_fatal);
37
}
51
+ return dev;
38
+ if (!virt) {
39
+ if (object_property_find(cpuobj, "has_el2", NULL)) {
40
+ object_property_set_bool(cpuobj, false, "has_el2", NULL);
41
+ }
42
+ }
43
44
if (object_property_find(cpuobj, "reset-cbar", NULL)) {
45
object_property_set_int(cpuobj, periphbase,
46
@@ -XXX,XX +XXX,XX @@ static void a9_daughterboard_init(const VexpressMachineState *vms,
47
memory_region_add_subregion(sysmem, 0x60000000, ram);
48
49
/* 0x1e000000 A9MPCore (SCU) private memory region */
50
- init_cpus(cpu_type, TYPE_A9MPCORE_PRIV, 0x1e000000, pic, vms->secure);
51
+ init_cpus(cpu_type, TYPE_A9MPCORE_PRIV, 0x1e000000, pic,
52
+ vms->secure, vms->virt);
53
54
/* Daughterboard peripherals : 0x10020000 .. 0x20000000 */
55
56
@@ -XXX,XX +XXX,XX @@ static void a15_daughterboard_init(const VexpressMachineState *vms,
57
memory_region_add_subregion(sysmem, 0x80000000, ram);
58
59
/* 0x2c000000 A15MPCore private memory region (GIC) */
60
- init_cpus(cpu_type, TYPE_A15MPCORE_PRIV, 0x2c000000, pic, vms->secure);
61
+ init_cpus(cpu_type, TYPE_A15MPCORE_PRIV, 0x2c000000, pic, vms->secure,
62
+ vms->virt);
63
64
/* A15 daughterboard peripherals: */
65
66
@@ -XXX,XX +XXX,XX @@ static void vexpress_set_secure(Object *obj, bool value, Error **errp)
67
vms->secure = value;
68
}
69
70
+static bool vexpress_get_virt(Object *obj, Error **errp)
71
+{
72
+ VexpressMachineState *vms = VEXPRESS_MACHINE(obj);
73
+
74
+ return vms->virt;
75
+}
52
+}
76
+
53
+
77
+static void vexpress_set_virt(Object *obj, bool value, Error **errp)
54
+bool i2c_realize_and_unref(DeviceState *dev, I2CBus *bus, Error **errp)
78
+{
55
+{
79
+ VexpressMachineState *vms = VEXPRESS_MACHINE(obj);
56
+ return qdev_realize_and_unref(dev, &bus->qbus, errp);
80
+
81
+ vms->virt = value;
82
+}
57
+}
83
+
58
+
84
static void vexpress_instance_init(Object *obj)
59
+DeviceState *i2c_create_slave(I2CBus *bus, const char *name, uint8_t addr)
85
{
60
+{
86
VexpressMachineState *vms = VEXPRESS_MACHINE(obj);
61
+ DeviceState *dev;
87
@@ -XXX,XX +XXX,XX @@ static void vexpress_instance_init(Object *obj)
62
+
88
NULL);
63
+ dev = i2c_try_create_slave(name, addr);
64
+ i2c_realize_and_unref(dev, bus, &error_fatal);
65
+
66
return dev;
89
}
67
}
90
68
91
+static void vexpress_a15_instance_init(Object *obj)
92
+{
93
+ VexpressMachineState *vms = VEXPRESS_MACHINE(obj);
94
+
95
+ /*
96
+ * For the vexpress-a15, EL2 is by default enabled if EL3 is,
97
+ * but can also be specifically set to on or off.
98
+ */
99
+ vms->virt = true;
100
+ object_property_add_bool(obj, "virtualization", vexpress_get_virt,
101
+ vexpress_set_virt, NULL);
102
+ object_property_set_description(obj, "virtualization",
103
+ "Set on/off to enable/disable the ARM "
104
+ "Virtualization Extensions "
105
+ "(defaults to same as 'secure')",
106
+ NULL);
107
+}
108
+
109
+static void vexpress_a9_instance_init(Object *obj)
110
+{
111
+ VexpressMachineState *vms = VEXPRESS_MACHINE(obj);
112
+
113
+ /* The A9 doesn't have the virt extensions */
114
+ vms->virt = false;
115
+}
116
+
117
static void vexpress_class_init(ObjectClass *oc, void *data)
118
{
119
MachineClass *mc = MACHINE_CLASS(oc);
120
@@ -XXX,XX +XXX,XX @@ static const TypeInfo vexpress_a9_info = {
121
.name = TYPE_VEXPRESS_A9_MACHINE,
122
.parent = TYPE_VEXPRESS_MACHINE,
123
.class_init = vexpress_a9_class_init,
124
+ .instance_init = vexpress_a9_instance_init,
125
};
126
127
static const TypeInfo vexpress_a15_info = {
128
.name = TYPE_VEXPRESS_A15_MACHINE,
129
.parent = TYPE_VEXPRESS_MACHINE,
130
.class_init = vexpress_a15_class_init,
131
+ .instance_init = vexpress_a15_instance_init,
132
};
133
134
static void vexpress_machine_init(void)
135
--
69
--
136
2.18.0
70
2.20.1
137
71
138
72
diff view generated by jsdifflib
1
Refactor the fb property setting code so that rather than
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
using a set of pointers to local variables to track
3
whether a config value has been updated in the current
4
mbox and if so what its new value is, we just copy
5
all the current settings of the fb at the start, and
6
then update that copy as we go along, before asking
7
the fb to switch to it at the end.
8
2
3
The PCA9552 device does not expose LEDs, but simple pins
4
to connnect LEDs to. To be clearer with the device model,
5
rename 'nr_leds' as 'pin_count'.
6
7
Reviewed-by: Cédric Le Goater <clg@kaod.org>
8
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Tested-by: Cédric Le Goater <clg@kaod.org>
10
Message-id: 20200623072723.6324-3-f4bug@amsat.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Message-id: 20180814144436.679-3-peter.maydell@linaro.org
12
---
12
---
13
include/hw/display/bcm2835_fb.h | 4 +-
13
include/hw/misc/pca9552.h | 2 +-
14
hw/display/bcm2835_fb.c | 27 ++---------
14
hw/misc/pca9552.c | 10 +++++-----
15
hw/misc/bcm2835_property.c | 80 ++++++++++++++-------------------
15
2 files changed, 6 insertions(+), 6 deletions(-)
16
3 files changed, 37 insertions(+), 74 deletions(-)
17
16
18
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
17
diff --git a/include/hw/misc/pca9552.h b/include/hw/misc/pca9552.h
19
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
20
--- a/include/hw/display/bcm2835_fb.h
19
--- a/include/hw/misc/pca9552.h
21
+++ b/include/hw/display/bcm2835_fb.h
20
+++ b/include/hw/misc/pca9552.h
22
@@ -XXX,XX +XXX,XX @@ typedef struct {
21
@@ -XXX,XX +XXX,XX @@ typedef struct PCA9552State {
23
uint32_t pitch;
22
24
} BCM2835FBState;
23
uint8_t regs[PCA9552_NR_REGS];
25
24
uint8_t max_reg;
26
-void bcm2835_fb_reconfigure(BCM2835FBState *s, uint32_t *xres, uint32_t *yres,
25
- uint8_t nr_leds;
27
- uint32_t *xoffset, uint32_t *yoffset, uint32_t *bpp,
26
+ uint8_t pin_count;
28
- uint32_t *pixo, uint32_t *alpha);
27
} PCA9552State;
29
+void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig);
30
28
31
#endif
29
#endif
32
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
30
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
33
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
34
--- a/hw/display/bcm2835_fb.c
32
--- a/hw/misc/pca9552.c
35
+++ b/hw/display/bcm2835_fb.c
33
+++ b/hw/misc/pca9552.c
36
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
34
@@ -XXX,XX +XXX,XX @@ static void pca9552_update_pin_input(PCA9552State *s)
37
s->lock = false;
38
}
39
40
-void bcm2835_fb_reconfigure(BCM2835FBState *s, uint32_t *xres, uint32_t *yres,
41
- uint32_t *xoffset, uint32_t *yoffset, uint32_t *bpp,
42
- uint32_t *pixo, uint32_t *alpha)
43
+void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
44
{
35
{
45
s->lock = true;
36
int i;
46
37
47
/* TODO: input validation! */
38
- for (i = 0; i < s->nr_leds; i++) {
48
- if (xres) {
39
+ for (i = 0; i < s->pin_count; i++) {
49
- s->config.xres = *xres;
40
uint8_t input_reg = PCA9552_INPUT0 + (i / 8);
50
- }
41
uint8_t input_shift = (i % 8);
51
- if (yres) {
42
uint8_t config = pca9552_pin_get_config(s, i);
52
- s->config.yres = *yres;
43
@@ -XXX,XX +XXX,XX @@ static void pca9552_get_led(Object *obj, Visitor *v, const char *name,
53
- }
44
error_setg(errp, "%s: error reading %s", __func__, name);
54
- if (xoffset) {
45
return;
55
- s->config.xoffset = *xoffset;
56
- }
57
- if (yoffset) {
58
- s->config.yoffset = *yoffset;
59
- }
60
- if (bpp) {
61
- s->config.bpp = *bpp;
62
- }
63
- if (pixo) {
64
- s->config.pixo = *pixo;
65
- }
66
- if (alpha) {
67
- s->config.alpha = *alpha;
68
- }
69
+
70
+ s->config = *newconfig;
71
72
/* TODO - Manage properly virtual resolution */
73
74
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
75
index XXXXXXX..XXXXXXX 100644
76
--- a/hw/misc/bcm2835_property.c
77
+++ b/hw/misc/bcm2835_property.c
78
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
79
uint32_t tmp;
80
int n;
81
uint32_t offset, length, color;
82
- uint32_t xres, yres, xoffset, yoffset, bpp, pixo, alpha;
83
- uint32_t tmp_xres, tmp_yres, tmp_xoffset, tmp_yoffset;
84
- uint32_t tmp_bpp, tmp_pixo, tmp_alpha;
85
- uint32_t *newxres = NULL, *newyres = NULL, *newxoffset = NULL,
86
- *newyoffset = NULL, *newbpp = NULL, *newpixo = NULL, *newalpha = NULL;
87
+
88
+ /*
89
+ * Copy the current state of the framebuffer config; we will update
90
+ * this copy as we process tags and then ask the framebuffer to use
91
+ * it at the end.
92
+ */
93
+ BCM2835FBConfig fbconfig = s->fbdev->config;
94
+ bool fbconfig_updated = false;
95
96
value &= ~0xf;
97
98
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
99
/* Frame buffer */
100
101
case 0x00040001: /* Allocate buffer */
102
- stl_le_phys(&s->dma_as, value + 12, s->fbdev->config.base);
103
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
104
- tmp_yres = newyres != NULL ? *newyres : s->fbdev->config.yres;
105
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
106
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.base);
107
stl_le_phys(&s->dma_as, value + 16,
108
- tmp_xres * tmp_yres * tmp_bpp / 8);
109
+ fbconfig.xres * fbconfig.yres * fbconfig.bpp / 8);
110
resplen = 8;
111
break;
112
case 0x00048001: /* Release buffer */
113
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
114
break;
115
case 0x00040003: /* Get display width/height */
116
case 0x00040004:
117
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
118
- tmp_yres = newyres != NULL ? *newyres : s->fbdev->config.yres;
119
- stl_le_phys(&s->dma_as, value + 12, tmp_xres);
120
- stl_le_phys(&s->dma_as, value + 16, tmp_yres);
121
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xres);
122
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yres);
123
resplen = 8;
124
break;
125
case 0x00044003: /* Test display width/height */
126
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
127
break;
128
case 0x00048003: /* Set display width/height */
129
case 0x00048004:
130
- xres = ldl_le_phys(&s->dma_as, value + 12);
131
- newxres = &xres;
132
- yres = ldl_le_phys(&s->dma_as, value + 16);
133
- newyres = &yres;
134
+ fbconfig.xres = ldl_le_phys(&s->dma_as, value + 12);
135
+ fbconfig.yres = ldl_le_phys(&s->dma_as, value + 16);
136
+ fbconfig_updated = true;
137
resplen = 8;
138
break;
139
case 0x00040005: /* Get depth */
140
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
141
- stl_le_phys(&s->dma_as, value + 12, tmp_bpp);
142
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.bpp);
143
resplen = 4;
144
break;
145
case 0x00044005: /* Test depth */
146
resplen = 4;
147
break;
148
case 0x00048005: /* Set depth */
149
- bpp = ldl_le_phys(&s->dma_as, value + 12);
150
- newbpp = &bpp;
151
+ fbconfig.bpp = ldl_le_phys(&s->dma_as, value + 12);
152
+ fbconfig_updated = true;
153
resplen = 4;
154
break;
155
case 0x00040006: /* Get pixel order */
156
- tmp_pixo = newpixo != NULL ? *newpixo : s->fbdev->config.pixo;
157
- stl_le_phys(&s->dma_as, value + 12, tmp_pixo);
158
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.pixo);
159
resplen = 4;
160
break;
161
case 0x00044006: /* Test pixel order */
162
resplen = 4;
163
break;
164
case 0x00048006: /* Set pixel order */
165
- pixo = ldl_le_phys(&s->dma_as, value + 12);
166
- newpixo = &pixo;
167
+ fbconfig.pixo = ldl_le_phys(&s->dma_as, value + 12);
168
+ fbconfig_updated = true;
169
resplen = 4;
170
break;
171
case 0x00040007: /* Get alpha */
172
- tmp_alpha = newalpha != NULL ? *newalpha : s->fbdev->config.alpha;
173
- stl_le_phys(&s->dma_as, value + 12, tmp_alpha);
174
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.alpha);
175
resplen = 4;
176
break;
177
case 0x00044007: /* Test pixel alpha */
178
resplen = 4;
179
break;
180
case 0x00048007: /* Set alpha */
181
- alpha = ldl_le_phys(&s->dma_as, value + 12);
182
- newalpha = &alpha;
183
+ fbconfig.alpha = ldl_le_phys(&s->dma_as, value + 12);
184
+ fbconfig_updated = true;
185
resplen = 4;
186
break;
187
case 0x00040008: /* Get pitch */
188
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
189
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
190
- stl_le_phys(&s->dma_as, value + 12, tmp_xres * tmp_bpp / 8);
191
+ stl_le_phys(&s->dma_as, value + 12,
192
+ fbconfig.xres * fbconfig.bpp / 8);
193
resplen = 4;
194
break;
195
case 0x00040009: /* Get virtual offset */
196
- tmp_xoffset = newxoffset != NULL ?
197
- *newxoffset : s->fbdev->config.xoffset;
198
- tmp_yoffset = newyoffset != NULL ?
199
- *newyoffset : s->fbdev->config.yoffset;
200
- stl_le_phys(&s->dma_as, value + 12, tmp_xoffset);
201
- stl_le_phys(&s->dma_as, value + 16, tmp_yoffset);
202
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xoffset);
203
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yoffset);
204
resplen = 8;
205
break;
206
case 0x00044009: /* Test virtual offset */
207
resplen = 8;
208
break;
209
case 0x00048009: /* Set virtual offset */
210
- xoffset = ldl_le_phys(&s->dma_as, value + 12);
211
- newxoffset = &xoffset;
212
- yoffset = ldl_le_phys(&s->dma_as, value + 16);
213
- newyoffset = &yoffset;
214
+ fbconfig.xoffset = ldl_le_phys(&s->dma_as, value + 12);
215
+ fbconfig.yoffset = ldl_le_phys(&s->dma_as, value + 16);
216
+ fbconfig_updated = true;
217
resplen = 8;
218
break;
219
case 0x0004000a: /* Get/Test/Set overscan */
220
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
221
}
46
}
222
47
- if (led < 0 || led > s->nr_leds) {
223
/* Reconfigure framebuffer if required */
48
+ if (led < 0 || led > s->pin_count) {
224
- if (newxres || newyres || newxoffset || newyoffset || newbpp || newpixo
49
error_setg(errp, "%s invalid led %s", __func__, name);
225
- || newalpha) {
50
return;
226
- bcm2835_fb_reconfigure(s->fbdev, newxres, newyres, newxoffset,
227
- newyoffset, newbpp, newpixo, newalpha);
228
+ if (fbconfig_updated) {
229
+ bcm2835_fb_reconfigure(s->fbdev, &fbconfig);
230
}
51
}
231
52
@@ -XXX,XX +XXX,XX @@ static void pca9552_set_led(Object *obj, Visitor *v, const char *name,
232
/* Buffer response code */
53
error_setg(errp, "%s: error reading %s", __func__, name);
54
return;
55
}
56
- if (led < 0 || led > s->nr_leds) {
57
+ if (led < 0 || led > s->pin_count) {
58
error_setg(errp, "%s invalid led %s", __func__, name);
59
return;
60
}
61
@@ -XXX,XX +XXX,XX @@ static void pca9552_initfn(Object *obj)
62
* PCA955X device
63
*/
64
s->max_reg = PCA9552_LS3;
65
- s->nr_leds = 16;
66
+ s->pin_count = 16;
67
68
- for (led = 0; led < s->nr_leds; led++) {
69
+ for (led = 0; led < s->pin_count; led++) {
70
char *name;
71
72
name = g_strdup_printf("led%d", led);
233
--
73
--
234
2.18.0
74
2.20.1
235
75
236
76
diff view generated by jsdifflib
1
The handling of framebuffer properties in the bcm2835_property code
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
is a bit clumsy, because for each of the many fb related properties
2
3
we try to track the value we're about to set and whether we're going
3
Various code from the PCA9552 device model is generic to the
4
to be setting a value, and then we hand all the new values off
4
PCA955X family. We'll split the generic code in a base class
5
to the framebuffer via a function which takes them all as separate
5
in the next commit. To ease review, first do a dumb renaming.
6
arguments. It would be simpler if the property code could easily
6
7
copy all the framebuffer's current settings, update them with
7
Reviewed-by: Cédric Le Goater <clg@kaod.org>
8
the new specified values and then ask the framebuffer to switch
8
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
to the new set.
9
Tested-by: Cédric Le Goater <clg@kaod.org>
10
10
Message-id: 20200623072723.6324-4-f4bug@amsat.org
11
As the first part of this refactoring, pull all the fb config
12
settings fields in BCM2835FBState out into their own struct.
13
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
16
Message-id: 20180814144436.679-2-peter.maydell@linaro.org
17
---
12
---
18
include/hw/display/bcm2835_fb.h | 26 ++++++--
13
include/hw/misc/pca9552.h | 10 ++---
19
hw/display/bcm2835_fb.c | 114 +++++++++++++++++---------------
14
hw/misc/pca9552.c | 80 +++++++++++++++++++--------------------
20
hw/misc/bcm2835_property.c | 28 ++++----
15
2 files changed, 45 insertions(+), 45 deletions(-)
21
3 files changed, 94 insertions(+), 74 deletions(-)
16
22
17
diff --git a/include/hw/misc/pca9552.h b/include/hw/misc/pca9552.h
23
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
24
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
25
--- a/include/hw/display/bcm2835_fb.h
19
--- a/include/hw/misc/pca9552.h
26
+++ b/include/hw/display/bcm2835_fb.h
20
+++ b/include/hw/misc/pca9552.h
27
@@ -XXX,XX +XXX,XX @@
21
@@ -XXX,XX +XXX,XX @@
28
#define TYPE_BCM2835_FB "bcm2835-fb"
22
#include "hw/i2c/i2c.h"
29
#define BCM2835_FB(obj) OBJECT_CHECK(BCM2835FBState, (obj), TYPE_BCM2835_FB)
23
30
24
#define TYPE_PCA9552 "pca9552"
31
+/*
25
-#define PCA9552(obj) OBJECT_CHECK(PCA9552State, (obj), TYPE_PCA9552)
32
+ * Configuration information about the fb which the guest can program
26
+#define PCA955X(obj) OBJECT_CHECK(PCA955xState, (obj), TYPE_PCA9552)
33
+ * via the mailbox property interface.
27
34
+ */
28
-#define PCA9552_NR_REGS 10
35
+typedef struct {
29
+#define PCA955X_NR_REGS 10
36
+ uint32_t xres, yres;
30
37
+ uint32_t xres_virtual, yres_virtual;
31
-typedef struct PCA9552State {
38
+ uint32_t xoffset, yoffset;
32
+typedef struct PCA955xState {
39
+ uint32_t bpp;
40
+ uint32_t base;
41
+ uint32_t pixo;
42
+ uint32_t alpha;
43
+} BCM2835FBConfig;
44
+
45
typedef struct {
46
/*< private >*/
33
/*< private >*/
47
SysBusDevice busdev;
34
I2CSlave i2c;
48
@@ -XXX,XX +XXX,XX @@ typedef struct {
35
/*< public >*/
49
qemu_irq mbox_irq;
36
@@ -XXX,XX +XXX,XX @@ typedef struct PCA9552State {
50
37
uint8_t len;
51
bool lock, invalidate, pending;
38
uint8_t pointer;
52
- uint32_t xres, yres;
39
53
- uint32_t xres_virtual, yres_virtual;
40
- uint8_t regs[PCA9552_NR_REGS];
54
- uint32_t xoffset, yoffset;
41
+ uint8_t regs[PCA955X_NR_REGS];
55
- uint32_t bpp;
42
uint8_t max_reg;
56
- uint32_t base, pitch, size;
43
uint8_t pin_count;
57
- uint32_t pixo, alpha;
44
-} PCA9552State;
58
+
45
+} PCA955xState;
59
+ BCM2835FBConfig config;
46
60
+
47
#endif
61
+ /* These are just cached values calculated from the config settings */
48
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
62
+ uint32_t size;
63
+ uint32_t pitch;
64
} BCM2835FBState;
65
66
void bcm2835_fb_reconfigure(BCM2835FBState *s, uint32_t *xres, uint32_t *yres,
67
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
68
index XXXXXXX..XXXXXXX 100644
49
index XXXXXXX..XXXXXXX 100644
69
--- a/hw/display/bcm2835_fb.c
50
--- a/hw/misc/pca9552.c
70
+++ b/hw/display/bcm2835_fb.c
51
+++ b/hw/misc/pca9552.c
71
@@ -XXX,XX +XXX,XX @@ static void draw_line_src16(void *opaque, uint8_t *dst, const uint8_t *src,
52
@@ -XXX,XX +XXX,XX @@
72
int bpp = surface_bits_per_pixel(surface);
53
73
54
static const char *led_state[] = {"on", "off", "pwm0", "pwm1"};
74
while (width--) {
55
75
- switch (s->bpp) {
56
-static uint8_t pca9552_pin_get_config(PCA9552State *s, int pin)
76
+ switch (s->config.bpp) {
57
+static uint8_t pca955x_pin_get_config(PCA955xState *s, int pin)
77
case 8:
58
{
78
/* lookup palette starting at video ram base
59
uint8_t reg = PCA9552_LS0 + (pin / 4);
79
* TODO: cache translation, rather than doing this each time!
60
uint8_t shift = (pin % 4) << 1;
80
@@ -XXX,XX +XXX,XX @@ static void draw_line_src16(void *opaque, uint8_t *dst, const uint8_t *src,
61
@@ -XXX,XX +XXX,XX @@ static uint8_t pca9552_pin_get_config(PCA9552State *s, int pin)
81
break;
62
return extract32(s->regs[reg], shift, 2);
82
}
63
}
83
64
84
- if (s->pixo == 0) {
65
-static void pca9552_update_pin_input(PCA9552State *s)
85
+ if (s->config.pixo == 0) {
66
+static void pca955x_update_pin_input(PCA955xState *s)
86
/* swap to BGR pixel format */
67
{
87
uint8_t tmp = r;
68
int i;
88
r = b;
69
89
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
70
for (i = 0; i < s->pin_count; i++) {
90
int src_width = 0;
71
uint8_t input_reg = PCA9552_INPUT0 + (i / 8);
91
int dest_width = 0;
72
uint8_t input_shift = (i % 8);
92
73
- uint8_t config = pca9552_pin_get_config(s, i);
93
- if (s->lock || !s->xres) {
74
+ uint8_t config = pca955x_pin_get_config(s, i);
94
+ if (s->lock || !s->config.xres) {
75
95
return;
76
switch (config) {
96
}
77
case PCA9552_LED_ON:
97
78
@@ -XXX,XX +XXX,XX @@ static void pca9552_update_pin_input(PCA9552State *s)
98
- src_width = s->xres * (s->bpp >> 3);
79
}
99
- dest_width = s->xres;
80
}
100
+ src_width = s->config.xres * (s->config.bpp >> 3);
81
101
+ dest_width = s->config.xres;
82
-static uint8_t pca9552_read(PCA9552State *s, uint8_t reg)
102
83
+static uint8_t pca955x_read(PCA955xState *s, uint8_t reg)
103
switch (surface_bits_per_pixel(surface)) {
84
{
104
case 0:
85
switch (reg) {
105
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
86
case PCA9552_INPUT0:
106
}
87
@@ -XXX,XX +XXX,XX @@ static uint8_t pca9552_read(PCA9552State *s, uint8_t reg)
107
88
}
108
if (s->invalidate) {
89
}
109
- framebuffer_update_memory_section(&s->fbsection, s->dma_mr, s->base,
90
110
- s->yres, src_width);
91
-static void pca9552_write(PCA9552State *s, uint8_t reg, uint8_t data)
111
+ framebuffer_update_memory_section(&s->fbsection, s->dma_mr,
92
+static void pca955x_write(PCA955xState *s, uint8_t reg, uint8_t data)
112
+ s->config.base,
93
{
113
+ s->config.yres, src_width);
94
switch (reg) {
114
}
95
case PCA9552_PSC0:
115
96
@@ -XXX,XX +XXX,XX @@ static void pca9552_write(PCA9552State *s, uint8_t reg, uint8_t data)
116
- framebuffer_update_display(surface, &s->fbsection, s->xres, s->yres,
97
case PCA9552_LS2:
117
+ framebuffer_update_display(surface, &s->fbsection,
98
case PCA9552_LS3:
118
+ s->config.xres, s->config.yres,
99
s->regs[reg] = data;
119
src_width, dest_width, 0, s->invalidate,
100
- pca9552_update_pin_input(s);
120
draw_line_src16, s, &first, &last);
101
+ pca955x_update_pin_input(s);
121
102
break;
122
if (first >= 0) {
103
123
- dpy_gfx_update(s->con, 0, first, s->xres, last - first + 1);
104
case PCA9552_INPUT0:
124
+ dpy_gfx_update(s->con, 0, first, s->config.xres, last - first + 1);
105
@@ -XXX,XX +XXX,XX @@ static void pca9552_write(PCA9552State *s, uint8_t reg, uint8_t data)
125
}
106
* after each byte is sent to or received by the device. The index
126
107
* rollovers to 0 when the maximum register address is reached.
127
s->invalidate = false;
108
*/
128
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
109
-static void pca9552_autoinc(PCA9552State *s)
129
110
+static void pca955x_autoinc(PCA955xState *s)
130
s->lock = true;
111
{
131
112
if (s->pointer != 0xFF && s->pointer & PCA9552_AUTOINC) {
132
- s->xres = ldl_le_phys(&s->dma_as, value);
113
uint8_t reg = s->pointer & 0xf;
133
- s->yres = ldl_le_phys(&s->dma_as, value + 4);
114
@@ -XXX,XX +XXX,XX @@ static void pca9552_autoinc(PCA9552State *s)
134
- s->xres_virtual = ldl_le_phys(&s->dma_as, value + 8);
115
}
135
- s->yres_virtual = ldl_le_phys(&s->dma_as, value + 12);
116
}
136
- s->bpp = ldl_le_phys(&s->dma_as, value + 20);
117
137
- s->xoffset = ldl_le_phys(&s->dma_as, value + 24);
118
-static uint8_t pca9552_recv(I2CSlave *i2c)
138
- s->yoffset = ldl_le_phys(&s->dma_as, value + 28);
119
+static uint8_t pca955x_recv(I2CSlave *i2c)
139
+ s->config.xres = ldl_le_phys(&s->dma_as, value);
120
{
140
+ s->config.yres = ldl_le_phys(&s->dma_as, value + 4);
121
- PCA9552State *s = PCA9552(i2c);
141
+ s->config.xres_virtual = ldl_le_phys(&s->dma_as, value + 8);
122
+ PCA955xState *s = PCA955X(i2c);
142
+ s->config.yres_virtual = ldl_le_phys(&s->dma_as, value + 12);
123
uint8_t ret;
143
+ s->config.bpp = ldl_le_phys(&s->dma_as, value + 20);
124
144
+ s->config.xoffset = ldl_le_phys(&s->dma_as, value + 24);
125
- ret = pca9552_read(s, s->pointer & 0xf);
145
+ s->config.yoffset = ldl_le_phys(&s->dma_as, value + 28);
126
+ ret = pca955x_read(s, s->pointer & 0xf);
146
127
147
- s->base = s->vcram_base | (value & 0xc0000000);
128
/*
148
- s->base += BCM2835_FB_OFFSET;
129
* From the Specs:
149
+ s->config.base = s->vcram_base | (value & 0xc0000000);
130
@@ -XXX,XX +XXX,XX @@ static uint8_t pca9552_recv(I2CSlave *i2c)
150
+ s->config.base += BCM2835_FB_OFFSET;
131
__func__);
151
132
}
152
/* TODO - Manage properly virtual resolution */
133
153
134
- pca9552_autoinc(s);
154
- s->pitch = s->xres * (s->bpp >> 3);
135
+ pca955x_autoinc(s);
155
- s->size = s->yres * s->pitch;
136
156
+ s->pitch = s->config.xres * (s->config.bpp >> 3);
137
return ret;
157
+ s->size = s->config.yres * s->pitch;
138
}
158
139
159
stl_le_phys(&s->dma_as, value + 16, s->pitch);
140
-static int pca9552_send(I2CSlave *i2c, uint8_t data)
160
- stl_le_phys(&s->dma_as, value + 32, s->base);
141
+static int pca955x_send(I2CSlave *i2c, uint8_t data)
161
+ stl_le_phys(&s->dma_as, value + 32, s->config.base);
142
{
162
stl_le_phys(&s->dma_as, value + 36, s->size);
143
- PCA9552State *s = PCA9552(i2c);
163
144
+ PCA955xState *s = PCA955X(i2c);
164
s->invalidate = true;
145
165
- qemu_console_resize(s->con, s->xres, s->yres);
146
/* First byte sent by is the register address */
166
+ qemu_console_resize(s->con, s->config.xres, s->config.yres);
147
if (s->len == 0) {
167
s->lock = false;
148
s->pointer = data;
168
}
149
s->len++;
169
150
} else {
170
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, uint32_t *xres, uint32_t *yres,
151
- pca9552_write(s, s->pointer & 0xf, data);
171
152
+ pca955x_write(s, s->pointer & 0xf, data);
172
/* TODO: input validation! */
153
173
if (xres) {
154
- pca9552_autoinc(s);
174
- s->xres = *xres;
155
+ pca955x_autoinc(s);
175
+ s->config.xres = *xres;
156
}
176
}
157
177
if (yres) {
158
return 0;
178
- s->yres = *yres;
159
}
179
+ s->config.yres = *yres;
160
180
}
161
-static int pca9552_event(I2CSlave *i2c, enum i2c_event event)
181
if (xoffset) {
162
+static int pca955x_event(I2CSlave *i2c, enum i2c_event event)
182
- s->xoffset = *xoffset;
163
{
183
+ s->config.xoffset = *xoffset;
164
- PCA9552State *s = PCA9552(i2c);
184
}
165
+ PCA955xState *s = PCA955X(i2c);
185
if (yoffset) {
166
186
- s->yoffset = *yoffset;
167
s->len = 0;
187
+ s->config.yoffset = *yoffset;
168
return 0;
188
}
169
}
189
if (bpp) {
170
190
- s->bpp = *bpp;
171
-static void pca9552_get_led(Object *obj, Visitor *v, const char *name,
191
+ s->config.bpp = *bpp;
172
+static void pca955x_get_led(Object *obj, Visitor *v, const char *name,
192
}
173
void *opaque, Error **errp)
193
if (pixo) {
174
{
194
- s->pixo = *pixo;
175
- PCA9552State *s = PCA9552(obj);
195
+ s->config.pixo = *pixo;
176
+ PCA955xState *s = PCA955X(obj);
196
}
177
int led, rc, reg;
197
if (alpha) {
178
uint8_t state;
198
- s->alpha = *alpha;
179
199
+ s->config.alpha = *alpha;
180
@@ -XXX,XX +XXX,XX @@ static void pca9552_get_led(Object *obj, Visitor *v, const char *name,
200
}
181
* reading the INPUTx reg
201
182
*/
202
/* TODO - Manage properly virtual resolution */
183
reg = PCA9552_LS0 + led / 4;
203
184
- state = (pca9552_read(s, reg) >> (led % 8)) & 0x3;
204
- s->pitch = s->xres * (s->bpp >> 3);
185
+ state = (pca955x_read(s, reg) >> (led % 8)) & 0x3;
205
- s->size = s->yres * s->pitch;
186
visit_type_str(v, name, (char **)&led_state[state], errp);
206
+ s->pitch = s->config.xres * (s->config.bpp >> 3);
187
}
207
+ s->size = s->config.yres * s->pitch;
188
208
189
@@ -XXX,XX +XXX,XX @@ static inline uint8_t pca955x_ledsel(uint8_t oldval, int led_num, int state)
209
s->invalidate = true;
190
((state & 0x3) << (led_num << 1));
210
- qemu_console_resize(s->con, s->xres, s->yres);
191
}
211
+ qemu_console_resize(s->con, s->config.xres, s->config.yres);
192
212
s->lock = false;
193
-static void pca9552_set_led(Object *obj, Visitor *v, const char *name,
213
}
194
+static void pca955x_set_led(Object *obj, Visitor *v, const char *name,
214
195
void *opaque, Error **errp)
215
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_bcm2835_fb = {
196
{
216
VMSTATE_BOOL(lock, BCM2835FBState),
197
- PCA9552State *s = PCA9552(obj);
217
VMSTATE_BOOL(invalidate, BCM2835FBState),
198
+ PCA955xState *s = PCA955X(obj);
218
VMSTATE_BOOL(pending, BCM2835FBState),
199
Error *local_err = NULL;
219
- VMSTATE_UINT32(xres, BCM2835FBState),
200
int led, rc, reg, val;
220
- VMSTATE_UINT32(yres, BCM2835FBState),
201
uint8_t state;
221
- VMSTATE_UINT32(xres_virtual, BCM2835FBState),
202
@@ -XXX,XX +XXX,XX @@ static void pca9552_set_led(Object *obj, Visitor *v, const char *name,
222
- VMSTATE_UINT32(yres_virtual, BCM2835FBState),
203
}
223
- VMSTATE_UINT32(xoffset, BCM2835FBState),
204
224
- VMSTATE_UINT32(yoffset, BCM2835FBState),
205
reg = PCA9552_LS0 + led / 4;
225
- VMSTATE_UINT32(bpp, BCM2835FBState),
206
- val = pca9552_read(s, reg);
226
- VMSTATE_UINT32(base, BCM2835FBState),
207
+ val = pca955x_read(s, reg);
227
+ VMSTATE_UINT32(config.xres, BCM2835FBState),
208
val = pca955x_ledsel(val, led % 4, state);
228
+ VMSTATE_UINT32(config.yres, BCM2835FBState),
209
- pca9552_write(s, reg, val);
229
+ VMSTATE_UINT32(config.xres_virtual, BCM2835FBState),
210
+ pca955x_write(s, reg, val);
230
+ VMSTATE_UINT32(config.yres_virtual, BCM2835FBState),
211
}
231
+ VMSTATE_UINT32(config.xoffset, BCM2835FBState),
212
232
+ VMSTATE_UINT32(config.yoffset, BCM2835FBState),
213
static const VMStateDescription pca9552_vmstate = {
233
+ VMSTATE_UINT32(config.bpp, BCM2835FBState),
214
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription pca9552_vmstate = {
234
+ VMSTATE_UINT32(config.base, BCM2835FBState),
215
.version_id = 0,
235
VMSTATE_UINT32(pitch, BCM2835FBState),
216
.minimum_version_id = 0,
236
VMSTATE_UINT32(size, BCM2835FBState),
217
.fields = (VMStateField[]) {
237
- VMSTATE_UINT32(pixo, BCM2835FBState),
218
- VMSTATE_UINT8(len, PCA9552State),
238
- VMSTATE_UINT32(alpha, BCM2835FBState),
219
- VMSTATE_UINT8(pointer, PCA9552State),
239
+ VMSTATE_UINT32(config.pixo, BCM2835FBState),
220
- VMSTATE_UINT8_ARRAY(regs, PCA9552State, PCA9552_NR_REGS),
240
+ VMSTATE_UINT32(config.alpha, BCM2835FBState),
221
- VMSTATE_I2C_SLAVE(i2c, PCA9552State),
222
+ VMSTATE_UINT8(len, PCA955xState),
223
+ VMSTATE_UINT8(pointer, PCA955xState),
224
+ VMSTATE_UINT8_ARRAY(regs, PCA955xState, PCA955X_NR_REGS),
225
+ VMSTATE_I2C_SLAVE(i2c, PCA955xState),
241
VMSTATE_END_OF_LIST()
226
VMSTATE_END_OF_LIST()
242
}
227
}
243
};
228
};
244
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_reset(DeviceState *dev)
229
245
230
static void pca9552_reset(DeviceState *dev)
246
s->pending = false;
231
{
247
232
- PCA9552State *s = PCA9552(dev);
248
- s->xres_virtual = s->xres;
233
+ PCA955xState *s = PCA955X(dev);
249
- s->yres_virtual = s->yres;
234
250
- s->xoffset = 0;
235
s->regs[PCA9552_PSC0] = 0xFF;
251
- s->yoffset = 0;
236
s->regs[PCA9552_PWM0] = 0x80;
252
- s->base = s->vcram_base + BCM2835_FB_OFFSET;
237
@@ -XXX,XX +XXX,XX @@ static void pca9552_reset(DeviceState *dev)
253
- s->pitch = s->xres * (s->bpp >> 3);
238
s->regs[PCA9552_LS2] = 0x55;
254
- s->size = s->yres * s->pitch;
239
s->regs[PCA9552_LS3] = 0x55;
255
+ s->config.xres_virtual = s->config.xres;
240
256
+ s->config.yres_virtual = s->config.yres;
241
- pca9552_update_pin_input(s);
257
+ s->config.xoffset = 0;
242
+ pca955x_update_pin_input(s);
258
+ s->config.yoffset = 0;
243
259
+ s->config.base = s->vcram_base + BCM2835_FB_OFFSET;
244
s->pointer = 0xFF;
260
+ s->pitch = s->config.xres * (s->config.bpp >> 3);
245
s->len = 0;
261
+ s->size = s->config.yres * s->pitch;
246
}
262
247
263
s->invalidate = true;
248
-static void pca9552_initfn(Object *obj)
264
s->lock = false;
249
+static void pca955x_initfn(Object *obj)
265
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_realize(DeviceState *dev, Error **errp)
250
{
266
bcm2835_fb_reset(dev);
251
- PCA9552State *s = PCA9552(obj);
267
252
+ PCA955xState *s = PCA955X(obj);
268
s->con = graphic_console_init(dev, 0, &vgafb_ops, s);
253
int led;
269
- qemu_console_resize(s->con, s->xres, s->yres);
254
270
+ qemu_console_resize(s->con, s->config.xres, s->config.yres);
255
/* If support for the other PCA955X devices are implemented, these
271
}
256
@@ -XXX,XX +XXX,XX @@ static void pca9552_initfn(Object *obj)
272
257
char *name;
273
static Property bcm2835_fb_props[] = {
258
274
DEFINE_PROP_UINT32("vcram-base", BCM2835FBState, vcram_base, 0),/*required*/
259
name = g_strdup_printf("led%d", led);
275
DEFINE_PROP_UINT32("vcram-size", BCM2835FBState, vcram_size,
260
- object_property_add(obj, name, "bool", pca9552_get_led, pca9552_set_led,
276
DEFAULT_VCRAM_SIZE),
261
+ object_property_add(obj, name, "bool", pca955x_get_led, pca955x_set_led,
277
- DEFINE_PROP_UINT32("xres", BCM2835FBState, xres, 640),
262
NULL, NULL);
278
- DEFINE_PROP_UINT32("yres", BCM2835FBState, yres, 480),
263
g_free(name);
279
- DEFINE_PROP_UINT32("bpp", BCM2835FBState, bpp, 16),
264
}
280
- DEFINE_PROP_UINT32("pixo", BCM2835FBState, pixo, 1), /* 1=RGB, 0=BGR */
265
@@ -XXX,XX +XXX,XX @@ static void pca9552_class_init(ObjectClass *klass, void *data)
281
- DEFINE_PROP_UINT32("alpha", BCM2835FBState, alpha, 2), /* alpha ignored */
266
DeviceClass *dc = DEVICE_CLASS(klass);
282
+ DEFINE_PROP_UINT32("xres", BCM2835FBState, config.xres, 640),
267
I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
283
+ DEFINE_PROP_UINT32("yres", BCM2835FBState, config.yres, 480),
268
284
+ DEFINE_PROP_UINT32("bpp", BCM2835FBState, config.bpp, 16),
269
- k->event = pca9552_event;
285
+ DEFINE_PROP_UINT32("pixo",
270
- k->recv = pca9552_recv;
286
+ BCM2835FBState, config.pixo, 1), /* 1=RGB, 0=BGR */
271
- k->send = pca9552_send;
287
+ DEFINE_PROP_UINT32("alpha",
272
+ k->event = pca955x_event;
288
+ BCM2835FBState, config.alpha, 2), /* alpha ignored */
273
+ k->recv = pca955x_recv;
289
DEFINE_PROP_END_OF_LIST()
274
+ k->send = pca955x_send;
275
dc->reset = pca9552_reset;
276
dc->vmsd = &pca9552_vmstate;
277
}
278
@@ -XXX,XX +XXX,XX @@ static void pca9552_class_init(ObjectClass *klass, void *data)
279
static const TypeInfo pca9552_info = {
280
.name = TYPE_PCA9552,
281
.parent = TYPE_I2C_SLAVE,
282
- .instance_init = pca9552_initfn,
283
- .instance_size = sizeof(PCA9552State),
284
+ .instance_init = pca955x_initfn,
285
+ .instance_size = sizeof(PCA955xState),
286
.class_init = pca9552_class_init,
290
};
287
};
291
288
292
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
289
-static void pca9552_register_types(void)
293
index XXXXXXX..XXXXXXX 100644
290
+static void pca955x_register_types(void)
294
--- a/hw/misc/bcm2835_property.c
291
{
295
+++ b/hw/misc/bcm2835_property.c
292
type_register_static(&pca9552_info);
296
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
293
}
297
/* Frame buffer */
294
298
295
-type_init(pca9552_register_types)
299
case 0x00040001: /* Allocate buffer */
296
+type_init(pca955x_register_types)
300
- stl_le_phys(&s->dma_as, value + 12, s->fbdev->base);
301
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->xres;
302
- tmp_yres = newyres != NULL ? *newyres : s->fbdev->yres;
303
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->bpp;
304
+ stl_le_phys(&s->dma_as, value + 12, s->fbdev->config.base);
305
+ tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
306
+ tmp_yres = newyres != NULL ? *newyres : s->fbdev->config.yres;
307
+ tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
308
stl_le_phys(&s->dma_as, value + 16,
309
tmp_xres * tmp_yres * tmp_bpp / 8);
310
resplen = 8;
311
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
312
break;
313
case 0x00040003: /* Get display width/height */
314
case 0x00040004:
315
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->xres;
316
- tmp_yres = newyres != NULL ? *newyres : s->fbdev->yres;
317
+ tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
318
+ tmp_yres = newyres != NULL ? *newyres : s->fbdev->config.yres;
319
stl_le_phys(&s->dma_as, value + 12, tmp_xres);
320
stl_le_phys(&s->dma_as, value + 16, tmp_yres);
321
resplen = 8;
322
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
323
resplen = 8;
324
break;
325
case 0x00040005: /* Get depth */
326
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->bpp;
327
+ tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
328
stl_le_phys(&s->dma_as, value + 12, tmp_bpp);
329
resplen = 4;
330
break;
331
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
332
resplen = 4;
333
break;
334
case 0x00040006: /* Get pixel order */
335
- tmp_pixo = newpixo != NULL ? *newpixo : s->fbdev->pixo;
336
+ tmp_pixo = newpixo != NULL ? *newpixo : s->fbdev->config.pixo;
337
stl_le_phys(&s->dma_as, value + 12, tmp_pixo);
338
resplen = 4;
339
break;
340
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
341
resplen = 4;
342
break;
343
case 0x00040007: /* Get alpha */
344
- tmp_alpha = newalpha != NULL ? *newalpha : s->fbdev->alpha;
345
+ tmp_alpha = newalpha != NULL ? *newalpha : s->fbdev->config.alpha;
346
stl_le_phys(&s->dma_as, value + 12, tmp_alpha);
347
resplen = 4;
348
break;
349
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
350
resplen = 4;
351
break;
352
case 0x00040008: /* Get pitch */
353
- tmp_xres = newxres != NULL ? *newxres : s->fbdev->xres;
354
- tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->bpp;
355
+ tmp_xres = newxres != NULL ? *newxres : s->fbdev->config.xres;
356
+ tmp_bpp = newbpp != NULL ? *newbpp : s->fbdev->config.bpp;
357
stl_le_phys(&s->dma_as, value + 12, tmp_xres * tmp_bpp / 8);
358
resplen = 4;
359
break;
360
case 0x00040009: /* Get virtual offset */
361
- tmp_xoffset = newxoffset != NULL ? *newxoffset : s->fbdev->xoffset;
362
- tmp_yoffset = newyoffset != NULL ? *newyoffset : s->fbdev->yoffset;
363
+ tmp_xoffset = newxoffset != NULL ?
364
+ *newxoffset : s->fbdev->config.xoffset;
365
+ tmp_yoffset = newyoffset != NULL ?
366
+ *newyoffset : s->fbdev->config.yoffset;
367
stl_le_phys(&s->dma_as, value + 12, tmp_xoffset);
368
stl_le_phys(&s->dma_as, value + 16, tmp_yoffset);
369
resplen = 8;
370
--
297
--
371
2.18.0
298
2.20.1
372
299
373
300
diff view generated by jsdifflib
New patch
1
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
3
Extract the code common to the PCA955x family in PCA955xClass,
4
keeping the PCA9552 specific parts into pca9552_class_init().
5
Remove the 'TODO' comment added in commit 5141d4158cf.
6
7
Suggested-by: Cédric Le Goater <clg@kaod.org>
8
Reviewed-by: Cédric Le Goater <clg@kaod.org>
9
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
10
Tested-by: Cédric Le Goater <clg@kaod.org>
11
Message-id: 20200623072723.6324-5-f4bug@amsat.org
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
14
include/hw/misc/pca9552.h | 6 ++--
15
hw/misc/pca9552.c | 66 ++++++++++++++++++++++++++++-----------
16
2 files changed, 51 insertions(+), 21 deletions(-)
17
18
diff --git a/include/hw/misc/pca9552.h b/include/hw/misc/pca9552.h
19
index XXXXXXX..XXXXXXX 100644
20
--- a/include/hw/misc/pca9552.h
21
+++ b/include/hw/misc/pca9552.h
22
@@ -XXX,XX +XXX,XX @@
23
#include "hw/i2c/i2c.h"
24
25
#define TYPE_PCA9552 "pca9552"
26
-#define PCA955X(obj) OBJECT_CHECK(PCA955xState, (obj), TYPE_PCA9552)
27
+#define TYPE_PCA955X "pca955x"
28
+#define PCA955X(obj) OBJECT_CHECK(PCA955xState, (obj), TYPE_PCA955X)
29
30
#define PCA955X_NR_REGS 10
31
+#define PCA955X_PIN_COUNT_MAX 16
32
33
typedef struct PCA955xState {
34
/*< private >*/
35
@@ -XXX,XX +XXX,XX @@ typedef struct PCA955xState {
36
uint8_t pointer;
37
38
uint8_t regs[PCA955X_NR_REGS];
39
- uint8_t max_reg;
40
- uint8_t pin_count;
41
} PCA955xState;
42
43
#endif
44
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
45
index XXXXXXX..XXXXXXX 100644
46
--- a/hw/misc/pca9552.c
47
+++ b/hw/misc/pca9552.c
48
@@ -XXX,XX +XXX,XX @@
49
* https://www.nxp.com/docs/en/application-note/AN264.pdf
50
*
51
* Copyright (c) 2017-2018, IBM Corporation.
52
+ * Copyright (c) 2020 Philippe Mathieu-Daudé
53
*
54
* This work is licensed under the terms of the GNU GPL, version 2 or
55
* later. See the COPYING file in the top-level directory.
56
@@ -XXX,XX +XXX,XX @@
57
#include "qapi/error.h"
58
#include "qapi/visitor.h"
59
60
+typedef struct PCA955xClass {
61
+ /*< private >*/
62
+ I2CSlaveClass parent_class;
63
+ /*< public >*/
64
+
65
+ uint8_t pin_count;
66
+ uint8_t max_reg;
67
+} PCA955xClass;
68
+
69
+#define PCA955X_CLASS(klass) \
70
+ OBJECT_CLASS_CHECK(PCA955xClass, (klass), TYPE_PCA955X)
71
+#define PCA955X_GET_CLASS(obj) \
72
+ OBJECT_GET_CLASS(PCA955xClass, (obj), TYPE_PCA955X)
73
+
74
#define PCA9552_LED_ON 0x0
75
#define PCA9552_LED_OFF 0x1
76
#define PCA9552_LED_PWM0 0x2
77
@@ -XXX,XX +XXX,XX @@ static uint8_t pca955x_pin_get_config(PCA955xState *s, int pin)
78
79
static void pca955x_update_pin_input(PCA955xState *s)
80
{
81
+ PCA955xClass *k = PCA955X_GET_CLASS(s);
82
int i;
83
84
- for (i = 0; i < s->pin_count; i++) {
85
+ for (i = 0; i < k->pin_count; i++) {
86
uint8_t input_reg = PCA9552_INPUT0 + (i / 8);
87
uint8_t input_shift = (i % 8);
88
uint8_t config = pca955x_pin_get_config(s, i);
89
@@ -XXX,XX +XXX,XX @@ static void pca955x_write(PCA955xState *s, uint8_t reg, uint8_t data)
90
*/
91
static void pca955x_autoinc(PCA955xState *s)
92
{
93
+ PCA955xClass *k = PCA955X_GET_CLASS(s);
94
+
95
if (s->pointer != 0xFF && s->pointer & PCA9552_AUTOINC) {
96
uint8_t reg = s->pointer & 0xf;
97
98
- reg = (reg + 1) % (s->max_reg + 1);
99
+ reg = (reg + 1) % (k->max_reg + 1);
100
s->pointer = reg | PCA9552_AUTOINC;
101
}
102
}
103
@@ -XXX,XX +XXX,XX @@ static int pca955x_event(I2CSlave *i2c, enum i2c_event event)
104
static void pca955x_get_led(Object *obj, Visitor *v, const char *name,
105
void *opaque, Error **errp)
106
{
107
+ PCA955xClass *k = PCA955X_GET_CLASS(obj);
108
PCA955xState *s = PCA955X(obj);
109
int led, rc, reg;
110
uint8_t state;
111
@@ -XXX,XX +XXX,XX @@ static void pca955x_get_led(Object *obj, Visitor *v, const char *name,
112
error_setg(errp, "%s: error reading %s", __func__, name);
113
return;
114
}
115
- if (led < 0 || led > s->pin_count) {
116
+ if (led < 0 || led > k->pin_count) {
117
error_setg(errp, "%s invalid led %s", __func__, name);
118
return;
119
}
120
@@ -XXX,XX +XXX,XX @@ static inline uint8_t pca955x_ledsel(uint8_t oldval, int led_num, int state)
121
static void pca955x_set_led(Object *obj, Visitor *v, const char *name,
122
void *opaque, Error **errp)
123
{
124
+ PCA955xClass *k = PCA955X_GET_CLASS(obj);
125
PCA955xState *s = PCA955X(obj);
126
Error *local_err = NULL;
127
int led, rc, reg, val;
128
@@ -XXX,XX +XXX,XX @@ static void pca955x_set_led(Object *obj, Visitor *v, const char *name,
129
error_setg(errp, "%s: error reading %s", __func__, name);
130
return;
131
}
132
- if (led < 0 || led > s->pin_count) {
133
+ if (led < 0 || led > k->pin_count) {
134
error_setg(errp, "%s invalid led %s", __func__, name);
135
return;
136
}
137
@@ -XXX,XX +XXX,XX @@ static void pca9552_reset(DeviceState *dev)
138
139
static void pca955x_initfn(Object *obj)
140
{
141
- PCA955xState *s = PCA955X(obj);
142
+ PCA955xClass *k = PCA955X_GET_CLASS(obj);
143
int led;
144
145
- /* If support for the other PCA955X devices are implemented, these
146
- * constant values might be part of class structure describing the
147
- * PCA955X device
148
- */
149
- s->max_reg = PCA9552_LS3;
150
- s->pin_count = 16;
151
-
152
- for (led = 0; led < s->pin_count; led++) {
153
+ assert(k->pin_count <= PCA955X_PIN_COUNT_MAX);
154
+ for (led = 0; led < k->pin_count; led++) {
155
char *name;
156
157
name = g_strdup_printf("led%d", led);
158
@@ -XXX,XX +XXX,XX @@ static void pca955x_initfn(Object *obj)
159
}
160
}
161
162
-static void pca9552_class_init(ObjectClass *klass, void *data)
163
+static void pca955x_class_init(ObjectClass *klass, void *data)
164
{
165
- DeviceClass *dc = DEVICE_CLASS(klass);
166
I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
167
168
k->event = pca955x_event;
169
k->recv = pca955x_recv;
170
k->send = pca955x_send;
171
+}
172
+
173
+static const TypeInfo pca955x_info = {
174
+ .name = TYPE_PCA955X,
175
+ .parent = TYPE_I2C_SLAVE,
176
+ .instance_init = pca955x_initfn,
177
+ .instance_size = sizeof(PCA955xState),
178
+ .class_init = pca955x_class_init,
179
+ .abstract = true,
180
+};
181
+
182
+static void pca9552_class_init(ObjectClass *oc, void *data)
183
+{
184
+ DeviceClass *dc = DEVICE_CLASS(oc);
185
+ PCA955xClass *pc = PCA955X_CLASS(oc);
186
+
187
dc->reset = pca9552_reset;
188
dc->vmsd = &pca9552_vmstate;
189
+ pc->max_reg = PCA9552_LS3;
190
+ pc->pin_count = 16;
191
}
192
193
static const TypeInfo pca9552_info = {
194
.name = TYPE_PCA9552,
195
- .parent = TYPE_I2C_SLAVE,
196
- .instance_init = pca955x_initfn,
197
- .instance_size = sizeof(PCA955xState),
198
+ .parent = TYPE_PCA955X,
199
.class_init = pca9552_class_init,
200
};
201
202
static void pca955x_register_types(void)
203
{
204
+ type_register_static(&pca955x_info);
205
type_register_static(&pca9552_info);
206
}
207
208
--
209
2.20.1
210
211
diff view generated by jsdifflib
1
The raspi framebuffir in bcm2835_fb supports the definition
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
of a virtual "viewport", which is smaller than the full
3
physical framebuffer size and at an adjustable offset within
4
it. Only the viewport area is sent to the screen. This allows
5
the guest to do things like double buffering, or scrolling
6
by adjusting the viewport origin. Currently QEMU doesn't
7
implement this at all.
8
2
9
Add support for this feature:
3
Add a description field to distinguish between multiple devices.
10
* the property mailbox code needs to distinguish the
11
virtual width/height from the physical width/height
12
* the framebuffer code needs to do something with the
13
virtual width/height/origin information
14
4
15
Note that the wiki documentation on the semantics of the
5
Reviewed-by: Cédric Le Goater <clg@kaod.org>
16
virtual and physical height and width has it the wrong way
6
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
17
around -- the virtual size is the size of the allocated
7
Tested-by: Cédric Le Goater <clg@kaod.org>
18
buffer, and the physical size is the size of the display,
8
Message-id: 20200623072723.6324-6-f4bug@amsat.org
19
so the virtual size is always the same as or larger than
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
20
the physical.
10
---
11
include/hw/misc/pca9552.h | 1 +
12
hw/misc/pca9552.c | 18 ++++++++++++++++++
13
2 files changed, 19 insertions(+)
21
14
22
If the viewport size is set smaller than the physical
15
diff --git a/include/hw/misc/pca9552.h b/include/hw/misc/pca9552.h
23
screen size, we ignore the viewport settings completely
24
and just display the physical screen area.
25
26
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
27
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
28
Message-id: 20180814144436.679-7-peter.maydell@linaro.org
29
---
30
include/hw/display/bcm2835_fb.h | 6 ++++--
31
hw/display/bcm2835_fb.c | 28 ++++++++++++++++++++++------
32
hw/misc/bcm2835_property.c | 21 +++++++++++++++------
33
3 files changed, 41 insertions(+), 14 deletions(-)
34
35
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
36
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
37
--- a/include/hw/display/bcm2835_fb.h
17
--- a/include/hw/misc/pca9552.h
38
+++ b/include/hw/display/bcm2835_fb.h
18
+++ b/include/hw/misc/pca9552.h
39
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig);
19
@@ -XXX,XX +XXX,XX @@ typedef struct PCA955xState {
40
*/
20
uint8_t pointer;
41
static inline uint32_t bcm2835_fb_get_pitch(BCM2835FBConfig *config)
21
42
{
22
uint8_t regs[PCA955X_NR_REGS];
43
- return config->xres * (config->bpp >> 3);
23
+ char *description; /* For debugging purpose only */
44
+ uint32_t xres = MAX(config->xres, config->xres_virtual);
24
} PCA955xState;
45
+ return xres * (config->bpp >> 3);
46
}
47
48
/**
49
@@ -XXX,XX +XXX,XX @@ static inline uint32_t bcm2835_fb_get_pitch(BCM2835FBConfig *config)
50
*/
51
static inline uint32_t bcm2835_fb_get_size(BCM2835FBConfig *config)
52
{
53
- return config->yres * bcm2835_fb_get_pitch(config);
54
+ uint32_t yres = MAX(config->yres, config->yres_virtual);
55
+ return yres * bcm2835_fb_get_pitch(config);
56
}
57
25
58
#endif
26
#endif
59
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
27
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
60
index XXXXXXX..XXXXXXX 100644
28
index XXXXXXX..XXXXXXX 100644
61
--- a/hw/display/bcm2835_fb.c
29
--- a/hw/misc/pca9552.c
62
+++ b/hw/display/bcm2835_fb.c
30
+++ b/hw/misc/pca9552.c
63
@@ -XXX,XX +XXX,XX @@ static void draw_line_src16(void *opaque, uint8_t *dst, const uint8_t *src,
31
@@ -XXX,XX +XXX,XX @@
32
#include "qemu/osdep.h"
33
#include "qemu/log.h"
34
#include "qemu/module.h"
35
+#include "hw/qdev-properties.h"
36
#include "hw/misc/pca9552.h"
37
#include "hw/misc/pca9552_regs.h"
38
#include "migration/vmstate.h"
39
@@ -XXX,XX +XXX,XX @@ static void pca955x_initfn(Object *obj)
64
}
40
}
65
}
41
}
66
42
67
+static bool fb_use_offsets(BCM2835FBConfig *config)
43
+static void pca955x_realize(DeviceState *dev, Error **errp)
68
+{
44
+{
69
+ /*
45
+ PCA955xState *s = PCA955X(dev);
70
+ * Return true if we should use the viewport offsets.
46
+
71
+ * Experimentally, the hardware seems to do this only if the
47
+ if (!s->description) {
72
+ * viewport size is larger than the physical screen. (It doesn't
48
+ s->description = g_strdup("pca-unspecified");
73
+ * prevent the guest setting this silly viewport setting, though...)
49
+ }
74
+ */
75
+ return config->xres_virtual > config->xres &&
76
+ config->yres_virtual > config->yres;
77
+}
50
+}
78
+
51
+
79
static void fb_update_display(void *opaque)
52
+static Property pca955x_properties[] = {
53
+ DEFINE_PROP_STRING("description", PCA955xState, description),
54
+ DEFINE_PROP_END_OF_LIST(),
55
+};
56
+
57
static void pca955x_class_init(ObjectClass *klass, void *data)
80
{
58
{
81
BCM2835FBState *s = opaque;
59
+ DeviceClass *dc = DEVICE_CLASS(klass);
82
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
60
I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
83
int last = 0;
61
84
int src_width = 0;
62
k->event = pca955x_event;
85
int dest_width = 0;
63
k->recv = pca955x_recv;
86
+ uint32_t xoff = 0, yoff = 0;
64
k->send = pca955x_send;
87
65
+ dc->realize = pca955x_realize;
88
if (s->lock || !s->config.xres) {
66
+ device_class_set_props(dc, pca955x_properties);
89
return;
67
}
90
}
68
91
69
static const TypeInfo pca955x_info = {
92
src_width = bcm2835_fb_get_pitch(&s->config);
93
+ if (fb_use_offsets(&s->config)) {
94
+ xoff = s->config.xoffset;
95
+ yoff = s->config.yoffset;
96
+ }
97
+
98
dest_width = s->config.xres;
99
100
switch (surface_bits_per_pixel(surface)) {
101
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
102
}
103
104
if (s->invalidate) {
105
+ hwaddr base = s->config.base + xoff + yoff * src_width;
106
framebuffer_update_memory_section(&s->fbsection, s->dma_mr,
107
- s->config.base,
108
+ base,
109
s->config.yres, src_width);
110
}
111
112
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
113
draw_line_src16, s, &first, &last);
114
115
if (first >= 0) {
116
- dpy_gfx_update(s->con, 0, first, s->config.xres, last - first + 1);
117
+ dpy_gfx_update(s->con, 0, first, s->config.xres,
118
+ last - first + 1);
119
}
120
121
s->invalidate = false;
122
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
123
s->config.base = s->vcram_base | (value & 0xc0000000);
124
s->config.base += BCM2835_FB_OFFSET;
125
126
- /* TODO - Manage properly virtual resolution */
127
-
128
pitch = bcm2835_fb_get_pitch(&s->config);
129
size = bcm2835_fb_get_size(&s->config);
130
131
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
132
133
s->config = *newconfig;
134
135
- /* TODO - Manage properly virtual resolution */
136
-
137
s->invalidate = true;
138
qemu_console_resize(s->con, s->config.xres, s->config.yres);
139
s->lock = false;
140
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
141
index XXXXXXX..XXXXXXX 100644
142
--- a/hw/misc/bcm2835_property.c
143
+++ b/hw/misc/bcm2835_property.c
144
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
145
case 0x00040002: /* Blank screen */
146
resplen = 4;
147
break;
148
- case 0x00040003: /* Get display width/height */
149
- case 0x00040004:
150
+ case 0x00040003: /* Get physical display width/height */
151
stl_le_phys(&s->dma_as, value + 12, fbconfig.xres);
152
stl_le_phys(&s->dma_as, value + 16, fbconfig.yres);
153
resplen = 8;
154
break;
155
- case 0x00044003: /* Test display width/height */
156
- case 0x00044004:
157
+ case 0x00040004: /* Get virtual display width/height */
158
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xres_virtual);
159
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yres_virtual);
160
resplen = 8;
161
break;
162
- case 0x00048003: /* Set display width/height */
163
- case 0x00048004:
164
+ case 0x00044003: /* Test physical display width/height */
165
+ case 0x00044004: /* Test virtual display width/height */
166
+ resplen = 8;
167
+ break;
168
+ case 0x00048003: /* Set physical display width/height */
169
fbconfig.xres = ldl_le_phys(&s->dma_as, value + 12);
170
fbconfig.yres = ldl_le_phys(&s->dma_as, value + 16);
171
fbconfig_updated = true;
172
resplen = 8;
173
break;
174
+ case 0x00048004: /* Set virtual display width/height */
175
+ fbconfig.xres_virtual = ldl_le_phys(&s->dma_as, value + 12);
176
+ fbconfig.yres_virtual = ldl_le_phys(&s->dma_as, value + 16);
177
+ fbconfig_updated = true;
178
+ resplen = 8;
179
+ break;
180
case 0x00040005: /* Get depth */
181
stl_le_phys(&s->dma_as, value + 12, fbconfig.bpp);
182
resplen = 4;
183
--
70
--
184
2.18.0
71
2.20.1
185
72
186
73
diff view generated by jsdifflib
1
The Arm IoTKit includes a system control element which
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
provides a block of read-only ID registers and a block
3
of read-write control registers. Implement a minimal
4
version of this.
5
2
3
Add a trivial representation of the PCA9552 GPIOs.
4
5
Example booting obmc-phosphor-image:
6
7
$ qemu-system-arm -M witherspoon-bmc -trace pca955x_gpio_status
8
1592689902.327837:pca955x_gpio_status pca-unspecified GPIOs 0-15 [*...............]
9
1592689902.329934:pca955x_gpio_status pca-unspecified GPIOs 0-15 [**..............]
10
1592689902.330717:pca955x_gpio_status pca-unspecified GPIOs 0-15 [***.............]
11
1592689902.331431:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****............]
12
1592689902.332163:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........*..]
13
1592689902.332888:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........**.]
14
1592689902.333629:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........***]
15
1592690032.793289:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........*.*]
16
1592690033.303163:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........***]
17
1592690033.812962:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........*.*]
18
1592690034.323234:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........***]
19
1592690034.832922:pca955x_gpio_status pca-unspecified GPIOs 0-15 [****.........*.*]
20
21
We notice the GPIO #14 (front-power LED) starts to blink.
22
23
This LED is described in the witherspoon device-tree [*]:
24
25
front-power {
26
retain-state-shutdown;
27
default-state = "keep";
28
gpios = <&pca0 14 GPIO_ACTIVE_LOW>;
29
};
30
31
[*] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/arch/arm/boot/dts/aspeed-bmc-opp-witherspoon.dts?id=b1f9be9392f0#n140
32
33
Suggested-by: Cédric Le Goater <clg@kaod.org>
34
Reviewed-by: Cédric Le Goater <clg@kaod.org>
35
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
36
Tested-by: Cédric Le Goater <clg@kaod.org>
37
Message-id: 20200623072723.6324-7-f4bug@amsat.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
38
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20180820141116.9118-9-peter.maydell@linaro.org
9
---
39
---
10
hw/misc/Makefile.objs | 1 +
40
hw/misc/pca9552.c | 39 +++++++++++++++++++++++++++++++++++++++
11
include/hw/misc/iotkit-sysctl.h | 49 ++++++
41
hw/misc/trace-events | 3 +++
12
hw/misc/iotkit-sysctl.c | 261 ++++++++++++++++++++++++++++++++
42
2 files changed, 42 insertions(+)
13
MAINTAINERS | 2 +
14
default-configs/arm-softmmu.mak | 1 +
15
hw/misc/trace-events | 7 +
16
6 files changed, 321 insertions(+)
17
create mode 100644 include/hw/misc/iotkit-sysctl.h
18
create mode 100644 hw/misc/iotkit-sysctl.c
19
43
20
diff --git a/hw/misc/Makefile.objs b/hw/misc/Makefile.objs
44
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
21
index XXXXXXX..XXXXXXX 100644
45
index XXXXXXX..XXXXXXX 100644
22
--- a/hw/misc/Makefile.objs
46
--- a/hw/misc/pca9552.c
23
+++ b/hw/misc/Makefile.objs
47
+++ b/hw/misc/pca9552.c
24
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_MPS2_SCC) += mps2-scc.o
25
obj-$(CONFIG_TZ_MPC) += tz-mpc.o
26
obj-$(CONFIG_TZ_PPC) += tz-ppc.o
27
obj-$(CONFIG_IOTKIT_SECCTL) += iotkit-secctl.o
28
+obj-$(CONFIG_IOTKIT_SYSCTL) += iotkit-sysctl.o
29
30
obj-$(CONFIG_PVPANIC) += pvpanic.o
31
obj-$(CONFIG_HYPERV_TESTDEV) += hyperv_testdev.o
32
diff --git a/include/hw/misc/iotkit-sysctl.h b/include/hw/misc/iotkit-sysctl.h
33
new file mode 100644
34
index XXXXXXX..XXXXXXX
35
--- /dev/null
36
+++ b/include/hw/misc/iotkit-sysctl.h
37
@@ -XXX,XX +XXX,XX @@
48
@@ -XXX,XX +XXX,XX @@
38
+/*
49
#include "qemu/osdep.h"
39
+ * ARM IoTKit system control element
50
#include "qemu/log.h"
40
+ *
51
#include "qemu/module.h"
41
+ * Copyright (c) 2018 Linaro Limited
52
+#include "qemu/bitops.h"
42
+ * Written by Peter Maydell
53
#include "hw/qdev-properties.h"
43
+ *
54
#include "hw/misc/pca9552.h"
44
+ * This program is free software; you can redistribute it and/or modify
55
#include "hw/misc/pca9552_regs.h"
45
+ * it under the terms of the GNU General Public License version 2 or
56
#include "migration/vmstate.h"
46
+ * (at your option) any later version.
57
#include "qapi/error.h"
47
+ */
58
#include "qapi/visitor.h"
48
+
49
+/*
50
+ * This is a model of the "system control element" which is part of the
51
+ * Arm IoTKit and documented in
52
+ * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
53
+ * Specifically, it implements the "system information block" and
54
+ * "system control register" blocks.
55
+ *
56
+ * QEMU interface:
57
+ * + sysbus MMIO region 0: the system information register bank
58
+ * + sysbus MMIO region 1: the system control register bank
59
+ */
60
+
61
+#ifndef HW_MISC_IOTKIT_SYSCTL_H
62
+#define HW_MISC_IOTKIT_SYSCTL_H
63
+
64
+#include "hw/sysbus.h"
65
+
66
+#define TYPE_IOTKIT_SYSCTL "iotkit-sysctl"
67
+#define IOTKIT_SYSCTL(obj) OBJECT_CHECK(IoTKitSysCtl, (obj), \
68
+ TYPE_IOTKIT_SYSCTL)
69
+
70
+typedef struct IoTKitSysCtl {
71
+ /*< private >*/
72
+ SysBusDevice parent_obj;
73
+
74
+ /*< public >*/
75
+ MemoryRegion iomem;
76
+
77
+ uint32_t secure_debug;
78
+ uint32_t reset_syndrome;
79
+ uint32_t reset_mask;
80
+ uint32_t gretreg;
81
+ uint32_t initsvrtor0;
82
+ uint32_t cpuwait;
83
+ uint32_t wicctrl;
84
+} IoTKitSysCtl;
85
+
86
+#endif
87
diff --git a/hw/misc/iotkit-sysctl.c b/hw/misc/iotkit-sysctl.c
88
new file mode 100644
89
index XXXXXXX..XXXXXXX
90
--- /dev/null
91
+++ b/hw/misc/iotkit-sysctl.c
92
@@ -XXX,XX +XXX,XX @@
93
+/*
94
+ * ARM IoTKit system control element
95
+ *
96
+ * Copyright (c) 2018 Linaro Limited
97
+ * Written by Peter Maydell
98
+ *
99
+ * This program is free software; you can redistribute it and/or modify
100
+ * it under the terms of the GNU General Public License version 2 or
101
+ * (at your option) any later version.
102
+ */
103
+
104
+/*
105
+ * This is a model of the "system control element" which is part of the
106
+ * Arm IoTKit and documented in
107
+ * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
108
+ * Specifically, it implements the "system control register" blocks.
109
+ */
110
+
111
+#include "qemu/osdep.h"
112
+#include "qemu/log.h"
113
+#include "trace.h"
59
+#include "trace.h"
114
+#include "qapi/error.h"
60
115
+#include "sysemu/sysemu.h"
61
typedef struct PCA955xClass {
116
+#include "hw/sysbus.h"
62
/*< private >*/
117
+#include "hw/registerfields.h"
63
@@ -XXX,XX +XXX,XX @@ static uint8_t pca955x_pin_get_config(PCA955xState *s, int pin)
118
+#include "hw/misc/iotkit-sysctl.h"
64
return extract32(s->regs[reg], shift, 2);
119
+
65
}
120
+REG32(SECDBGSTAT, 0x0)
66
121
+REG32(SECDBGSET, 0x4)
67
+/* Return INPUT status (bit #N belongs to GPIO #N) */
122
+REG32(SECDBGCLR, 0x8)
68
+static uint16_t pca955x_pins_get_status(PCA955xState *s)
123
+REG32(RESET_SYNDROME, 0x100)
124
+REG32(RESET_MASK, 0x104)
125
+REG32(SWRESET, 0x108)
126
+ FIELD(SWRESET, SWRESETREQ, 9, 1)
127
+REG32(GRETREG, 0x10c)
128
+REG32(INITSVRTOR0, 0x110)
129
+REG32(CPUWAIT, 0x118)
130
+REG32(BUSWAIT, 0x11c)
131
+REG32(WICCTRL, 0x120)
132
+REG32(PID4, 0xfd0)
133
+REG32(PID5, 0xfd4)
134
+REG32(PID6, 0xfd8)
135
+REG32(PID7, 0xfdc)
136
+REG32(PID0, 0xfe0)
137
+REG32(PID1, 0xfe4)
138
+REG32(PID2, 0xfe8)
139
+REG32(PID3, 0xfec)
140
+REG32(CID0, 0xff0)
141
+REG32(CID1, 0xff4)
142
+REG32(CID2, 0xff8)
143
+REG32(CID3, 0xffc)
144
+
145
+/* PID/CID values */
146
+static const int sysctl_id[] = {
147
+ 0x04, 0x00, 0x00, 0x00, /* PID4..PID7 */
148
+ 0x54, 0xb8, 0x0b, 0x00, /* PID0..PID3 */
149
+ 0x0d, 0xf0, 0x05, 0xb1, /* CID0..CID3 */
150
+};
151
+
152
+static uint64_t iotkit_sysctl_read(void *opaque, hwaddr offset,
153
+ unsigned size)
154
+{
69
+{
155
+ IoTKitSysCtl *s = IOTKIT_SYSCTL(opaque);
70
+ return (s->regs[PCA9552_INPUT1] << 8) | s->regs[PCA9552_INPUT0];
156
+ uint64_t r;
157
+
158
+ switch (offset) {
159
+ case A_SECDBGSTAT:
160
+ r = s->secure_debug;
161
+ break;
162
+ case A_RESET_SYNDROME:
163
+ r = s->reset_syndrome;
164
+ break;
165
+ case A_RESET_MASK:
166
+ r = s->reset_mask;
167
+ break;
168
+ case A_GRETREG:
169
+ r = s->gretreg;
170
+ break;
171
+ case A_INITSVRTOR0:
172
+ r = s->initsvrtor0;
173
+ break;
174
+ case A_CPUWAIT:
175
+ r = s->cpuwait;
176
+ break;
177
+ case A_BUSWAIT:
178
+ /* In IoTKit BUSWAIT is reserved, R/O, zero */
179
+ r = 0;
180
+ break;
181
+ case A_WICCTRL:
182
+ r = s->wicctrl;
183
+ break;
184
+ case A_PID4 ... A_CID3:
185
+ r = sysctl_id[(offset - A_PID4) / 4];
186
+ break;
187
+ case A_SECDBGSET:
188
+ case A_SECDBGCLR:
189
+ case A_SWRESET:
190
+ qemu_log_mask(LOG_GUEST_ERROR,
191
+ "IoTKit SysCtl read: read of WO offset %x\n",
192
+ (int)offset);
193
+ r = 0;
194
+ break;
195
+ default:
196
+ qemu_log_mask(LOG_GUEST_ERROR,
197
+ "IoTKit SysCtl read: bad offset %x\n", (int)offset);
198
+ r = 0;
199
+ break;
200
+ }
201
+ trace_iotkit_sysctl_read(offset, r, size);
202
+ return r;
203
+}
71
+}
204
+
72
+
205
+static void iotkit_sysctl_write(void *opaque, hwaddr offset,
73
+static void pca955x_display_pins_status(PCA955xState *s,
206
+ uint64_t value, unsigned size)
74
+ uint16_t previous_pins_status)
207
+{
75
+{
208
+ IoTKitSysCtl *s = IOTKIT_SYSCTL(opaque);
76
+ PCA955xClass *k = PCA955X_GET_CLASS(s);
77
+ uint16_t pins_status, pins_changed;
78
+ int i;
209
+
79
+
210
+ trace_iotkit_sysctl_write(offset, value, size);
80
+ pins_status = pca955x_pins_get_status(s);
81
+ pins_changed = previous_pins_status ^ pins_status;
82
+ if (!pins_changed) {
83
+ return;
84
+ }
85
+ if (trace_event_get_state_backends(TRACE_PCA955X_GPIO_STATUS)) {
86
+ char *buf = g_newa(char, k->pin_count + 1);
211
+
87
+
212
+ /*
88
+ for (i = 0; i < k->pin_count; i++) {
213
+ * Most of the state here has to do with control of reset and
89
+ if (extract32(pins_status, i, 1)) {
214
+ * similar kinds of power up -- for instance the guest can ask
90
+ buf[i] = '*';
215
+ * what the reason for the last reset was, or forbid reset for
91
+ } else {
216
+ * some causes (like the non-secure watchdog). Most of this is
92
+ buf[i] = '.';
217
+ * not relevant to QEMU, which doesn't really model anything other
93
+ }
218
+ * than a full power-on reset.
219
+ * We just model the registers as reads-as-written.
220
+ */
221
+
222
+ switch (offset) {
223
+ case A_RESET_SYNDROME:
224
+ qemu_log_mask(LOG_UNIMP,
225
+ "IoTKit SysCtl RESET_SYNDROME unimplemented\n");
226
+ s->reset_syndrome = value;
227
+ break;
228
+ case A_RESET_MASK:
229
+ qemu_log_mask(LOG_UNIMP, "IoTKit SysCtl RESET_MASK unimplemented\n");
230
+ s->reset_mask = value;
231
+ break;
232
+ case A_GRETREG:
233
+ /*
234
+ * General retention register, which is only reset by a power-on
235
+ * reset. Technically this implementation is complete, since
236
+ * QEMU only supports power-on resets...
237
+ */
238
+ s->gretreg = value;
239
+ break;
240
+ case A_INITSVRTOR0:
241
+ qemu_log_mask(LOG_UNIMP, "IoTKit SysCtl INITSVRTOR0 unimplemented\n");
242
+ s->initsvrtor0 = value;
243
+ break;
244
+ case A_CPUWAIT:
245
+ qemu_log_mask(LOG_UNIMP, "IoTKit SysCtl CPUWAIT unimplemented\n");
246
+ s->cpuwait = value;
247
+ break;
248
+ case A_WICCTRL:
249
+ qemu_log_mask(LOG_UNIMP, "IoTKit SysCtl WICCTRL unimplemented\n");
250
+ s->wicctrl = value;
251
+ break;
252
+ case A_SECDBGSET:
253
+ /* write-1-to-set */
254
+ qemu_log_mask(LOG_UNIMP, "IoTKit SysCtl SECDBGSET unimplemented\n");
255
+ s->secure_debug |= value;
256
+ break;
257
+ case A_SECDBGCLR:
258
+ /* write-1-to-clear */
259
+ s->secure_debug &= ~value;
260
+ break;
261
+ case A_SWRESET:
262
+ /* One w/o bit to request a reset; all other bits reserved */
263
+ if (value & R_SWRESET_SWRESETREQ_MASK) {
264
+ qemu_system_reset_request(SHUTDOWN_CAUSE_GUEST_RESET);
265
+ }
94
+ }
266
+ break;
95
+ buf[i] = '\0';
267
+ case A_BUSWAIT: /* In IoTKit BUSWAIT is reserved, R/O, zero */
96
+ trace_pca955x_gpio_status(s->description, buf);
268
+ case A_SECDBGSTAT:
269
+ case A_PID4 ... A_CID3:
270
+ qemu_log_mask(LOG_GUEST_ERROR,
271
+ "IoTKit SysCtl write: write of RO offset %x\n",
272
+ (int)offset);
273
+ break;
274
+ default:
275
+ qemu_log_mask(LOG_GUEST_ERROR,
276
+ "IoTKit SysCtl write: bad offset %x\n", (int)offset);
277
+ break;
278
+ }
97
+ }
279
+}
98
+}
280
+
99
+
281
+static const MemoryRegionOps iotkit_sysctl_ops = {
100
static void pca955x_update_pin_input(PCA955xState *s)
282
+ .read = iotkit_sysctl_read,
101
{
283
+ .write = iotkit_sysctl_write,
102
PCA955xClass *k = PCA955X_GET_CLASS(s);
284
+ .endianness = DEVICE_LITTLE_ENDIAN,
103
@@ -XXX,XX +XXX,XX @@ static uint8_t pca955x_read(PCA955xState *s, uint8_t reg)
285
+ /* byte/halfword accesses are just zero-padded on reads and writes */
104
286
+ .impl.min_access_size = 4,
105
static void pca955x_write(PCA955xState *s, uint8_t reg, uint8_t data)
287
+ .impl.max_access_size = 4,
106
{
288
+ .valid.min_access_size = 1,
107
+ uint16_t pins_status;
289
+ .valid.max_access_size = 4,
290
+};
291
+
108
+
292
+static void iotkit_sysctl_reset(DeviceState *dev)
109
switch (reg) {
293
+{
110
case PCA9552_PSC0:
294
+ IoTKitSysCtl *s = IOTKIT_SYSCTL(dev);
111
case PCA9552_PWM0:
295
+
112
@@ -XXX,XX +XXX,XX @@ static void pca955x_write(PCA955xState *s, uint8_t reg, uint8_t data)
296
+ trace_iotkit_sysctl_reset();
113
case PCA9552_LS1:
297
+ s->secure_debug = 0;
114
case PCA9552_LS2:
298
+ s->reset_syndrome = 1;
115
case PCA9552_LS3:
299
+ s->reset_mask = 0;
116
+ pins_status = pca955x_pins_get_status(s);
300
+ s->gretreg = 0;
117
s->regs[reg] = data;
301
+ s->initsvrtor0 = 0x10000000;
118
pca955x_update_pin_input(s);
302
+ s->cpuwait = 0;
119
+ pca955x_display_pins_status(s, pins_status);
303
+ s->wicctrl = 0;
120
break;
304
+}
121
305
+
122
case PCA9552_INPUT0:
306
+static void iotkit_sysctl_init(Object *obj)
307
+{
308
+ SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
309
+ IoTKitSysCtl *s = IOTKIT_SYSCTL(obj);
310
+
311
+ memory_region_init_io(&s->iomem, obj, &iotkit_sysctl_ops,
312
+ s, "iotkit-sysctl", 0x1000);
313
+ sysbus_init_mmio(sbd, &s->iomem);
314
+}
315
+
316
+static const VMStateDescription iotkit_sysctl_vmstate = {
317
+ .name = "iotkit-sysctl",
318
+ .version_id = 1,
319
+ .minimum_version_id = 1,
320
+ .fields = (VMStateField[]) {
321
+ VMSTATE_UINT32(secure_debug, IoTKitSysCtl),
322
+ VMSTATE_UINT32(reset_syndrome, IoTKitSysCtl),
323
+ VMSTATE_UINT32(reset_mask, IoTKitSysCtl),
324
+ VMSTATE_UINT32(gretreg, IoTKitSysCtl),
325
+ VMSTATE_UINT32(initsvrtor0, IoTKitSysCtl),
326
+ VMSTATE_UINT32(cpuwait, IoTKitSysCtl),
327
+ VMSTATE_UINT32(wicctrl, IoTKitSysCtl),
328
+ VMSTATE_END_OF_LIST()
329
+ }
330
+};
331
+
332
+static void iotkit_sysctl_class_init(ObjectClass *klass, void *data)
333
+{
334
+ DeviceClass *dc = DEVICE_CLASS(klass);
335
+
336
+ dc->vmsd = &iotkit_sysctl_vmstate;
337
+ dc->reset = iotkit_sysctl_reset;
338
+}
339
+
340
+static const TypeInfo iotkit_sysctl_info = {
341
+ .name = TYPE_IOTKIT_SYSCTL,
342
+ .parent = TYPE_SYS_BUS_DEVICE,
343
+ .instance_size = sizeof(IoTKitSysCtl),
344
+ .instance_init = iotkit_sysctl_init,
345
+ .class_init = iotkit_sysctl_class_init,
346
+};
347
+
348
+static void iotkit_sysctl_register_types(void)
349
+{
350
+ type_register_static(&iotkit_sysctl_info);
351
+}
352
+
353
+type_init(iotkit_sysctl_register_types);
354
diff --git a/MAINTAINERS b/MAINTAINERS
355
index XXXXXXX..XXXXXXX 100644
356
--- a/MAINTAINERS
357
+++ b/MAINTAINERS
358
@@ -XXX,XX +XXX,XX @@ F: hw/misc/mps2-*.c
359
F: include/hw/misc/mps2-*.h
360
F: hw/arm/iotkit.c
361
F: include/hw/arm/iotkit.h
362
+F: hw/misc/iotkit-sysctl.c
363
+F: include/hw/misc/iotkit-sysctl.h
364
365
Musicpal
366
M: Jan Kiszka <jan.kiszka@web.de>
367
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
368
index XXXXXXX..XXXXXXX 100644
369
--- a/default-configs/arm-softmmu.mak
370
+++ b/default-configs/arm-softmmu.mak
371
@@ -XXX,XX +XXX,XX @@ CONFIG_TZ_MPC=y
372
CONFIG_TZ_PPC=y
373
CONFIG_IOTKIT=y
374
CONFIG_IOTKIT_SECCTL=y
375
+CONFIG_IOTKIT_SYSCTL=y
376
377
CONFIG_VERSATILE=y
378
CONFIG_VERSATILE_PCI=y
379
diff --git a/hw/misc/trace-events b/hw/misc/trace-events
123
diff --git a/hw/misc/trace-events b/hw/misc/trace-events
380
index XXXXXXX..XXXXXXX 100644
124
index XXXXXXX..XXXXXXX 100644
381
--- a/hw/misc/trace-events
125
--- a/hw/misc/trace-events
382
+++ b/hw/misc/trace-events
126
+++ b/hw/misc/trace-events
383
@@ -XXX,XX +XXX,XX @@ ccm_freq(uint32_t freq) "freq = %d\n"
127
@@ -XXX,XX +XXX,XX @@ via1_adb_poll(uint8_t data, const char *vadbint, int status, int index, int size
384
ccm_clock_freq(uint32_t clock, uint32_t freq) "(Clock = %d) = %d\n"
128
# grlib_ahb_apb_pnp.c
385
ccm_read_reg(const char *reg_name, uint32_t value) "reg[%s] <= 0x%" PRIx32 "\n"
129
grlib_ahb_pnp_read(uint64_t addr, uint32_t value) "AHB PnP read addr:0x%03"PRIx64" data:0x%08x"
386
ccm_write_reg(const char *reg_name, uint32_t value) "reg[%s] => 0x%" PRIx32 "\n"
130
grlib_apb_pnp_read(uint64_t addr, uint32_t value) "APB PnP read addr:0x%03"PRIx64" data:0x%08x"
387
+
131
+
388
+# hw/misc/iotkit-sysctl.c
132
+# pca9552.c
389
+iotkit_sysinfo_read(uint64_t offset, uint64_t data, unsigned size) "IoTKit SysInfo read: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
133
+pca955x_gpio_status(const char *description, const char *buf) "%s GPIOs 0-15 [%s]"
390
+iotkit_sysinfo_write(uint64_t offset, uint64_t data, unsigned size) "IoTKit SysInfo write: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
391
+iotkit_sysctl_read(uint64_t offset, uint64_t data, unsigned size) "IoTKit SysCtl read: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
392
+iotkit_sysctl_write(uint64_t offset, uint64_t data, unsigned size) "IoTKit SysCtl write: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
393
+iotkit_sysctl_reset(void) "IoTKit SysCtl: reset"
394
--
134
--
395
2.18.0
135
2.20.1
396
136
397
137
diff view generated by jsdifflib
1
The PL022 interrupt registers have bits allocated as:
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
0: ROR (receive overrun)
3
1: RT (receive timeout)
4
2: RX (receive FIFO half full or less)
5
3: TX (transmit FIFO half full or less)
6
2
7
A cut and paste error meant we had the wrong value for
3
We have 2 distinct PCA9552 devices. Set their description
8
the PL022_INT_RT constant. This bug doesn't affect device
4
to distinguish them when looking at the trace events.
9
behaviour, because we don't implement the receive timeout
10
feature and so never set that interrupt bit.
11
5
6
Description name taken from:
7
https://github.com/open-power/witherspoon-xml/blob/master/witherspoon.xml
8
9
Reviewed-by: Cédric Le Goater <clg@kaod.org>
10
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
11
Reviewed-by: Corey Minyard <cminyard@mvista.com>
12
Reviewed-by: Markus Armbruster <armbru@redhat.com>
13
Tested-by: Cédric Le Goater <clg@kaod.org>
14
Message-id: 20200623072723.6324-8-f4bug@amsat.org
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Message-id: 20180820141116.9118-20-peter.maydell@linaro.org
14
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
15
---
16
---
16
hw/ssi/pl022.c | 2 +-
17
hw/arm/aspeed.c | 13 +++++++++----
17
1 file changed, 1 insertion(+), 1 deletion(-)
18
1 file changed, 9 insertions(+), 4 deletions(-)
18
19
19
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
20
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
20
index XXXXXXX..XXXXXXX 100644
21
index XXXXXXX..XXXXXXX 100644
21
--- a/hw/ssi/pl022.c
22
--- a/hw/arm/aspeed.c
22
+++ b/hw/ssi/pl022.c
23
+++ b/hw/arm/aspeed.c
23
@@ -XXX,XX +XXX,XX @@ do { fprintf(stderr, "pl022: error: " fmt , ## __VA_ARGS__);} while (0)
24
@@ -XXX,XX +XXX,XX @@ static void witherspoon_bmc_i2c_init(AspeedMachineState *bmc)
24
#define PL022_SR_BSY 0x10
25
{
25
26
AspeedSoCState *soc = &bmc->soc;
26
#define PL022_INT_ROR 0x01
27
uint8_t *eeprom_buf = g_malloc0(8 * 1024);
27
-#define PL022_INT_RT 0x04
28
+ DeviceState *dev;
28
+#define PL022_INT_RT 0x02
29
29
#define PL022_INT_RX 0x04
30
/* Bus 3: TODO bmp280@77 */
30
#define PL022_INT_TX 0x08
31
/* Bus 3: TODO max31785@52 */
32
/* Bus 3: TODO dps310@76 */
33
- i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 3), TYPE_PCA9552,
34
- 0x60);
35
+ dev = i2c_try_create_slave(TYPE_PCA9552, 0x60);
36
+ qdev_prop_set_string(dev, "description", "pca1");
37
+ i2c_realize_and_unref(dev, aspeed_i2c_get_bus(DEVICE(&soc->i2c), 3),
38
+ &error_fatal);
39
40
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 4), "tmp423", 0x4c);
41
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 5), "tmp423", 0x4c);
42
@@ -XXX,XX +XXX,XX @@ static void witherspoon_bmc_i2c_init(AspeedMachineState *bmc)
43
44
smbus_eeprom_init_one(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), 0x51,
45
eeprom_buf);
46
- i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), TYPE_PCA9552,
47
- 0x60);
48
+ dev = i2c_try_create_slave(TYPE_PCA9552, 0x60);
49
+ qdev_prop_set_string(dev, "description", "pca0");
50
+ i2c_realize_and_unref(dev, aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11),
51
+ &error_fatal);
52
/* Bus 11: TODO ucd90160@64 */
53
}
31
54
32
--
55
--
33
2.18.0
56
2.20.1
34
57
35
58
diff view generated by jsdifflib
1
For the A15MPCore internal peripheral object, we handle GIC
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
security extensions support by checking whether the CPUs
3
have EL3 enabled; if so then we enable it also on the GIC.
4
Handle the virtualization extensions in the same way: if the
5
CPU has EL2 then enable it on the GIC and wire up the
6
virtualization-specific memory regions and the maintenance
7
interrupt.
8
2
3
Emit a trace event when a GPIO change its state.
4
5
Example booting obmc-phosphor-image:
6
7
$ qemu-system-arm -M witherspoon-bmc -trace pca955x_gpio_change
8
1592690552.687372:pca955x_gpio_change pca1 GPIO id:0 status: 0 -> 1
9
1592690552.690169:pca955x_gpio_change pca1 GPIO id:1 status: 0 -> 1
10
1592690552.691673:pca955x_gpio_change pca1 GPIO id:2 status: 0 -> 1
11
1592690552.696886:pca955x_gpio_change pca1 GPIO id:3 status: 0 -> 1
12
1592690552.698614:pca955x_gpio_change pca1 GPIO id:13 status: 0 -> 1
13
1592690552.699833:pca955x_gpio_change pca1 GPIO id:14 status: 0 -> 1
14
1592690552.700842:pca955x_gpio_change pca1 GPIO id:15 status: 0 -> 1
15
1592690683.841921:pca955x_gpio_change pca1 GPIO id:14 status: 1 -> 0
16
1592690683.861660:pca955x_gpio_change pca1 GPIO id:14 status: 0 -> 1
17
1592690684.371460:pca955x_gpio_change pca1 GPIO id:14 status: 1 -> 0
18
1592690684.882115:pca955x_gpio_change pca1 GPIO id:14 status: 0 -> 1
19
1592690685.391411:pca955x_gpio_change pca1 GPIO id:14 status: 1 -> 0
20
1592690685.901391:pca955x_gpio_change pca1 GPIO id:14 status: 0 -> 1
21
1592690686.411678:pca955x_gpio_change pca1 GPIO id:14 status: 1 -> 0
22
1592690686.921279:pca955x_gpio_change pca1 GPIO id:14 status: 0 -> 1
23
24
We notice the GPIO #14 (front-power LED) starts to blink.
25
26
This LED is described in the witherspoon device-tree [*]:
27
28
front-power {
29
retain-state-shutdown;
30
default-state = "keep";
31
gpios = <&pca0 14 GPIO_ACTIVE_LOW>;
32
};
33
34
[*] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/arch/arm/boot/dts/aspeed-bmc-opp-witherspoon.dts?id=b1f9be9392f0#n140
35
36
Reviewed-by: Cédric Le Goater <clg@kaod.org>
37
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
38
Tested-by: Cédric Le Goater <clg@kaod.org>
39
Message-id: 20200623072723.6324-9-f4bug@amsat.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
40
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
11
Message-id: 20180821132811.17675-8-peter.maydell@linaro.org
12
---
41
---
13
hw/cpu/a15mpcore.c | 31 ++++++++++++++++++++++++++++---
42
hw/misc/pca9552.c | 15 +++++++++++++++
14
1 file changed, 28 insertions(+), 3 deletions(-)
43
hw/misc/trace-events | 1 +
44
2 files changed, 16 insertions(+)
15
45
16
diff --git a/hw/cpu/a15mpcore.c b/hw/cpu/a15mpcore.c
46
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
17
index XXXXXXX..XXXXXXX 100644
47
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/cpu/a15mpcore.c
48
--- a/hw/misc/pca9552.c
19
+++ b/hw/cpu/a15mpcore.c
49
+++ b/hw/misc/pca9552.c
20
@@ -XXX,XX +XXX,XX @@ static void a15mp_priv_realize(DeviceState *dev, Error **errp)
50
@@ -XXX,XX +XXX,XX @@ static void pca955x_display_pins_status(PCA955xState *s,
21
int i;
51
buf[i] = '\0';
22
Error *err = NULL;
52
trace_pca955x_gpio_status(s->description, buf);
23
bool has_el3;
24
+ bool has_el2;
25
Object *cpuobj;
26
27
gicdev = DEVICE(&s->gic);
28
@@ -XXX,XX +XXX,XX @@ static void a15mp_priv_realize(DeviceState *dev, Error **errp)
29
has_el3 = object_property_find(cpuobj, "has_el3", NULL) &&
30
object_property_get_bool(cpuobj, "has_el3", &error_abort);
31
qdev_prop_set_bit(gicdev, "has-security-extensions", has_el3);
32
+ /* Similarly for virtualization support */
33
+ has_el2 = object_property_find(cpuobj, "has_el2", NULL) &&
34
+ object_property_get_bool(cpuobj, "has_el2", &error_abort);
35
+ qdev_prop_set_bit(gicdev, "has-virtualization-extensions", has_el2);
36
}
53
}
37
54
+ if (trace_event_get_state_backends(TRACE_PCA955X_GPIO_CHANGE)) {
38
object_property_set_bool(OBJECT(&s->gic), true, "realized", &err);
55
+ for (i = 0; i < k->pin_count; i++) {
39
@@ -XXX,XX +XXX,XX @@ static void a15mp_priv_realize(DeviceState *dev, Error **errp)
56
+ if (extract32(pins_changed, i, 1)) {
40
qdev_get_gpio_in(gicdev,
57
+ unsigned new_state = extract32(pins_status, i, 1);
41
ppibase + timer_irq[irq]));
58
+
42
}
59
+ /*
43
+ if (has_el2) {
60
+ * We display the state using the PCA logic ("active-high").
44
+ /* Connect the GIC maintenance interrupt to PPI ID 25 */
61
+ * This is not the state of the LED, which signal might be
45
+ sysbus_connect_irq(SYS_BUS_DEVICE(gicdev), i + 4 * s->num_cpu,
62
+ * wired "active-low" on the board.
46
+ qdev_get_gpio_in(gicdev, ppibase + 25));
63
+ */
47
+ }
64
+ trace_pca955x_gpio_change(s->description, i,
48
}
65
+ !new_state, new_state);
49
66
+ }
50
/* Memory map (addresses are offsets from PERIPHBASE):
51
* 0x0000-0x0fff -- reserved
52
* 0x1000-0x1fff -- GIC Distributor
53
* 0x2000-0x3fff -- GIC CPU interface
54
- * 0x4000-0x4fff -- GIC virtual interface control (not modelled)
55
- * 0x5000-0x5fff -- GIC virtual interface control (not modelled)
56
- * 0x6000-0x7fff -- GIC virtual CPU interface (not modelled)
57
+ * 0x4000-0x4fff -- GIC virtual interface control for this CPU
58
+ * 0x5000-0x51ff -- GIC virtual interface control for CPU 0
59
+ * 0x5200-0x53ff -- GIC virtual interface control for CPU 1
60
+ * 0x5400-0x55ff -- GIC virtual interface control for CPU 2
61
+ * 0x5600-0x57ff -- GIC virtual interface control for CPU 3
62
+ * 0x6000-0x7fff -- GIC virtual CPU interface
63
*/
64
memory_region_add_subregion(&s->container, 0x1000,
65
sysbus_mmio_get_region(busdev, 0));
66
memory_region_add_subregion(&s->container, 0x2000,
67
sysbus_mmio_get_region(busdev, 1));
68
+ if (has_el2) {
69
+ memory_region_add_subregion(&s->container, 0x4000,
70
+ sysbus_mmio_get_region(busdev, 2));
71
+ memory_region_add_subregion(&s->container, 0x6000,
72
+ sysbus_mmio_get_region(busdev, 3));
73
+ for (i = 0; i < s->num_cpu; i++) {
74
+ hwaddr base = 0x5000 + i * 0x200;
75
+ MemoryRegion *mr = sysbus_mmio_get_region(busdev,
76
+ 4 + s->num_cpu + i);
77
+ memory_region_add_subregion(&s->container, base, mr);
78
+ }
67
+ }
79
+ }
68
+ }
80
}
69
}
81
70
82
static Property a15mp_priv_properties[] = {
71
static void pca955x_update_pin_input(PCA955xState *s)
72
diff --git a/hw/misc/trace-events b/hw/misc/trace-events
73
index XXXXXXX..XXXXXXX 100644
74
--- a/hw/misc/trace-events
75
+++ b/hw/misc/trace-events
76
@@ -XXX,XX +XXX,XX @@ grlib_apb_pnp_read(uint64_t addr, uint32_t value) "APB PnP read addr:0x%03"PRIx6
77
78
# pca9552.c
79
pca955x_gpio_status(const char *description, const char *buf) "%s GPIOs 0-15 [%s]"
80
+pca955x_gpio_change(const char *description, unsigned id, unsigned prev_state, unsigned current_state) "%s GPIO id:%u status: %u -> %u"
83
--
81
--
84
2.18.0
82
2.20.1
85
83
86
84
diff view generated by jsdifflib
1
Some of the config register values we were setting for the MPS2 SCC
1
From: Philippe Mathieu-Daudé <f4bug@amsat.org>
2
weren't correct:
3
* the SCC_AID bits [23:20] specify the FPGA build target board revision,
4
and the SCC_CFG4 register specifies the actual board revision, so
5
these should have matching values. Claim to be board revision C,
6
consistently -- we had the revision in the wrong part of SCC_AID.
7
* SCC_ID bits [15:4] should be 0x505, not decimal 505
8
2
3
The PCA9552 has 16 GPIOs which can be used as input,
4
output or PWM mode. QEMU models the output GPIO with
5
the qemu_irq type. Let the device expose the 16 GPIOs
6
to allow us to later connect LEDs to these outputs.
7
8
Reviewed-by: Cédric Le Goater <clg@kaod.org>
9
Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
10
Tested-by: Cédric Le Goater <clg@kaod.org>
11
Message-id: 20200623072723.6324-10-f4bug@amsat.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Message-id: 20180820141116.9118-23-peter.maydell@linaro.org
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
---
13
---
13
hw/arm/mps2-tz.c | 4 ++--
14
include/hw/misc/pca9552.h | 1 +
14
1 file changed, 2 insertions(+), 2 deletions(-)
15
hw/misc/pca9552.c | 6 ++++++
16
2 files changed, 7 insertions(+)
15
17
16
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
18
diff --git a/include/hw/misc/pca9552.h b/include/hw/misc/pca9552.h
17
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/mps2-tz.c
20
--- a/include/hw/misc/pca9552.h
19
+++ b/hw/arm/mps2-tz.c
21
+++ b/include/hw/misc/pca9552.h
20
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_scc(MPS2TZMachineState *mms, void *opaque,
22
@@ -XXX,XX +XXX,XX @@ typedef struct PCA955xState {
21
sccdev = DEVICE(scc);
23
uint8_t pointer;
22
qdev_set_parent_bus(sccdev, sysbus_get_default());
24
23
qdev_prop_set_uint32(sccdev, "scc-cfg4", 0x2);
25
uint8_t regs[PCA955X_NR_REGS];
24
- qdev_prop_set_uint32(sccdev, "scc-aid", 0x02000008);
26
+ qemu_irq gpio[PCA955X_PIN_COUNT_MAX];
25
+ qdev_prop_set_uint32(sccdev, "scc-aid", 0x00200008);
27
char *description; /* For debugging purpose only */
26
qdev_prop_set_uint32(sccdev, "scc-id", mmc->scc_id);
28
} PCA955xState;
27
object_property_set_bool(OBJECT(scc), true, "realized", &error_fatal);
29
28
return sysbus_mmio_get_region(SYS_BUS_DEVICE(sccdev), 0);
30
diff --git a/hw/misc/pca9552.c b/hw/misc/pca9552.c
29
@@ -XXX,XX +XXX,XX @@ static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
31
index XXXXXXX..XXXXXXX 100644
30
mc->desc = "ARM MPS2 with AN505 FPGA image for Cortex-M33";
32
--- a/hw/misc/pca9552.c
31
mmc->fpga_type = FPGA_AN505;
33
+++ b/hw/misc/pca9552.c
32
mc->default_cpu_type = ARM_CPU_TYPE_NAME("cortex-m33");
34
@@ -XXX,XX +XXX,XX @@
33
- mmc->scc_id = 0x41040000 | (505 << 4);
35
#include "hw/qdev-properties.h"
34
+ mmc->scc_id = 0x41045050;
36
#include "hw/misc/pca9552.h"
37
#include "hw/misc/pca9552_regs.h"
38
+#include "hw/irq.h"
39
#include "migration/vmstate.h"
40
#include "qapi/error.h"
41
#include "qapi/visitor.h"
42
@@ -XXX,XX +XXX,XX @@ static void pca955x_update_pin_input(PCA955xState *s)
43
44
switch (config) {
45
case PCA9552_LED_ON:
46
+ qemu_set_irq(s->gpio[i], 1);
47
s->regs[input_reg] |= 1 << input_shift;
48
break;
49
case PCA9552_LED_OFF:
50
+ qemu_set_irq(s->gpio[i], 0);
51
s->regs[input_reg] &= ~(1 << input_shift);
52
break;
53
case PCA9552_LED_PWM0:
54
@@ -XXX,XX +XXX,XX @@ static void pca955x_initfn(Object *obj)
55
56
static void pca955x_realize(DeviceState *dev, Error **errp)
57
{
58
+ PCA955xClass *k = PCA955X_GET_CLASS(dev);
59
PCA955xState *s = PCA955X(dev);
60
61
if (!s->description) {
62
s->description = g_strdup("pca-unspecified");
63
}
64
+
65
+ qdev_init_gpio_out(dev, s->gpio, k->pin_count);
35
}
66
}
36
67
37
static const TypeInfo mps2tz_info = {
68
static Property pca955x_properties[] = {
38
--
69
--
39
2.18.0
70
2.20.1
40
71
41
72
diff view generated by jsdifflib
1
The SPI controllers in the MPS2 AN505 board are PL022s.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
We have a model of the PL022, so create these devices.
3
2
4
We don't currently model the LCD controller that sits behind
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
one of the PL022s; the others are intended to control devices
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
6
that sit on the FPGA's general purpose SPI connector or
5
Message-id: 20200626033144.790098-2-richard.henderson@linaro.org
7
"shield" expansion connectors.
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
8
target/arm/cpu.h | 10 ++++++++++
9
1 file changed, 10 insertions(+)
8
10
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
10
Message-id: 20180820141116.9118-22-peter.maydell@linaro.org
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
---
13
hw/arm/mps2-tz.c | 38 ++++++++++++++++++++++++++++++++------
14
1 file changed, 32 insertions(+), 6 deletions(-)
15
16
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
17
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/mps2-tz.c
13
--- a/target/arm/cpu.h
19
+++ b/hw/arm/mps2-tz.c
14
+++ b/target/arm/cpu.h
20
@@ -XXX,XX +XXX,XX @@
15
@@ -XXX,XX +XXX,XX @@ static inline bool isar_feature_aa64_bti(const ARMISARegisters *id)
21
#include "hw/misc/tz-msc.h"
16
return FIELD_EX64(id->id_aa64pfr1, ID_AA64PFR1, BT) != 0;
22
#include "hw/arm/iotkit.h"
23
#include "hw/dma/pl080.h"
24
+#include "hw/ssi/pl022.h"
25
#include "hw/devices.h"
26
#include "net/net.h"
27
#include "hw/core/split-irq.h"
28
@@ -XXX,XX +XXX,XX @@ typedef struct {
29
MPS2FPGAIO fpgaio;
30
TZPPC ppc[5];
31
TZMPC ssram_mpc[3];
32
- UnimplementedDeviceState spi[5];
33
+ PL022State spi[5];
34
UnimplementedDeviceState i2c[4];
35
UnimplementedDeviceState i2s_audio;
36
UnimplementedDeviceState gpio[4];
37
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
38
return sysbus_mmio_get_region(s, 0);
39
}
17
}
40
18
41
+static MemoryRegion *make_spi(MPS2TZMachineState *mms, void *opaque,
19
+static inline bool isar_feature_aa64_mte_insn_reg(const ARMISARegisters *id)
42
+ const char *name, hwaddr size)
43
+{
20
+{
44
+ /*
21
+ return FIELD_EX64(id->id_aa64pfr1, ID_AA64PFR1, MTE) != 0;
45
+ * The AN505 has five PL022 SPI controllers.
46
+ * One of these should have the LCD controller behind it; the others
47
+ * are connected only to the FPGA's "general purpose SPI connector"
48
+ * or "shield" expansion connectors.
49
+ * Note that if we do implement devices behind SPI, the chip select
50
+ * lines are set via the "MISC" register in the MPS2 FPGAIO device.
51
+ */
52
+ PL022State *spi = opaque;
53
+ int i = spi - &mms->spi[0];
54
+ DeviceState *iotkitdev = DEVICE(&mms->iotkit);
55
+ SysBusDevice *s;
56
+
57
+ sysbus_init_child_obj(OBJECT(mms), name, spi, sizeof(mms->spi[0]),
58
+ TYPE_PL022);
59
+ object_property_set_bool(OBJECT(spi), true, "realized", &error_fatal);
60
+ s = SYS_BUS_DEVICE(spi);
61
+ sysbus_connect_irq(s, 0,
62
+ qdev_get_gpio_in_named(iotkitdev, "EXP_IRQ", 51 + i));
63
+ return sysbus_mmio_get_region(s, 0);
64
+}
22
+}
65
+
23
+
66
static void mps2tz_common_init(MachineState *machine)
24
+static inline bool isar_feature_aa64_mte(const ARMISARegisters *id)
25
+{
26
+ return FIELD_EX64(id->id_aa64pfr1, ID_AA64PFR1, MTE) >= 2;
27
+}
28
+
29
static inline bool isar_feature_aa64_pmu_8_1(const ARMISARegisters *id)
67
{
30
{
68
MPS2TZMachineState *mms = MPS2TZ_MACHINE(machine);
31
return FIELD_EX64(id->id_aa64dfr0, ID_AA64DFR0, PMUVER) >= 4 &&
69
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
70
}, {
71
.name = "apb_ppcexp1",
72
.ports = {
73
- { "spi0", make_unimp_dev, &mms->spi[0], 0x40205000, 0x1000 },
74
- { "spi1", make_unimp_dev, &mms->spi[1], 0x40206000, 0x1000 },
75
- { "spi2", make_unimp_dev, &mms->spi[2], 0x40209000, 0x1000 },
76
- { "spi3", make_unimp_dev, &mms->spi[3], 0x4020a000, 0x1000 },
77
- { "spi4", make_unimp_dev, &mms->spi[4], 0x4020b000, 0x1000 },
78
+ { "spi0", make_spi, &mms->spi[0], 0x40205000, 0x1000 },
79
+ { "spi1", make_spi, &mms->spi[1], 0x40206000, 0x1000 },
80
+ { "spi2", make_spi, &mms->spi[2], 0x40209000, 0x1000 },
81
+ { "spi3", make_spi, &mms->spi[3], 0x4020a000, 0x1000 },
82
+ { "spi4", make_spi, &mms->spi[4], 0x4020b000, 0x1000 },
83
{ "uart0", make_uart, &mms->uart[0], 0x40200000, 0x1000 },
84
{ "uart1", make_uart, &mms->uart[1], 0x40201000, 0x1000 },
85
{ "uart2", make_uart, &mms->uart[2], 0x40202000, 0x1000 },
86
--
32
--
87
2.18.0
33
2.20.1
88
34
89
35
diff view generated by jsdifflib
1
On 32-bit exception entry, CPSR.J must always be set to 0
1
From: Richard Henderson <richard.henderson@linaro.org>
2
(see v7A Arm ARM DDI0406C.c B1.8.5). CPSR.IL must also
3
be cleared on 32-bit exception entry (see v8A Arm ARM
4
DDI0487C.a G1.10).
5
2
6
Clear these bits. (This fixes a bug which will never be noticed
3
Protect reads of aa64 id registers with ARM_CP_STATE_AA64.
7
by non-buggy guests.)
4
Use this as a simpler test than arm_el_is_aa64, since EL3
5
cannot change mode.
8
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-3-richard.henderson@linaro.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
12
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
13
Message-id: 20180820153020.21478-6-peter.maydell@linaro.org
14
---
11
---
15
target/arm/helper.c | 2 ++
12
target/arm/helper.c | 15 ++++++++-------
16
1 file changed, 2 insertions(+)
13
1 file changed, 8 insertions(+), 7 deletions(-)
17
14
18
diff --git a/target/arm/helper.c b/target/arm/helper.c
15
diff --git a/target/arm/helper.c b/target/arm/helper.c
19
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
20
--- a/target/arm/helper.c
17
--- a/target/arm/helper.c
21
+++ b/target/arm/helper.c
18
+++ b/target/arm/helper.c
22
@@ -XXX,XX +XXX,XX @@ static void take_aarch32_exception(CPUARMState *env, int new_mode,
19
@@ -XXX,XX +XXX,XX @@ static void scr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
23
if (env->cp15.sctlr_el[arm_current_el(env)] & SCTLR_EE) {
20
uint32_t valid_mask = 0x3fff;
24
env->uncached_cpsr |= CPSR_E;
21
ARMCPU *cpu = env_archcpu(env);
22
23
- if (arm_el_is_aa64(env, 3)) {
24
+ if (ri->state == ARM_CP_STATE_AA64) {
25
value |= SCR_FW | SCR_AW; /* these two bits are RES1. */
26
valid_mask &= ~SCR_NET;
27
+
28
+ if (cpu_isar_feature(aa64_lor, cpu)) {
29
+ valid_mask |= SCR_TLOR;
30
+ }
31
+ if (cpu_isar_feature(aa64_pauth, cpu)) {
32
+ valid_mask |= SCR_API | SCR_APK;
33
+ }
34
} else {
35
valid_mask &= ~(SCR_RW | SCR_ST);
25
}
36
}
26
+ /* J and IL must always be cleared for exception entry */
37
@@ -XXX,XX +XXX,XX @@ static void scr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
27
+ env->uncached_cpsr &= ~(CPSR_IL | CPSR_J);
38
valid_mask &= ~SCR_SMD;
28
env->daif |= mask;
39
}
29
40
}
30
if (new_mode == ARM_CPU_MODE_HYP) {
41
- if (cpu_isar_feature(aa64_lor, cpu)) {
42
- valid_mask |= SCR_TLOR;
43
- }
44
- if (cpu_isar_feature(aa64_pauth, cpu)) {
45
- valid_mask |= SCR_API | SCR_APK;
46
- }
47
48
/* Clear all-context RES0 bits. */
49
value &= valid_mask;
31
--
50
--
32
2.18.0
51
2.20.1
33
52
34
53
diff view generated by jsdifflib
1
The bcm2835_fb's initial resolution and other parameters are set
1
From: Richard Henderson <richard.henderson@linaro.org>
2
via QOM properties. We should reset to those initial values on
3
device reset, which means we need to save the QOM property
4
values somewhere that they are not overwritten by guest
5
changes to the framebuffer configuration.
6
2
3
This does not attempt to rectify all of the res0 bits, but does
4
clear the mte bits when not enabled. Since there is no high-part
5
mapping of SCTLR, aa32 mode cannot write to these bits.
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-4-richard.henderson@linaro.org
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20180814144436.679-5-peter.maydell@linaro.org
10
---
11
---
11
include/hw/display/bcm2835_fb.h | 1 +
12
target/arm/helper.c | 23 +++++++++++++++++------
12
hw/display/bcm2835_fb.c | 27 +++++++++++++++------------
13
1 file changed, 17 insertions(+), 6 deletions(-)
13
2 files changed, 16 insertions(+), 12 deletions(-)
14
14
15
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
15
diff --git a/target/arm/helper.c b/target/arm/helper.c
16
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
17
--- a/include/hw/display/bcm2835_fb.h
17
--- a/target/arm/helper.c
18
+++ b/include/hw/display/bcm2835_fb.h
18
+++ b/target/arm/helper.c
19
@@ -XXX,XX +XXX,XX @@ typedef struct {
19
@@ -XXX,XX +XXX,XX @@ static void sctlr_write(CPUARMState *env, const ARMCPRegInfo *ri,
20
bool lock, invalidate, pending;
20
{
21
21
ARMCPU *cpu = env_archcpu(env);
22
BCM2835FBConfig config;
22
23
+ BCM2835FBConfig initial_config;
23
+ if (arm_feature(env, ARM_FEATURE_PMSA) && !cpu->has_mpu) {
24
} BCM2835FBState;
24
+ /* M bit is RAZ/WI for PMSA with no MPU implemented */
25
25
+ value &= ~SCTLR_M;
26
void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig);
26
+ }
27
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
27
+
28
index XXXXXXX..XXXXXXX 100644
28
+ /* ??? Lots of these bits are not implemented. */
29
--- a/hw/display/bcm2835_fb.c
29
+
30
+++ b/hw/display/bcm2835_fb.c
30
+ if (ri->state == ARM_CP_STATE_AA64 && !cpu_isar_feature(aa64_mte, cpu)) {
31
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_reset(DeviceState *dev)
31
+ if (ri->opc1 == 6) { /* SCTLR_EL3 */
32
32
+ value &= ~(SCTLR_ITFSB | SCTLR_TCF | SCTLR_ATA);
33
s->pending = false;
33
+ } else {
34
34
+ value &= ~(SCTLR_ITFSB | SCTLR_TCF0 | SCTLR_TCF |
35
- s->config.xres_virtual = s->config.xres;
35
+ SCTLR_ATA0 | SCTLR_ATA);
36
- s->config.yres_virtual = s->config.yres;
36
+ }
37
- s->config.xoffset = 0;
37
+ }
38
- s->config.yoffset = 0;
38
+
39
- s->config.base = s->vcram_base + BCM2835_FB_OFFSET;
39
if (raw_read(env, ri) == value) {
40
+ s->config = s->initial_config;
40
/* Skip the TLB flush if nothing actually changed; Linux likes
41
41
* to do a lot of pointless SCTLR writes.
42
s->invalidate = true;
42
@@ -XXX,XX +XXX,XX @@ static void sctlr_write(CPUARMState *env, const ARMCPRegInfo *ri,
43
s->lock = false;
44
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_realize(DeviceState *dev, Error **errp)
45
return;
43
return;
46
}
44
}
47
45
48
+ /* Fill in the parts of initial_config that are not set by QOM properties */
46
- if (arm_feature(env, ARM_FEATURE_PMSA) && !cpu->has_mpu) {
49
+ s->initial_config.xres_virtual = s->initial_config.xres;
47
- /* M bit is RAZ/WI for PMSA with no MPU implemented */
50
+ s->initial_config.yres_virtual = s->initial_config.yres;
48
- value &= ~SCTLR_M;
51
+ s->initial_config.xoffset = 0;
49
- }
52
+ s->initial_config.yoffset = 0;
50
-
53
+ s->initial_config.base = s->vcram_base + BCM2835_FB_OFFSET;
51
raw_write(env, ri, value);
52
- /* ??? Lots of these bits are not implemented. */
54
+
53
+
55
s->dma_mr = MEMORY_REGION(obj);
54
/* This may enable/disable the MMU, so do a TLB flush. */
56
address_space_init(&s->dma_as, s->dma_mr, NULL);
55
tlb_flush(CPU(cpu));
57
58
@@ -XXX,XX +XXX,XX @@ static Property bcm2835_fb_props[] = {
59
DEFINE_PROP_UINT32("vcram-base", BCM2835FBState, vcram_base, 0),/*required*/
60
DEFINE_PROP_UINT32("vcram-size", BCM2835FBState, vcram_size,
61
DEFAULT_VCRAM_SIZE),
62
- DEFINE_PROP_UINT32("xres", BCM2835FBState, config.xres, 640),
63
- DEFINE_PROP_UINT32("yres", BCM2835FBState, config.yres, 480),
64
- DEFINE_PROP_UINT32("bpp", BCM2835FBState, config.bpp, 16),
65
- DEFINE_PROP_UINT32("pixo",
66
- BCM2835FBState, config.pixo, 1), /* 1=RGB, 0=BGR */
67
- DEFINE_PROP_UINT32("alpha",
68
- BCM2835FBState, config.alpha, 2), /* alpha ignored */
69
+ DEFINE_PROP_UINT32("xres", BCM2835FBState, initial_config.xres, 640),
70
+ DEFINE_PROP_UINT32("yres", BCM2835FBState, initial_config.yres, 480),
71
+ DEFINE_PROP_UINT32("bpp", BCM2835FBState, initial_config.bpp, 16),
72
+ DEFINE_PROP_UINT32("pixo", BCM2835FBState,
73
+ initial_config.pixo, 1), /* 1=RGB, 0=BGR */
74
+ DEFINE_PROP_UINT32("alpha", BCM2835FBState,
75
+ initial_config.alpha, 2), /* alpha ignored */
76
DEFINE_PROP_END_OF_LIST()
77
};
78
56
79
--
57
--
80
2.18.0
58
2.20.1
81
59
82
60
diff view generated by jsdifflib
1
The v8 AArch32 HACTLR2 register maps to bits [63:32] of ACTLR_EL2.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
We implement ACTLR_EL2 as RAZ/WI, so make HACTLR2 also RAZ/WI.
3
(We put the regdef next to ACTLR_EL2 as a reminder in case we
4
ever make ACTLR_EL2 something other than RAZ/WI).
5
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-5-richard.henderson@linaro.org
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>
8
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
10
Message-id: 20180820153020.21478-2-peter.maydell@linaro.org
11
---
7
---
12
target/arm/helper.c | 10 ++++++++++
8
target/arm/helper.c | 14 +++++++++++---
13
1 file changed, 10 insertions(+)
9
1 file changed, 11 insertions(+), 3 deletions(-)
14
10
15
diff --git a/target/arm/helper.c b/target/arm/helper.c
11
diff --git a/target/arm/helper.c b/target/arm/helper.c
16
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
17
--- a/target/arm/helper.c
13
--- a/target/arm/helper.c
18
+++ b/target/arm/helper.c
14
+++ b/target/arm/helper.c
19
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
15
@@ -XXX,XX +XXX,XX @@ static void scr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
20
REGINFO_SENTINEL
16
if (cpu_isar_feature(aa64_pauth, cpu)) {
21
};
17
valid_mask |= SCR_API | SCR_APK;
22
define_arm_cp_regs(cpu, auxcr_reginfo);
18
}
23
+ if (arm_feature(env, ARM_FEATURE_V8)) {
19
+ if (cpu_isar_feature(aa64_mte, cpu)) {
24
+ /* HACTLR2 maps to ACTLR_EL2[63:32] and is not in ARMv7 */
20
+ valid_mask |= SCR_ATA;
25
+ ARMCPRegInfo hactlr2_reginfo = {
21
+ }
26
+ .name = "HACTLR2", .state = ARM_CP_STATE_AA32,
22
} else {
27
+ .cp = 15, .opc1 = 4, .crn = 1, .crm = 0, .opc2 = 3,
23
valid_mask &= ~(SCR_RW | SCR_ST);
28
+ .access = PL2_RW, .type = ARM_CP_CONST,
24
}
29
+ .resetvalue = 0
25
@@ -XXX,XX +XXX,XX @@ static void do_hcr_write(CPUARMState *env, uint64_t value, uint64_t valid_mask)
30
+ };
26
if (cpu_isar_feature(aa64_pauth, cpu)) {
31
+ define_one_arm_cp_reg(cpu, &hactlr2_reginfo);
27
valid_mask |= HCR_API | HCR_APK;
28
}
29
+ if (cpu_isar_feature(aa64_mte, cpu)) {
30
+ valid_mask |= HCR_ATA | HCR_DCT | HCR_TID5;
32
+ }
31
+ }
33
}
32
}
34
33
35
if (arm_feature(env, ARM_FEATURE_CBAR)) {
34
/* Clear RES0 bits. */
35
value &= valid_mask;
36
37
- /* These bits change the MMU setup:
38
+ /*
39
+ * These bits change the MMU setup:
40
* HCR_VM enables stage 2 translation
41
* HCR_PTW forbids certain page-table setups
42
- * HCR_DC Disables stage1 and enables stage2 translation
43
+ * HCR_DC disables stage1 and enables stage2 translation
44
+ * HCR_DCT enables tagging on (disabled) stage1 translation
45
*/
46
- if ((env->cp15.hcr_el2 ^ value) & (HCR_VM | HCR_PTW | HCR_DC)) {
47
+ if ((env->cp15.hcr_el2 ^ value) & (HCR_VM | HCR_PTW | HCR_DC | HCR_DCT)) {
48
tlb_flush(CPU(cpu));
49
}
50
env->cp15.hcr_el2 = value;
36
--
51
--
37
2.18.0
52
2.20.1
38
53
39
54
diff view generated by jsdifflib
1
Currently the PL022 calls pl022_reset() from its class init
1
From: Richard Henderson <richard.henderson@linaro.org>
2
function. Make it register a DeviceState reset method instead,
3
so that we reset the device on system reset.
4
2
3
Emphasize that the is_jmp option exits to the main loop.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-6-richard.henderson@linaro.org
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Message-id: 20180820141116.9118-17-peter.maydell@linaro.org
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
---
9
---
10
hw/ssi/pl022.c | 7 +++++--
10
target/arm/translate.h | 14 ++++++++------
11
1 file changed, 5 insertions(+), 2 deletions(-)
11
target/arm/translate-a64.c | 8 ++++----
12
target/arm/translate-vfp.inc.c | 4 ++--
13
target/arm/translate.c | 12 ++++++------
14
4 files changed, 20 insertions(+), 18 deletions(-)
12
15
13
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
16
diff --git a/target/arm/translate.h b/target/arm/translate.h
14
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/ssi/pl022.c
18
--- a/target/arm/translate.h
16
+++ b/hw/ssi/pl022.c
19
+++ b/target/arm/translate.h
17
@@ -XXX,XX +XXX,XX @@ static void pl022_write(void *opaque, hwaddr offset,
20
@@ -XXX,XX +XXX,XX @@ static inline void disas_set_insn_syndrome(DisasContext *s, uint32_t syn)
21
22
/* is_jmp field values */
23
#define DISAS_JUMP DISAS_TARGET_0 /* only pc was modified dynamically */
24
-#define DISAS_UPDATE DISAS_TARGET_1 /* cpu state was modified dynamically */
25
+/* CPU state was modified dynamically; exit to main loop for interrupts. */
26
+#define DISAS_UPDATE_EXIT DISAS_TARGET_1
27
/* These instructions trap after executing, so the A32/T32 decoder must
28
* defer them until after the conditional execution state has been updated.
29
* WFI also needs special handling when single-stepping.
30
@@ -XXX,XX +XXX,XX @@ static inline void disas_set_insn_syndrome(DisasContext *s, uint32_t syn)
31
* custom end-of-TB code)
32
*/
33
#define DISAS_BX_EXCRET DISAS_TARGET_8
34
-/* For instructions which want an immediate exit to the main loop,
35
- * as opposed to attempting to use lookup_and_goto_ptr. Unlike
36
- * DISAS_UPDATE this doesn't write the PC on exiting the translation
37
- * loop so you need to ensure something (gen_a64_set_pc_im or runtime
38
- * helper) has done so before we reach return from cpu_tb_exec.
39
+/*
40
+ * For instructions which want an immediate exit to the main loop, as opposed
41
+ * to attempting to use lookup_and_goto_ptr. Unlike DISAS_UPDATE_EXIT, this
42
+ * doesn't write the PC on exiting the translation loop so you need to ensure
43
+ * something (gen_a64_set_pc_im or runtime helper) has done so before we reach
44
+ * return from cpu_tb_exec.
45
*/
46
#define DISAS_EXIT DISAS_TARGET_9
47
48
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
49
index XXXXXXX..XXXXXXX 100644
50
--- a/target/arm/translate-a64.c
51
+++ b/target/arm/translate-a64.c
52
@@ -XXX,XX +XXX,XX @@ static void handle_msr_i(DisasContext *s, uint32_t insn,
53
gen_helper_msr_i_daifclear(cpu_env, t1);
54
tcg_temp_free_i32(t1);
55
/* For DAIFClear, exit the cpu loop to re-evaluate pending IRQs. */
56
- s->base.is_jmp = DISAS_UPDATE;
57
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
58
break;
59
60
default:
61
@@ -XXX,XX +XXX,XX @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
62
63
if ((tb_cflags(s->base.tb) & CF_USE_ICOUNT) && (ri->type & ARM_CP_IO)) {
64
/* I/O operations must end the TB here (whether read or write) */
65
- s->base.is_jmp = DISAS_UPDATE;
66
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
67
}
68
if (!isread && !(ri->type & ARM_CP_SUPPRESS_TB_END)) {
69
/*
70
@@ -XXX,XX +XXX,XX @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
71
* but allow this to be suppressed by the register definition
72
* (usually only necessary to work around guest bugs).
73
*/
74
- s->base.is_jmp = DISAS_UPDATE;
75
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
18
}
76
}
19
}
77
}
20
78
21
-static void pl022_reset(PL022State *s)
79
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
22
+static void pl022_reset(DeviceState *dev)
80
gen_goto_tb(dc, 1, dc->base.pc_next);
23
{
81
break;
24
+ PL022State *s = PL022(dev);
82
default:
25
+
83
- case DISAS_UPDATE:
26
s->rx_fifo_len = 0;
84
+ case DISAS_UPDATE_EXIT:
27
s->tx_fifo_len = 0;
85
gen_a64_set_pc_im(dc->base.pc_next);
28
s->im = 0;
86
/* fall through */
29
@@ -XXX,XX +XXX,XX @@ static int pl022_init(SysBusDevice *sbd)
87
case DISAS_EXIT:
30
sysbus_init_mmio(sbd, &s->iomem);
88
diff --git a/target/arm/translate-vfp.inc.c b/target/arm/translate-vfp.inc.c
31
sysbus_init_irq(sbd, &s->irq);
89
index XXXXXXX..XXXXXXX 100644
32
s->ssi = ssi_create_bus(dev, "ssi");
90
--- a/target/arm/translate-vfp.inc.c
33
- pl022_reset(s);
91
+++ b/target/arm/translate-vfp.inc.c
34
vmstate_register(dev, -1, &vmstate_pl022, s);
92
@@ -XXX,XX +XXX,XX @@ static bool full_vfp_access_check(DisasContext *s, bool ignore_vfp_enabled)
35
return 0;
93
* this to be the last insn in the TB).
94
*/
95
if (tb_cflags(s->base.tb) & CF_USE_ICOUNT) {
96
- s->base.is_jmp = DISAS_UPDATE;
97
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
98
gen_io_start();
99
}
100
gen_helper_v7m_preserve_fp_state(cpu_env);
101
@@ -XXX,XX +XXX,XX @@ static bool trans_VLLDM_VLSTM(DisasContext *s, arg_VLLDM_VLSTM *a)
102
tcg_temp_free_i32(fptr);
103
104
/* End the TB, because we have updated FP control bits */
105
- s->base.is_jmp = DISAS_UPDATE;
106
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
107
return true;
36
}
108
}
37
@@ -XXX,XX +XXX,XX @@ static int pl022_init(SysBusDevice *sbd)
109
diff --git a/target/arm/translate.c b/target/arm/translate.c
38
static void pl022_class_init(ObjectClass *klass, void *data)
110
index XXXXXXX..XXXXXXX 100644
39
{
111
--- a/target/arm/translate.c
40
SysBusDeviceClass *sdc = SYS_BUS_DEVICE_CLASS(klass);
112
+++ b/target/arm/translate.c
41
+ DeviceClass *dc = DEVICE_CLASS(klass);
113
@@ -XXX,XX +XXX,XX @@ static void gen_msr_banked(DisasContext *s, int r, int sysm, int rn)
42
114
tcg_temp_free_i32(tcg_tgtmode);
43
sdc->init = pl022_init;
115
tcg_temp_free_i32(tcg_regno);
44
+ dc->reset = pl022_reset;
116
tcg_temp_free_i32(tcg_reg);
117
- s->base.is_jmp = DISAS_UPDATE;
118
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
45
}
119
}
46
120
47
static const TypeInfo pl022_info = {
121
static void gen_mrs_banked(DisasContext *s, int r, int sysm, int rn)
122
@@ -XXX,XX +XXX,XX @@ static void gen_mrs_banked(DisasContext *s, int r, int sysm, int rn)
123
tcg_temp_free_i32(tcg_tgtmode);
124
tcg_temp_free_i32(tcg_regno);
125
store_reg(s, rn, tcg_reg);
126
- s->base.is_jmp = DISAS_UPDATE;
127
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
128
}
129
130
/* Store value to PC as for an exception return (ie don't
131
@@ -XXX,XX +XXX,XX @@ static void gen_srs(DisasContext *s,
132
tcg_temp_free_i32(tmp);
133
}
134
tcg_temp_free_i32(addr);
135
- s->base.is_jmp = DISAS_UPDATE;
136
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
137
}
138
139
/* Generate a label used for skipping this instruction */
140
@@ -XXX,XX +XXX,XX @@ static bool trans_SETEND(DisasContext *s, arg_SETEND *a)
141
}
142
if (a->E != (s->be_data == MO_BE)) {
143
gen_helper_setend(cpu_env);
144
- s->base.is_jmp = DISAS_UPDATE;
145
+ s->base.is_jmp = DISAS_UPDATE_EXIT;
146
}
147
return true;
148
}
149
@@ -XXX,XX +XXX,XX @@ static void arm_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
150
break;
151
case DISAS_NEXT:
152
case DISAS_TOO_MANY:
153
- case DISAS_UPDATE:
154
+ case DISAS_UPDATE_EXIT:
155
gen_set_pc_im(dc, dc->base.pc_next);
156
/* fall through */
157
default:
158
@@ -XXX,XX +XXX,XX @@ static void arm_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
159
case DISAS_JUMP:
160
gen_goto_ptr();
161
break;
162
- case DISAS_UPDATE:
163
+ case DISAS_UPDATE_EXIT:
164
gen_set_pc_im(dc, dc->base.pc_next);
165
/* fall through */
166
default:
48
--
167
--
49
2.18.0
168
2.20.1
50
169
51
170
diff view generated by jsdifflib
1
Untabify the arm translate.c. This affects only some lines,
1
From: Richard Henderson <richard.henderson@linaro.org>
2
mostly comments, in the iwMMXt code. We've never touched
3
that code in years, so it's not going to get fixed up
4
by our "change when touched" process, and a bulk change
5
is not going to be too disruptive.
6
2
7
This commit was produced using Emacs "untabify"; it is
3
Add an option that writes back the PC, like DISAS_UPDATE_EXIT,
8
a whitespace-only change.
4
but does not exit back to the main loop.
9
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-7-richard.henderson@linaro.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Message-id: 20180821165215.29069-2-peter.maydell@linaro.org
12
---
10
---
13
target/arm/translate.c | 122 ++++++++++++++++++++---------------------
11
target/arm/translate.h | 2 ++
14
1 file changed, 61 insertions(+), 61 deletions(-)
12
target/arm/translate-a64.c | 3 +++
13
target/arm/translate.c | 4 ++++
14
3 files changed, 9 insertions(+)
15
15
16
diff --git a/target/arm/translate.h b/target/arm/translate.h
17
index XXXXXXX..XXXXXXX 100644
18
--- a/target/arm/translate.h
19
+++ b/target/arm/translate.h
20
@@ -XXX,XX +XXX,XX @@ static inline void disas_set_insn_syndrome(DisasContext *s, uint32_t syn)
21
* return from cpu_tb_exec.
22
*/
23
#define DISAS_EXIT DISAS_TARGET_9
24
+/* CPU state was modified dynamically; no need to exit, but do not chain. */
25
+#define DISAS_UPDATE_NOCHAIN DISAS_TARGET_10
26
27
#ifdef TARGET_AARCH64
28
void a64_translate_init(void);
29
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
30
index XXXXXXX..XXXXXXX 100644
31
--- a/target/arm/translate-a64.c
32
+++ b/target/arm/translate-a64.c
33
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
34
case DISAS_EXIT:
35
tcg_gen_exit_tb(NULL, 0);
36
break;
37
+ case DISAS_UPDATE_NOCHAIN:
38
+ gen_a64_set_pc_im(dc->base.pc_next);
39
+ /* fall through */
40
case DISAS_JUMP:
41
tcg_gen_lookup_and_goto_ptr();
42
break;
16
diff --git a/target/arm/translate.c b/target/arm/translate.c
43
diff --git a/target/arm/translate.c b/target/arm/translate.c
17
index XXXXXXX..XXXXXXX 100644
44
index XXXXXXX..XXXXXXX 100644
18
--- a/target/arm/translate.c
45
--- a/target/arm/translate.c
19
+++ b/target/arm/translate.c
46
+++ b/target/arm/translate.c
20
@@ -XXX,XX +XXX,XX @@ static inline void gen_mov_vreg_F0(int dp, int reg)
47
@@ -XXX,XX +XXX,XX @@ static void arm_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
21
tcg_gen_st_f32(cpu_F0s, cpu_env, vfp_reg_offset(dp, reg));
48
case DISAS_NEXT:
22
}
49
case DISAS_TOO_MANY:
23
50
case DISAS_UPDATE_EXIT:
24
-#define ARM_CP_RW_BIT    (1 << 20)
51
+ case DISAS_UPDATE_NOCHAIN:
25
+#define ARM_CP_RW_BIT (1 << 20)
52
gen_set_pc_im(dc, dc->base.pc_next);
26
53
/* fall through */
27
static inline void iwmmxt_load_reg(TCGv_i64 var, int reg)
54
default:
28
{
55
@@ -XXX,XX +XXX,XX @@ static void arm_tr_tb_stop(DisasContextBase *dcbase, CPUState *cpu)
29
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
56
case DISAS_TOO_MANY:
30
wrd = insn & 0xf;
57
gen_goto_tb(dc, 1, dc->base.pc_next);
31
rdlo = (insn >> 12) & 0xf;
32
rdhi = (insn >> 16) & 0xf;
33
- if (insn & ARM_CP_RW_BIT) {            /* TMRRC */
34
+ if (insn & ARM_CP_RW_BIT) { /* TMRRC */
35
iwmmxt_load_reg(cpu_V0, wrd);
36
tcg_gen_extrl_i64_i32(cpu_R[rdlo], cpu_V0);
37
tcg_gen_shri_i64(cpu_V0, cpu_V0, 32);
38
tcg_gen_extrl_i64_i32(cpu_R[rdhi], cpu_V0);
39
- } else {                    /* TMCRR */
40
+ } else { /* TMCRR */
41
tcg_gen_concat_i32_i64(cpu_V0, cpu_R[rdlo], cpu_R[rdhi]);
42
iwmmxt_store_reg(cpu_V0, wrd);
43
gen_op_iwmmxt_set_mup();
44
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
45
return 1;
46
}
47
if (insn & ARM_CP_RW_BIT) {
48
- if ((insn >> 28) == 0xf) {            /* WLDRW wCx */
49
+ if ((insn >> 28) == 0xf) { /* WLDRW wCx */
50
tmp = tcg_temp_new_i32();
51
gen_aa32_ld32u(s, tmp, addr, get_mem_index(s));
52
iwmmxt_store_creg(wrd, tmp);
53
} else {
54
i = 1;
55
if (insn & (1 << 8)) {
56
- if (insn & (1 << 22)) {        /* WLDRD */
57
+ if (insn & (1 << 22)) { /* WLDRD */
58
gen_aa32_ld64(s, cpu_M0, addr, get_mem_index(s));
59
i = 0;
60
- } else {                /* WLDRW wRd */
61
+ } else { /* WLDRW wRd */
62
tmp = tcg_temp_new_i32();
63
gen_aa32_ld32u(s, tmp, addr, get_mem_index(s));
64
}
65
} else {
66
tmp = tcg_temp_new_i32();
67
- if (insn & (1 << 22)) {        /* WLDRH */
68
+ if (insn & (1 << 22)) { /* WLDRH */
69
gen_aa32_ld16u(s, tmp, addr, get_mem_index(s));
70
- } else {                /* WLDRB */
71
+ } else { /* WLDRB */
72
gen_aa32_ld8u(s, tmp, addr, get_mem_index(s));
73
}
74
}
75
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
76
gen_op_iwmmxt_movq_wRn_M0(wrd);
77
}
78
} else {
79
- if ((insn >> 28) == 0xf) {            /* WSTRW wCx */
80
+ if ((insn >> 28) == 0xf) { /* WSTRW wCx */
81
tmp = iwmmxt_load_creg(wrd);
82
gen_aa32_st32(s, tmp, addr, get_mem_index(s));
83
} else {
84
gen_op_iwmmxt_movq_M0_wRn(wrd);
85
tmp = tcg_temp_new_i32();
86
if (insn & (1 << 8)) {
87
- if (insn & (1 << 22)) {        /* WSTRD */
88
+ if (insn & (1 << 22)) { /* WSTRD */
89
gen_aa32_st64(s, cpu_M0, addr, get_mem_index(s));
90
- } else {                /* WSTRW wRd */
91
+ } else { /* WSTRW wRd */
92
tcg_gen_extrl_i64_i32(tmp, cpu_M0);
93
gen_aa32_st32(s, tmp, addr, get_mem_index(s));
94
}
95
} else {
96
- if (insn & (1 << 22)) {        /* WSTRH */
97
+ if (insn & (1 << 22)) { /* WSTRH */
98
tcg_gen_extrl_i64_i32(tmp, cpu_M0);
99
gen_aa32_st16(s, tmp, addr, get_mem_index(s));
100
- } else {                /* WSTRB */
101
+ } else { /* WSTRB */
102
tcg_gen_extrl_i64_i32(tmp, cpu_M0);
103
gen_aa32_st8(s, tmp, addr, get_mem_index(s));
104
}
105
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
106
return 1;
107
108
switch (((insn >> 12) & 0xf00) | ((insn >> 4) & 0xff)) {
109
- case 0x000:                        /* WOR */
110
+ case 0x000: /* WOR */
111
wrd = (insn >> 12) & 0xf;
112
rd0 = (insn >> 0) & 0xf;
113
rd1 = (insn >> 16) & 0xf;
114
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
115
gen_op_iwmmxt_set_mup();
116
gen_op_iwmmxt_set_cup();
117
break;
118
- case 0x011:                        /* TMCR */
119
+ case 0x011: /* TMCR */
120
if (insn & 0xf)
121
return 1;
122
rd = (insn >> 12) & 0xf;
123
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
124
return 1;
125
}
126
break;
127
- case 0x100:                        /* WXOR */
128
+ case 0x100: /* WXOR */
129
wrd = (insn >> 12) & 0xf;
130
rd0 = (insn >> 0) & 0xf;
131
rd1 = (insn >> 16) & 0xf;
132
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
133
gen_op_iwmmxt_set_mup();
134
gen_op_iwmmxt_set_cup();
135
break;
136
- case 0x111:                        /* TMRC */
137
+ case 0x111: /* TMRC */
138
if (insn & 0xf)
139
return 1;
140
rd = (insn >> 12) & 0xf;
141
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
142
tmp = iwmmxt_load_creg(wrd);
143
store_reg(s, rd, tmp);
144
break;
145
- case 0x300:                        /* WANDN */
146
+ case 0x300: /* WANDN */
147
wrd = (insn >> 12) & 0xf;
148
rd0 = (insn >> 0) & 0xf;
149
rd1 = (insn >> 16) & 0xf;
150
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
151
gen_op_iwmmxt_set_mup();
152
gen_op_iwmmxt_set_cup();
153
break;
154
- case 0x200:                        /* WAND */
155
+ case 0x200: /* WAND */
156
wrd = (insn >> 12) & 0xf;
157
rd0 = (insn >> 0) & 0xf;
158
rd1 = (insn >> 16) & 0xf;
159
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
160
gen_op_iwmmxt_set_mup();
161
gen_op_iwmmxt_set_cup();
162
break;
163
- case 0x810: case 0xa10:                /* WMADD */
164
+ case 0x810: case 0xa10: /* WMADD */
165
wrd = (insn >> 12) & 0xf;
166
rd0 = (insn >> 0) & 0xf;
167
rd1 = (insn >> 16) & 0xf;
168
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
169
gen_op_iwmmxt_movq_wRn_M0(wrd);
170
gen_op_iwmmxt_set_mup();
171
break;
172
- case 0x10e: case 0x50e: case 0x90e: case 0xd0e:    /* WUNPCKIL */
173
+ case 0x10e: case 0x50e: case 0x90e: case 0xd0e: /* WUNPCKIL */
174
wrd = (insn >> 12) & 0xf;
175
rd0 = (insn >> 16) & 0xf;
176
rd1 = (insn >> 0) & 0xf;
177
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
178
gen_op_iwmmxt_set_mup();
179
gen_op_iwmmxt_set_cup();
180
break;
181
- case 0x10c: case 0x50c: case 0x90c: case 0xd0c:    /* WUNPCKIH */
182
+ case 0x10c: case 0x50c: case 0x90c: case 0xd0c: /* WUNPCKIH */
183
wrd = (insn >> 12) & 0xf;
184
rd0 = (insn >> 16) & 0xf;
185
rd1 = (insn >> 0) & 0xf;
186
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
187
gen_op_iwmmxt_set_mup();
188
gen_op_iwmmxt_set_cup();
189
break;
190
- case 0x012: case 0x112: case 0x412: case 0x512:    /* WSAD */
191
+ case 0x012: case 0x112: case 0x412: case 0x512: /* WSAD */
192
wrd = (insn >> 12) & 0xf;
193
rd0 = (insn >> 16) & 0xf;
194
rd1 = (insn >> 0) & 0xf;
195
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
196
gen_op_iwmmxt_movq_wRn_M0(wrd);
197
gen_op_iwmmxt_set_mup();
198
break;
199
- case 0x010: case 0x110: case 0x210: case 0x310:    /* WMUL */
200
+ case 0x010: case 0x110: case 0x210: case 0x310: /* WMUL */
201
wrd = (insn >> 12) & 0xf;
202
rd0 = (insn >> 16) & 0xf;
203
rd1 = (insn >> 0) & 0xf;
204
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
205
gen_op_iwmmxt_movq_wRn_M0(wrd);
206
gen_op_iwmmxt_set_mup();
207
break;
208
- case 0x410: case 0x510: case 0x610: case 0x710:    /* WMAC */
209
+ case 0x410: case 0x510: case 0x610: case 0x710: /* WMAC */
210
wrd = (insn >> 12) & 0xf;
211
rd0 = (insn >> 16) & 0xf;
212
rd1 = (insn >> 0) & 0xf;
213
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
214
gen_op_iwmmxt_movq_wRn_M0(wrd);
215
gen_op_iwmmxt_set_mup();
216
break;
217
- case 0x006: case 0x406: case 0x806: case 0xc06:    /* WCMPEQ */
218
+ case 0x006: case 0x406: case 0x806: case 0xc06: /* WCMPEQ */
219
wrd = (insn >> 12) & 0xf;
220
rd0 = (insn >> 16) & 0xf;
221
rd1 = (insn >> 0) & 0xf;
222
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
223
gen_op_iwmmxt_set_mup();
224
gen_op_iwmmxt_set_cup();
225
break;
226
- case 0x800: case 0x900: case 0xc00: case 0xd00:    /* WAVG2 */
227
+ case 0x800: case 0x900: case 0xc00: case 0xd00: /* WAVG2 */
228
wrd = (insn >> 12) & 0xf;
229
rd0 = (insn >> 16) & 0xf;
230
rd1 = (insn >> 0) & 0xf;
231
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
232
gen_op_iwmmxt_set_mup();
233
gen_op_iwmmxt_set_cup();
234
break;
235
- case 0x802: case 0x902: case 0xa02: case 0xb02:    /* WALIGNR */
236
+ case 0x802: case 0x902: case 0xa02: case 0xb02: /* WALIGNR */
237
wrd = (insn >> 12) & 0xf;
238
rd0 = (insn >> 16) & 0xf;
239
rd1 = (insn >> 0) & 0xf;
240
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
241
gen_op_iwmmxt_movq_wRn_M0(wrd);
242
gen_op_iwmmxt_set_mup();
243
break;
244
- case 0x601: case 0x605: case 0x609: case 0x60d:    /* TINSR */
245
+ case 0x601: case 0x605: case 0x609: case 0x60d: /* TINSR */
246
if (((insn >> 6) & 3) == 3)
247
return 1;
248
rd = (insn >> 12) & 0xf;
249
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
250
gen_op_iwmmxt_movq_wRn_M0(wrd);
251
gen_op_iwmmxt_set_mup();
252
break;
253
- case 0x107: case 0x507: case 0x907: case 0xd07:    /* TEXTRM */
254
+ case 0x107: case 0x507: case 0x907: case 0xd07: /* TEXTRM */
255
rd = (insn >> 12) & 0xf;
256
wrd = (insn >> 16) & 0xf;
257
if (rd == 15 || ((insn >> 22) & 3) == 3)
258
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
259
}
260
store_reg(s, rd, tmp);
261
break;
262
- case 0x117: case 0x517: case 0x917: case 0xd17:    /* TEXTRC */
263
+ case 0x117: case 0x517: case 0x917: case 0xd17: /* TEXTRC */
264
if ((insn & 0x000ff008) != 0x0003f000 || ((insn >> 22) & 3) == 3)
265
return 1;
266
tmp = iwmmxt_load_creg(ARM_IWMMXT_wCASF);
267
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
268
gen_set_nzcv(tmp);
269
tcg_temp_free_i32(tmp);
270
break;
271
- case 0x401: case 0x405: case 0x409: case 0x40d:    /* TBCST */
272
+ case 0x401: case 0x405: case 0x409: case 0x40d: /* TBCST */
273
if (((insn >> 6) & 3) == 3)
274
return 1;
275
rd = (insn >> 12) & 0xf;
276
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
277
gen_op_iwmmxt_movq_wRn_M0(wrd);
278
gen_op_iwmmxt_set_mup();
279
break;
280
- case 0x113: case 0x513: case 0x913: case 0xd13:    /* TANDC */
281
+ case 0x113: case 0x513: case 0x913: case 0xd13: /* TANDC */
282
if ((insn & 0x000ff00f) != 0x0003f000 || ((insn >> 22) & 3) == 3)
283
return 1;
284
tmp = iwmmxt_load_creg(ARM_IWMMXT_wCASF);
285
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
286
tcg_temp_free_i32(tmp2);
287
tcg_temp_free_i32(tmp);
288
break;
289
- case 0x01c: case 0x41c: case 0x81c: case 0xc1c:    /* WACC */
290
+ case 0x01c: case 0x41c: case 0x81c: case 0xc1c: /* WACC */
291
wrd = (insn >> 12) & 0xf;
292
rd0 = (insn >> 16) & 0xf;
293
gen_op_iwmmxt_movq_M0_wRn(rd0);
294
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
295
gen_op_iwmmxt_movq_wRn_M0(wrd);
296
gen_op_iwmmxt_set_mup();
297
break;
298
- case 0x115: case 0x515: case 0x915: case 0xd15:    /* TORC */
299
+ case 0x115: case 0x515: case 0x915: case 0xd15: /* TORC */
300
if ((insn & 0x000ff00f) != 0x0003f000 || ((insn >> 22) & 3) == 3)
301
return 1;
302
tmp = iwmmxt_load_creg(ARM_IWMMXT_wCASF);
303
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
304
tcg_temp_free_i32(tmp2);
305
tcg_temp_free_i32(tmp);
306
break;
307
- case 0x103: case 0x503: case 0x903: case 0xd03:    /* TMOVMSK */
308
+ case 0x103: case 0x503: case 0x903: case 0xd03: /* TMOVMSK */
309
rd = (insn >> 12) & 0xf;
310
rd0 = (insn >> 16) & 0xf;
311
if ((insn & 0xf) != 0 || ((insn >> 22) & 3) == 3)
312
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
313
}
314
store_reg(s, rd, tmp);
315
break;
316
- case 0x106: case 0x306: case 0x506: case 0x706:    /* WCMPGT */
317
+ case 0x106: case 0x306: case 0x506: case 0x706: /* WCMPGT */
318
case 0x906: case 0xb06: case 0xd06: case 0xf06:
319
wrd = (insn >> 12) & 0xf;
320
rd0 = (insn >> 16) & 0xf;
321
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
322
gen_op_iwmmxt_set_mup();
323
gen_op_iwmmxt_set_cup();
324
break;
325
- case 0x00e: case 0x20e: case 0x40e: case 0x60e:    /* WUNPCKEL */
326
+ case 0x00e: case 0x20e: case 0x40e: case 0x60e: /* WUNPCKEL */
327
case 0x80e: case 0xa0e: case 0xc0e: case 0xe0e:
328
wrd = (insn >> 12) & 0xf;
329
rd0 = (insn >> 16) & 0xf;
330
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
331
gen_op_iwmmxt_set_mup();
332
gen_op_iwmmxt_set_cup();
333
break;
334
- case 0x00c: case 0x20c: case 0x40c: case 0x60c:    /* WUNPCKEH */
335
+ case 0x00c: case 0x20c: case 0x40c: case 0x60c: /* WUNPCKEH */
336
case 0x80c: case 0xa0c: case 0xc0c: case 0xe0c:
337
wrd = (insn >> 12) & 0xf;
338
rd0 = (insn >> 16) & 0xf;
339
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
340
gen_op_iwmmxt_set_mup();
341
gen_op_iwmmxt_set_cup();
342
break;
343
- case 0x204: case 0x604: case 0xa04: case 0xe04:    /* WSRL */
344
+ case 0x204: case 0x604: case 0xa04: case 0xe04: /* WSRL */
345
case 0x214: case 0x614: case 0xa14: case 0xe14:
346
if (((insn >> 22) & 3) == 0)
347
return 1;
348
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
349
gen_op_iwmmxt_set_mup();
350
gen_op_iwmmxt_set_cup();
351
break;
352
- case 0x004: case 0x404: case 0x804: case 0xc04:    /* WSRA */
353
+ case 0x004: case 0x404: case 0x804: case 0xc04: /* WSRA */
354
case 0x014: case 0x414: case 0x814: case 0xc14:
355
if (((insn >> 22) & 3) == 0)
356
return 1;
357
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
358
gen_op_iwmmxt_set_mup();
359
gen_op_iwmmxt_set_cup();
360
break;
361
- case 0x104: case 0x504: case 0x904: case 0xd04:    /* WSLL */
362
+ case 0x104: case 0x504: case 0x904: case 0xd04: /* WSLL */
363
case 0x114: case 0x514: case 0x914: case 0xd14:
364
if (((insn >> 22) & 3) == 0)
365
return 1;
366
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
367
gen_op_iwmmxt_set_mup();
368
gen_op_iwmmxt_set_cup();
369
break;
370
- case 0x304: case 0x704: case 0xb04: case 0xf04:    /* WROR */
371
+ case 0x304: case 0x704: case 0xb04: case 0xf04: /* WROR */
372
case 0x314: case 0x714: case 0xb14: case 0xf14:
373
if (((insn >> 22) & 3) == 0)
374
return 1;
375
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
376
gen_op_iwmmxt_set_mup();
377
gen_op_iwmmxt_set_cup();
378
break;
379
- case 0x116: case 0x316: case 0x516: case 0x716:    /* WMIN */
380
+ case 0x116: case 0x316: case 0x516: case 0x716: /* WMIN */
381
case 0x916: case 0xb16: case 0xd16: case 0xf16:
382
wrd = (insn >> 12) & 0xf;
383
rd0 = (insn >> 16) & 0xf;
384
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
385
gen_op_iwmmxt_movq_wRn_M0(wrd);
386
gen_op_iwmmxt_set_mup();
387
break;
388
- case 0x016: case 0x216: case 0x416: case 0x616:    /* WMAX */
389
+ case 0x016: case 0x216: case 0x416: case 0x616: /* WMAX */
390
case 0x816: case 0xa16: case 0xc16: case 0xe16:
391
wrd = (insn >> 12) & 0xf;
392
rd0 = (insn >> 16) & 0xf;
393
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
394
gen_op_iwmmxt_movq_wRn_M0(wrd);
395
gen_op_iwmmxt_set_mup();
396
break;
397
- case 0x002: case 0x102: case 0x202: case 0x302:    /* WALIGNI */
398
+ case 0x002: case 0x102: case 0x202: case 0x302: /* WALIGNI */
399
case 0x402: case 0x502: case 0x602: case 0x702:
400
wrd = (insn >> 12) & 0xf;
401
rd0 = (insn >> 16) & 0xf;
402
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
403
gen_op_iwmmxt_movq_wRn_M0(wrd);
404
gen_op_iwmmxt_set_mup();
405
break;
406
- case 0x01a: case 0x11a: case 0x21a: case 0x31a:    /* WSUB */
407
+ case 0x01a: case 0x11a: case 0x21a: case 0x31a: /* WSUB */
408
case 0x41a: case 0x51a: case 0x61a: case 0x71a:
409
case 0x81a: case 0x91a: case 0xa1a: case 0xb1a:
410
case 0xc1a: case 0xd1a: case 0xe1a: case 0xf1a:
411
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
412
gen_op_iwmmxt_set_mup();
413
gen_op_iwmmxt_set_cup();
414
break;
415
- case 0x01e: case 0x11e: case 0x21e: case 0x31e:    /* WSHUFH */
416
+ case 0x01e: case 0x11e: case 0x21e: case 0x31e: /* WSHUFH */
417
case 0x41e: case 0x51e: case 0x61e: case 0x71e:
418
case 0x81e: case 0x91e: case 0xa1e: case 0xb1e:
419
case 0xc1e: case 0xd1e: case 0xe1e: case 0xf1e:
420
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
421
gen_op_iwmmxt_set_mup();
422
gen_op_iwmmxt_set_cup();
423
break;
424
- case 0x018: case 0x118: case 0x218: case 0x318:    /* WADD */
425
+ case 0x018: case 0x118: case 0x218: case 0x318: /* WADD */
426
case 0x418: case 0x518: case 0x618: case 0x718:
427
case 0x818: case 0x918: case 0xa18: case 0xb18:
428
case 0xc18: case 0xd18: case 0xe18: case 0xf18:
429
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
430
gen_op_iwmmxt_set_mup();
431
gen_op_iwmmxt_set_cup();
432
break;
433
- case 0x008: case 0x108: case 0x208: case 0x308:    /* WPACK */
434
+ case 0x008: case 0x108: case 0x208: case 0x308: /* WPACK */
435
case 0x408: case 0x508: case 0x608: case 0x708:
436
case 0x808: case 0x908: case 0xa08: case 0xb08:
437
case 0xc08: case 0xd08: case 0xe08: case 0xf08:
438
@@ -XXX,XX +XXX,XX @@ static int disas_iwmmxt_insn(DisasContext *s, uint32_t insn)
439
tmp = load_reg(s, rd0);
440
tmp2 = load_reg(s, rd1);
441
switch ((insn >> 16) & 0xf) {
442
- case 0x0:                    /* TMIA */
443
+ case 0x0: /* TMIA */
444
gen_helper_iwmmxt_muladdsl(cpu_M0, cpu_M0, tmp, tmp2);
445
break;
58
break;
446
- case 0x8:                    /* TMIAPH */
59
+ case DISAS_UPDATE_NOCHAIN:
447
+ case 0x8: /* TMIAPH */
60
+ gen_set_pc_im(dc, dc->base.pc_next);
448
gen_helper_iwmmxt_muladdsw(cpu_M0, cpu_M0, tmp, tmp2);
61
+ /* fall through */
62
case DISAS_JUMP:
63
gen_goto_ptr();
449
break;
64
break;
450
- case 0xc: case 0xd: case 0xe: case 0xf:        /* TMIAxy */
451
+ case 0xc: case 0xd: case 0xe: case 0xf: /* TMIAxy */
452
if (insn & (1 << 16))
453
tcg_gen_shri_i32(tmp, tmp, 16);
454
if (insn & (1 << 17))
455
@@ -XXX,XX +XXX,XX @@ static int disas_dsp_insn(DisasContext *s, uint32_t insn)
456
tmp = load_reg(s, rd0);
457
tmp2 = load_reg(s, rd1);
458
switch ((insn >> 16) & 0xf) {
459
- case 0x0:                    /* MIA */
460
+ case 0x0: /* MIA */
461
gen_helper_iwmmxt_muladdsl(cpu_M0, cpu_M0, tmp, tmp2);
462
break;
463
- case 0x8:                    /* MIAPH */
464
+ case 0x8: /* MIAPH */
465
gen_helper_iwmmxt_muladdsw(cpu_M0, cpu_M0, tmp, tmp2);
466
break;
467
- case 0xc:                    /* MIABB */
468
- case 0xd:                    /* MIABT */
469
- case 0xe:                    /* MIATB */
470
- case 0xf:                    /* MIATT */
471
+ case 0xc: /* MIABB */
472
+ case 0xd: /* MIABT */
473
+ case 0xe: /* MIATB */
474
+ case 0xf: /* MIATT */
475
if (insn & (1 << 16))
476
tcg_gen_shri_i32(tmp, tmp, 16);
477
if (insn & (1 << 17))
478
@@ -XXX,XX +XXX,XX @@ static int disas_dsp_insn(DisasContext *s, uint32_t insn)
479
if (acc != 0)
480
return 1;
481
482
- if (insn & ARM_CP_RW_BIT) {            /* MRA */
483
+ if (insn & ARM_CP_RW_BIT) { /* MRA */
484
iwmmxt_load_reg(cpu_V0, acc);
485
tcg_gen_extrl_i64_i32(cpu_R[rdlo], cpu_V0);
486
tcg_gen_shri_i64(cpu_V0, cpu_V0, 32);
487
tcg_gen_extrl_i64_i32(cpu_R[rdhi], cpu_V0);
488
tcg_gen_andi_i32(cpu_R[rdhi], cpu_R[rdhi], (1 << (40 - 32)) - 1);
489
- } else {                    /* MAR */
490
+ } else { /* MAR */
491
tcg_gen_concat_i32_i64(cpu_V0, cpu_R[rdlo], cpu_R[rdhi]);
492
iwmmxt_store_reg(cpu_V0, acc);
493
}
494
--
65
--
495
2.18.0
66
2.20.1
496
67
497
68
diff view generated by jsdifflib
New patch
1
1
From: Richard Henderson <richard.henderson@linaro.org>
2
3
This is TFSRE0_EL1, TFSR_EL1, TFSR_EL2, TFSR_EL3,
4
RGSR_EL1, GCR_EL1, GMID_EL1, and PSTATE.TCO.
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-8-richard.henderson@linaro.org
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
11
target/arm/cpu.h | 4 ++
12
target/arm/internals.h | 9 ++++
13
target/arm/helper.c | 94 ++++++++++++++++++++++++++++++++++++++
14
target/arm/translate-a64.c | 21 +++++++++
15
4 files changed, 128 insertions(+)
16
17
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
18
index XXXXXXX..XXXXXXX 100644
19
--- a/target/arm/cpu.h
20
+++ b/target/arm/cpu.h
21
@@ -XXX,XX +XXX,XX @@ typedef struct CPUARMState {
22
uint64_t pmccfiltr_el0; /* Performance Monitor Filter Register */
23
uint64_t vpidr_el2; /* Virtualization Processor ID Register */
24
uint64_t vmpidr_el2; /* Virtualization Multiprocessor ID Register */
25
+ uint64_t tfsr_el[4]; /* tfsre0_el1 is index 0. */
26
+ uint64_t gcr_el1;
27
+ uint64_t rgsr_el1;
28
} cp15;
29
30
struct {
31
@@ -XXX,XX +XXX,XX @@ void pmu_init(ARMCPU *cpu);
32
#define PSTATE_SS (1U << 21)
33
#define PSTATE_PAN (1U << 22)
34
#define PSTATE_UAO (1U << 23)
35
+#define PSTATE_TCO (1U << 25)
36
#define PSTATE_V (1U << 28)
37
#define PSTATE_C (1U << 29)
38
#define PSTATE_Z (1U << 30)
39
diff --git a/target/arm/internals.h b/target/arm/internals.h
40
index XXXXXXX..XXXXXXX 100644
41
--- a/target/arm/internals.h
42
+++ b/target/arm/internals.h
43
@@ -XXX,XX +XXX,XX @@ static inline uint32_t aarch64_pstate_valid_mask(const ARMISARegisters *id)
44
if (isar_feature_aa64_uao(id)) {
45
valid |= PSTATE_UAO;
46
}
47
+ if (isar_feature_aa64_mte(id)) {
48
+ valid |= PSTATE_TCO;
49
+ }
50
51
return valid;
52
}
53
@@ -XXX,XX +XXX,XX @@ void arm_log_exception(int idx);
54
55
#endif /* !CONFIG_USER_ONLY */
56
57
+/*
58
+ * The log2 of the words in the tag block, for GMID_EL1.BS.
59
+ * The is the maximum, 256 bytes, which manipulates 64-bits of tags.
60
+ */
61
+#define GMID_EL1_BS 6
62
+
63
#endif
64
diff --git a/target/arm/helper.c b/target/arm/helper.c
65
index XXXXXXX..XXXXXXX 100644
66
--- a/target/arm/helper.c
67
+++ b/target/arm/helper.c
68
@@ -XXX,XX +XXX,XX @@ static void define_arm_vh_e2h_redirects_aliases(ARMCPU *cpu)
69
{ K(3, 0, 1, 2, 0), K(3, 4, 1, 2, 0), K(3, 5, 1, 2, 0),
70
"ZCR_EL1", "ZCR_EL2", "ZCR_EL12", isar_feature_aa64_sve },
71
72
+ { K(3, 0, 5, 6, 0), K(3, 4, 5, 6, 0), K(3, 5, 5, 6, 0),
73
+ "TFSR_EL1", "TFSR_EL2", "TFSR_EL12", isar_feature_aa64_mte },
74
+
75
/* TODO: ARMv8.2-SPE -- PMSCR_EL2 */
76
/* TODO: ARMv8.4-Trace -- TRFCR_EL2 */
77
};
78
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo dcpodp_reg[] = {
79
};
80
#endif /*CONFIG_USER_ONLY*/
81
82
+static CPAccessResult access_aa64_tid5(CPUARMState *env, const ARMCPRegInfo *ri,
83
+ bool isread)
84
+{
85
+ if ((arm_current_el(env) < 2) && (arm_hcr_el2_eff(env) & HCR_TID5)) {
86
+ return CP_ACCESS_TRAP_EL2;
87
+ }
88
+
89
+ return CP_ACCESS_OK;
90
+}
91
+
92
+static CPAccessResult access_mte(CPUARMState *env, const ARMCPRegInfo *ri,
93
+ bool isread)
94
+{
95
+ int el = arm_current_el(env);
96
+
97
+ if (el < 2 &&
98
+ arm_feature(env, ARM_FEATURE_EL2) &&
99
+ !(arm_hcr_el2_eff(env) & HCR_ATA)) {
100
+ return CP_ACCESS_TRAP_EL2;
101
+ }
102
+ if (el < 3 &&
103
+ arm_feature(env, ARM_FEATURE_EL3) &&
104
+ !(env->cp15.scr_el3 & SCR_ATA)) {
105
+ return CP_ACCESS_TRAP_EL3;
106
+ }
107
+ return CP_ACCESS_OK;
108
+}
109
+
110
+static uint64_t tco_read(CPUARMState *env, const ARMCPRegInfo *ri)
111
+{
112
+ return env->pstate & PSTATE_TCO;
113
+}
114
+
115
+static void tco_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t val)
116
+{
117
+ env->pstate = (env->pstate & ~PSTATE_TCO) | (val & PSTATE_TCO);
118
+}
119
+
120
+static const ARMCPRegInfo mte_reginfo[] = {
121
+ { .name = "TFSRE0_EL1", .state = ARM_CP_STATE_AA64,
122
+ .opc0 = 3, .opc1 = 0, .crn = 5, .crm = 6, .opc2 = 1,
123
+ .access = PL1_RW, .accessfn = access_mte,
124
+ .fieldoffset = offsetof(CPUARMState, cp15.tfsr_el[0]) },
125
+ { .name = "TFSR_EL1", .state = ARM_CP_STATE_AA64,
126
+ .opc0 = 3, .opc1 = 0, .crn = 5, .crm = 6, .opc2 = 0,
127
+ .access = PL1_RW, .accessfn = access_mte,
128
+ .fieldoffset = offsetof(CPUARMState, cp15.tfsr_el[1]) },
129
+ { .name = "TFSR_EL2", .state = ARM_CP_STATE_AA64,
130
+ .opc0 = 3, .opc1 = 4, .crn = 5, .crm = 6, .opc2 = 0,
131
+ .access = PL2_RW, .accessfn = access_mte,
132
+ .fieldoffset = offsetof(CPUARMState, cp15.tfsr_el[2]) },
133
+ { .name = "TFSR_EL3", .state = ARM_CP_STATE_AA64,
134
+ .opc0 = 3, .opc1 = 6, .crn = 5, .crm = 6, .opc2 = 0,
135
+ .access = PL3_RW,
136
+ .fieldoffset = offsetof(CPUARMState, cp15.tfsr_el[3]) },
137
+ { .name = "RGSR_EL1", .state = ARM_CP_STATE_AA64,
138
+ .opc0 = 3, .opc1 = 0, .crn = 1, .crm = 0, .opc2 = 5,
139
+ .access = PL1_RW, .accessfn = access_mte,
140
+ .fieldoffset = offsetof(CPUARMState, cp15.rgsr_el1) },
141
+ { .name = "GCR_EL1", .state = ARM_CP_STATE_AA64,
142
+ .opc0 = 3, .opc1 = 0, .crn = 1, .crm = 0, .opc2 = 6,
143
+ .access = PL1_RW, .accessfn = access_mte,
144
+ .fieldoffset = offsetof(CPUARMState, cp15.gcr_el1) },
145
+ { .name = "GMID_EL1", .state = ARM_CP_STATE_AA64,
146
+ .opc0 = 3, .opc1 = 1, .crn = 0, .crm = 0, .opc2 = 4,
147
+ .access = PL1_R, .accessfn = access_aa64_tid5,
148
+ .type = ARM_CP_CONST, .resetvalue = GMID_EL1_BS },
149
+ { .name = "TCO", .state = ARM_CP_STATE_AA64,
150
+ .opc0 = 3, .opc1 = 3, .crn = 4, .crm = 2, .opc2 = 7,
151
+ .type = ARM_CP_NO_RAW,
152
+ .access = PL0_RW, .readfn = tco_read, .writefn = tco_write },
153
+ REGINFO_SENTINEL
154
+};
155
+
156
+static const ARMCPRegInfo mte_tco_ro_reginfo[] = {
157
+ { .name = "TCO", .state = ARM_CP_STATE_AA64,
158
+ .opc0 = 3, .opc1 = 3, .crn = 4, .crm = 2, .opc2 = 7,
159
+ .type = ARM_CP_CONST, .access = PL0_RW, },
160
+ REGINFO_SENTINEL
161
+};
162
#endif
163
164
static CPAccessResult access_predinv(CPUARMState *env, const ARMCPRegInfo *ri,
165
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
166
}
167
}
168
#endif /*CONFIG_USER_ONLY*/
169
+
170
+ /*
171
+ * If full MTE is enabled, add all of the system registers.
172
+ * If only "instructions available at EL0" are enabled,
173
+ * then define only a RAZ/WI version of PSTATE.TCO.
174
+ */
175
+ if (cpu_isar_feature(aa64_mte, cpu)) {
176
+ define_arm_cp_regs(cpu, mte_reginfo);
177
+ } else if (cpu_isar_feature(aa64_mte_insn_reg, cpu)) {
178
+ define_arm_cp_regs(cpu, mte_tco_ro_reginfo);
179
+ }
180
#endif
181
182
if (cpu_isar_feature(any_predinv, cpu)) {
183
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
184
index XXXXXXX..XXXXXXX 100644
185
--- a/target/arm/translate-a64.c
186
+++ b/target/arm/translate-a64.c
187
@@ -XXX,XX +XXX,XX @@ static void handle_msr_i(DisasContext *s, uint32_t insn,
188
s->base.is_jmp = DISAS_UPDATE_EXIT;
189
break;
190
191
+ case 0x1c: /* TCO */
192
+ if (dc_isar_feature(aa64_mte, s)) {
193
+ /* Full MTE is enabled -- set the TCO bit as directed. */
194
+ if (crm & 1) {
195
+ set_pstate_bits(PSTATE_TCO);
196
+ } else {
197
+ clear_pstate_bits(PSTATE_TCO);
198
+ }
199
+ t1 = tcg_const_i32(s->current_el);
200
+ gen_helper_rebuild_hflags_a64(cpu_env, t1);
201
+ tcg_temp_free_i32(t1);
202
+ /* Many factors, including TCO, go into MTE_ACTIVE. */
203
+ s->base.is_jmp = DISAS_UPDATE_NOCHAIN;
204
+ } else if (dc_isar_feature(aa64_mte_insn_reg, s)) {
205
+ /* Only "instructions accessible at EL0" -- PSTATE.TCO is WI. */
206
+ s->base.is_jmp = DISAS_NEXT;
207
+ } else {
208
+ goto do_unallocated;
209
+ }
210
+ break;
211
+
212
default:
213
do_unallocated:
214
unallocated_encoding(s);
215
--
216
2.20.1
217
218
diff view generated by jsdifflib
1
Factor out the code which changes the CPU state so as to
1
From: Richard Henderson <richard.henderson@linaro.org>
2
actually take an exception to AArch32. We're going to want
3
to use this for handling exception entry to Hyp mode.
4
2
3
Cache the composite ATA setting.
4
5
Cache when MTE is fully enabled, i.e. access to tags are enabled
6
and tag checks affect the PE. Do this for both the normal context
7
and the UNPRIV context.
8
9
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
11
Message-id: 20200626033144.790098-9-richard.henderson@linaro.org
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20180820153020.21478-4-peter.maydell@linaro.org
10
---
13
---
11
target/arm/helper.c | 64 +++++++++++++++++++++++++++++----------------
14
target/arm/cpu.h | 12 ++++++++----
12
1 file changed, 41 insertions(+), 23 deletions(-)
15
target/arm/internals.h | 18 +++++++++++++++++
16
target/arm/translate.h | 5 +++++
17
target/arm/helper.c | 40 ++++++++++++++++++++++++++++++++++++++
18
target/arm/translate-a64.c | 4 ++++
19
5 files changed, 75 insertions(+), 4 deletions(-)
13
20
21
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
22
index XXXXXXX..XXXXXXX 100644
23
--- a/target/arm/cpu.h
24
+++ b/target/arm/cpu.h
25
@@ -XXX,XX +XXX,XX @@ typedef ARMCPU ArchCPU;
26
* | | | TBFLAG_A32 | |
27
* | | +-----+----------+ TBFLAG_AM32 |
28
* | TBFLAG_ANY | |TBFLAG_M32| |
29
- * | | +-+----------+--------------|
30
- * | | | TBFLAG_A64 |
31
- * +--------------+---------+---------------------------+
32
- * 31 20 15 0
33
+ * | +-----------+----------+--------------|
34
+ * | | TBFLAG_A64 |
35
+ * +--------------+-------------------------------------+
36
+ * 31 20 0
37
*
38
* Unless otherwise noted, these bits are cached in env->hflags.
39
*/
40
@@ -XXX,XX +XXX,XX @@ FIELD(TBFLAG_A64, BT, 9, 1)
41
FIELD(TBFLAG_A64, BTYPE, 10, 2) /* Not cached. */
42
FIELD(TBFLAG_A64, TBID, 12, 2)
43
FIELD(TBFLAG_A64, UNPRIV, 14, 1)
44
+FIELD(TBFLAG_A64, ATA, 15, 1)
45
+FIELD(TBFLAG_A64, TCMA, 16, 2)
46
+FIELD(TBFLAG_A64, MTE_ACTIVE, 18, 1)
47
+FIELD(TBFLAG_A64, MTE0_ACTIVE, 19, 1)
48
49
/**
50
* cpu_mmu_index:
51
diff --git a/target/arm/internals.h b/target/arm/internals.h
52
index XXXXXXX..XXXXXXX 100644
53
--- a/target/arm/internals.h
54
+++ b/target/arm/internals.h
55
@@ -XXX,XX +XXX,XX @@ static inline int exception_target_el(CPUARMState *env)
56
return target_el;
57
}
58
59
+/* Determine if allocation tags are available. */
60
+static inline bool allocation_tag_access_enabled(CPUARMState *env, int el,
61
+ uint64_t sctlr)
62
+{
63
+ if (el < 3
64
+ && arm_feature(env, ARM_FEATURE_EL3)
65
+ && !(env->cp15.scr_el3 & SCR_ATA)) {
66
+ return false;
67
+ }
68
+ if (el < 2
69
+ && arm_feature(env, ARM_FEATURE_EL2)
70
+ && !(arm_hcr_el2_eff(env) & HCR_ATA)) {
71
+ return false;
72
+ }
73
+ sctlr &= (el == 0 ? SCTLR_ATA0 : SCTLR_ATA);
74
+ return sctlr != 0;
75
+}
76
+
77
#ifndef CONFIG_USER_ONLY
78
79
/* Security attributes for an address, as returned by v8m_security_lookup. */
80
diff --git a/target/arm/translate.h b/target/arm/translate.h
81
index XXXXXXX..XXXXXXX 100644
82
--- a/target/arm/translate.h
83
+++ b/target/arm/translate.h
84
@@ -XXX,XX +XXX,XX @@ typedef struct DisasContext {
85
ARMMMUIdx mmu_idx; /* MMU index to use for normal loads/stores */
86
uint8_t tbii; /* TBI1|TBI0 for insns */
87
uint8_t tbid; /* TBI1|TBI0 for data */
88
+ uint8_t tcma; /* TCMA1|TCMA0 for MTE */
89
bool ns; /* Use non-secure CPREG bank on access */
90
int fp_excp_el; /* FP exception EL or 0 if enabled */
91
int sve_excp_el; /* SVE exception EL or 0 if enabled */
92
@@ -XXX,XX +XXX,XX @@ typedef struct DisasContext {
93
bool unpriv;
94
/* True if v8.3-PAuth is active. */
95
bool pauth_active;
96
+ /* True if v8.5-MTE access to tags is enabled. */
97
+ bool ata;
98
+ /* True if v8.5-MTE tag checks affect the PE; index with is_unpriv. */
99
+ bool mte_active[2];
100
/* True with v8.5-BTI and SCTLR_ELx.BT* set. */
101
bool bt;
102
/* True if any CP15 access is trapped by HSTR_EL2 */
14
diff --git a/target/arm/helper.c b/target/arm/helper.c
103
diff --git a/target/arm/helper.c b/target/arm/helper.c
15
index XXXXXXX..XXXXXXX 100644
104
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/helper.c
105
--- a/target/arm/helper.c
17
+++ b/target/arm/helper.c
106
+++ b/target/arm/helper.c
18
@@ -XXX,XX +XXX,XX @@ void aarch64_sync_64_to_32(CPUARMState *env)
107
@@ -XXX,XX +XXX,XX @@ static int aa64_va_parameter_tbid(uint64_t tcr, ARMMMUIdx mmu_idx)
19
env->regs[15] = env->pc;
108
}
20
}
109
}
21
110
22
+static void take_aarch32_exception(CPUARMState *env, int new_mode,
111
+static int aa64_va_parameter_tcma(uint64_t tcr, ARMMMUIdx mmu_idx)
23
+ uint32_t mask, uint32_t offset,
24
+ uint32_t newpc)
25
+{
112
+{
26
+ /* Change the CPU state so as to actually take the exception. */
113
+ if (regime_has_2_ranges(mmu_idx)) {
27
+ switch_mode(env, new_mode);
114
+ return extract64(tcr, 57, 2);
28
+ /*
115
+ } else {
29
+ * For exceptions taken to AArch32 we must clear the SS bit in both
116
+ /* Replicate the single TCMA bit so we always have 2 bits. */
30
+ * PSTATE and in the old-state value we save to SPSR_<mode>, so zero it now.
117
+ return extract32(tcr, 30, 1) * 3;
31
+ */
32
+ env->uncached_cpsr &= ~PSTATE_SS;
33
+ env->spsr = cpsr_read(env);
34
+ /* Clear IT bits. */
35
+ env->condexec_bits = 0;
36
+ /* Switch to the new mode, and to the correct instruction set. */
37
+ env->uncached_cpsr = (env->uncached_cpsr & ~CPSR_M) | new_mode;
38
+ /* Set new mode endianness */
39
+ env->uncached_cpsr &= ~CPSR_E;
40
+ if (env->cp15.sctlr_el[arm_current_el(env)] & SCTLR_EE) {
41
+ env->uncached_cpsr |= CPSR_E;
42
+ }
118
+ }
43
+ env->daif |= mask;
44
+
45
+ if (new_mode == ARM_CPU_MODE_HYP) {
46
+ env->thumb = (env->cp15.sctlr_el[2] & SCTLR_TE) != 0;
47
+ env->elr_el[2] = env->regs[15];
48
+ } else {
49
+ /*
50
+ * this is a lie, as there was no c1_sys on V4T/V5, but who cares
51
+ * and we should just guard the thumb mode on V4
52
+ */
53
+ if (arm_feature(env, ARM_FEATURE_V4T)) {
54
+ env->thumb =
55
+ (A32_BANKED_CURRENT_REG_GET(env, sctlr) & SCTLR_TE) != 0;
56
+ }
57
+ env->regs[14] = env->regs[15] + offset;
58
+ }
59
+ env->regs[15] = newpc;
60
+}
119
+}
61
+
120
+
62
static void arm_cpu_do_interrupt_aarch32(CPUState *cs)
121
ARMVAParameters aa64_va_parameters(CPUARMState *env, uint64_t va,
122
ARMMMUIdx mmu_idx, bool data)
63
{
123
{
64
ARMCPU *cpu = ARM_CPU(cs);
124
@@ -XXX,XX +XXX,XX @@ static uint32_t rebuild_hflags_a64(CPUARMState *env, int el, int fp_el,
65
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_do_interrupt_aarch32(CPUState *cs)
125
}
66
env->cp15.scr_el3 &= ~SCR_NS;
67
}
126
}
68
127
69
- switch_mode (env, new_mode);
128
+ if (cpu_isar_feature(aa64_mte, env_archcpu(env))) {
70
- /* For exceptions taken to AArch32 we must clear the SS bit in both
129
+ /*
71
- * PSTATE and in the old-state value we save to SPSR_<mode>, so zero it now.
130
+ * Set MTE_ACTIVE if any access may be Checked, and leave clear
72
- */
131
+ * if all accesses must be Unchecked:
73
- env->uncached_cpsr &= ~PSTATE_SS;
132
+ * 1) If no TBI, then there are no tags in the address to check,
74
- env->spsr = cpsr_read(env);
133
+ * 2) If Tag Check Override, then all accesses are Unchecked,
75
- /* Clear IT bits. */
134
+ * 3) If Tag Check Fail == 0, then Checked access have no effect,
76
- env->condexec_bits = 0;
135
+ * 4) If no Allocation Tag Access, then all accesses are Unchecked.
77
- /* Switch to the new mode, and to the correct instruction set. */
136
+ */
78
- env->uncached_cpsr = (env->uncached_cpsr & ~CPSR_M) | new_mode;
137
+ if (allocation_tag_access_enabled(env, el, sctlr)) {
79
- /* Set new mode endianness */
138
+ flags = FIELD_DP32(flags, TBFLAG_A64, ATA, 1);
80
- env->uncached_cpsr &= ~CPSR_E;
139
+ if (tbid
81
- if (env->cp15.sctlr_el[arm_current_el(env)] & SCTLR_EE) {
140
+ && !(env->pstate & PSTATE_TCO)
82
- env->uncached_cpsr |= CPSR_E;
141
+ && (sctlr & (el == 0 ? SCTLR_TCF0 : SCTLR_TCF))) {
83
- }
142
+ flags = FIELD_DP32(flags, TBFLAG_A64, MTE_ACTIVE, 1);
84
- env->daif |= mask;
143
+ }
85
- /* this is a lie, as the was no c1_sys on V4T/V5, but who cares
144
+ }
86
- * and we should just guard the thumb mode on V4 */
145
+ /* And again for unprivileged accesses, if required. */
87
- if (arm_feature(env, ARM_FEATURE_V4T)) {
146
+ if (FIELD_EX32(flags, TBFLAG_A64, UNPRIV)
88
- env->thumb = (A32_BANKED_CURRENT_REG_GET(env, sctlr) & SCTLR_TE) != 0;
147
+ && tbid
89
- }
148
+ && !(env->pstate & PSTATE_TCO)
90
- env->regs[14] = env->regs[15] + offset;
149
+ && (sctlr & SCTLR_TCF0)
91
- env->regs[15] = addr;
150
+ && allocation_tag_access_enabled(env, 0, sctlr)) {
92
+ take_aarch32_exception(env, new_mode, mask, offset, addr);
151
+ flags = FIELD_DP32(flags, TBFLAG_A64, MTE0_ACTIVE, 1);
152
+ }
153
+ /* Cache TCMA as well as TBI. */
154
+ flags = FIELD_DP32(flags, TBFLAG_A64, TCMA,
155
+ aa64_va_parameter_tcma(tcr, mmu_idx));
156
+ }
157
+
158
return rebuild_hflags_common(env, fp_el, mmu_idx, flags);
93
}
159
}
94
160
95
/* Handle exception entry to a target EL which is using AArch64 */
161
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
162
index XXXXXXX..XXXXXXX 100644
163
--- a/target/arm/translate-a64.c
164
+++ b/target/arm/translate-a64.c
165
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_init_disas_context(DisasContextBase *dcbase,
166
dc->mmu_idx = core_to_aa64_mmu_idx(core_mmu_idx);
167
dc->tbii = FIELD_EX32(tb_flags, TBFLAG_A64, TBII);
168
dc->tbid = FIELD_EX32(tb_flags, TBFLAG_A64, TBID);
169
+ dc->tcma = FIELD_EX32(tb_flags, TBFLAG_A64, TCMA);
170
dc->current_el = arm_mmu_idx_to_el(dc->mmu_idx);
171
#if !defined(CONFIG_USER_ONLY)
172
dc->user = (dc->current_el == 0);
173
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_init_disas_context(DisasContextBase *dcbase,
174
dc->bt = FIELD_EX32(tb_flags, TBFLAG_A64, BT);
175
dc->btype = FIELD_EX32(tb_flags, TBFLAG_A64, BTYPE);
176
dc->unpriv = FIELD_EX32(tb_flags, TBFLAG_A64, UNPRIV);
177
+ dc->ata = FIELD_EX32(tb_flags, TBFLAG_A64, ATA);
178
+ dc->mte_active[0] = FIELD_EX32(tb_flags, TBFLAG_A64, MTE_ACTIVE);
179
+ dc->mte_active[1] = FIELD_EX32(tb_flags, TBFLAG_A64, MTE0_ACTIVE);
180
dc->vec_len = 0;
181
dc->vec_stride = 0;
182
dc->cp_regs = arm_cpu->cp_regs;
96
--
183
--
97
2.18.0
184
2.20.1
98
185
99
186
diff view generated by jsdifflib
1
Implement the IoTKit system control element's system information
1
From: Richard Henderson <richard.henderson@linaro.org>
2
block; this is just a pair of read-only version/config registers,
3
plus the usual PID/CID ID registers.
4
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-10-richard.henderson@linaro.org
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20180820141116.9118-10-peter.maydell@linaro.org
9
---
7
---
10
hw/misc/Makefile.objs | 1 +
8
target/arm/helper-a64.h | 2 ++
11
include/hw/misc/iotkit-sysinfo.h | 37 +++++++++
9
target/arm/internals.h | 5 +++
12
hw/misc/iotkit-sysinfo.c | 128 +++++++++++++++++++++++++++++++
10
target/arm/mte_helper.c | 72 ++++++++++++++++++++++++++++++++++++++
13
MAINTAINERS | 2 +
11
target/arm/translate-a64.c | 18 ++++++++++
14
default-configs/arm-softmmu.mak | 1 +
12
target/arm/Makefile.objs | 1 +
15
5 files changed, 169 insertions(+)
13
5 files changed, 98 insertions(+)
16
create mode 100644 include/hw/misc/iotkit-sysinfo.h
14
create mode 100644 target/arm/mte_helper.c
17
create mode 100644 hw/misc/iotkit-sysinfo.c
18
15
19
diff --git a/hw/misc/Makefile.objs b/hw/misc/Makefile.objs
16
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
20
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
21
--- a/hw/misc/Makefile.objs
18
--- a/target/arm/helper-a64.h
22
+++ b/hw/misc/Makefile.objs
19
+++ b/target/arm/helper-a64.h
23
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_TZ_MPC) += tz-mpc.o
20
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_3(autda, TCG_CALL_NO_WG, i64, env, i64, i64)
24
obj-$(CONFIG_TZ_PPC) += tz-ppc.o
21
DEF_HELPER_FLAGS_3(autdb, TCG_CALL_NO_WG, i64, env, i64, i64)
25
obj-$(CONFIG_IOTKIT_SECCTL) += iotkit-secctl.o
22
DEF_HELPER_FLAGS_2(xpaci, TCG_CALL_NO_RWG_SE, i64, env, i64)
26
obj-$(CONFIG_IOTKIT_SYSCTL) += iotkit-sysctl.o
23
DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
27
+obj-$(CONFIG_IOTKIT_SYSINFO) += iotkit-sysinfo.o
24
+
28
25
+DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
29
obj-$(CONFIG_PVPANIC) += pvpanic.o
26
diff --git a/target/arm/internals.h b/target/arm/internals.h
30
obj-$(CONFIG_HYPERV_TESTDEV) += hyperv_testdev.o
27
index XXXXXXX..XXXXXXX 100644
31
diff --git a/include/hw/misc/iotkit-sysinfo.h b/include/hw/misc/iotkit-sysinfo.h
28
--- a/target/arm/internals.h
29
+++ b/target/arm/internals.h
30
@@ -XXX,XX +XXX,XX @@ void arm_log_exception(int idx);
31
*/
32
#define GMID_EL1_BS 6
33
34
+static inline uint64_t address_with_allocation_tag(uint64_t ptr, int rtag)
35
+{
36
+ return deposit64(ptr, 56, 4, rtag);
37
+}
38
+
39
#endif
40
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
32
new file mode 100644
41
new file mode 100644
33
index XXXXXXX..XXXXXXX
42
index XXXXXXX..XXXXXXX
34
--- /dev/null
43
--- /dev/null
35
+++ b/include/hw/misc/iotkit-sysinfo.h
44
+++ b/target/arm/mte_helper.c
36
@@ -XXX,XX +XXX,XX @@
45
@@ -XXX,XX +XXX,XX @@
37
+/*
46
+/*
38
+ * ARM IoTKit system information block
47
+ * ARM v8.5-MemTag Operations
39
+ *
48
+ *
40
+ * Copyright (c) 2018 Linaro Limited
49
+ * Copyright (c) 2020 Linaro, Ltd.
41
+ * Written by Peter Maydell
42
+ *
50
+ *
43
+ * This program is free software; you can redistribute it and/or modify
51
+ * This library is free software; you can redistribute it and/or
44
+ * it under the terms of the GNU General Public License version 2 or
52
+ * modify it under the terms of the GNU Lesser General Public
45
+ * (at your option) any later version.
53
+ * License as published by the Free Software Foundation; either
46
+ */
54
+ * version 2.1 of the License, or (at your option) any later version.
47
+
48
+/*
49
+ * This is a model of the "system information block" which is part of the
50
+ * Arm IoTKit and documented in
51
+ * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
52
+ * QEMU interface:
53
+ * + sysbus MMIO region 0: the system information register bank
54
+ */
55
+
56
+#ifndef HW_MISC_IOTKIT_SYSINFO_H
57
+#define HW_MISC_IOTKIT_SYSINFO_H
58
+
59
+#include "hw/sysbus.h"
60
+
61
+#define TYPE_IOTKIT_SYSINFO "iotkit-sysinfo"
62
+#define IOTKIT_SYSINFO(obj) OBJECT_CHECK(IoTKitSysInfo, (obj), \
63
+ TYPE_IOTKIT_SYSINFO)
64
+
65
+typedef struct IoTKitSysInfo {
66
+ /*< private >*/
67
+ SysBusDevice parent_obj;
68
+
69
+ /*< public >*/
70
+ MemoryRegion iomem;
71
+} IoTKitSysInfo;
72
+
73
+#endif
74
diff --git a/hw/misc/iotkit-sysinfo.c b/hw/misc/iotkit-sysinfo.c
75
new file mode 100644
76
index XXXXXXX..XXXXXXX
77
--- /dev/null
78
+++ b/hw/misc/iotkit-sysinfo.c
79
@@ -XXX,XX +XXX,XX @@
80
+/*
81
+ * ARM IoTKit system information block
82
+ *
55
+ *
83
+ * Copyright (c) 2018 Linaro Limited
56
+ * This library is distributed in the hope that it will be useful,
84
+ * Written by Peter Maydell
57
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
58
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
59
+ * Lesser General Public License for more details.
85
+ *
60
+ *
86
+ * This program is free software; you can redistribute it and/or modify
61
+ * You should have received a copy of the GNU Lesser General Public
87
+ * it under the terms of the GNU General Public License version 2 or
62
+ * License along with this library; if not, see <http://www.gnu.org/licenses/>.
88
+ * (at your option) any later version.
89
+ */
90
+
91
+/*
92
+ * This is a model of the "system information block" which is part of the
93
+ * Arm IoTKit and documented in
94
+ * http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ecm0601256/index.html
95
+ * It consists of 2 read-only version/config registers, plus the
96
+ * usual ID registers.
97
+ */
63
+ */
98
+
64
+
99
+#include "qemu/osdep.h"
65
+#include "qemu/osdep.h"
100
+#include "qemu/log.h"
66
+#include "cpu.h"
101
+#include "trace.h"
67
+#include "internals.h"
102
+#include "qapi/error.h"
68
+#include "exec/exec-all.h"
103
+#include "sysemu/sysemu.h"
69
+#include "exec/cpu_ldst.h"
104
+#include "hw/sysbus.h"
70
+#include "exec/helper-proto.h"
105
+#include "hw/registerfields.h"
106
+#include "hw/misc/iotkit-sysinfo.h"
107
+
71
+
108
+REG32(SYS_VERSION, 0x0)
109
+REG32(SYS_CONFIG, 0x4)
110
+REG32(PID4, 0xfd0)
111
+REG32(PID5, 0xfd4)
112
+REG32(PID6, 0xfd8)
113
+REG32(PID7, 0xfdc)
114
+REG32(PID0, 0xfe0)
115
+REG32(PID1, 0xfe4)
116
+REG32(PID2, 0xfe8)
117
+REG32(PID3, 0xfec)
118
+REG32(CID0, 0xff0)
119
+REG32(CID1, 0xff4)
120
+REG32(CID2, 0xff8)
121
+REG32(CID3, 0xffc)
122
+
72
+
123
+/* PID/CID values */
73
+static int choose_nonexcluded_tag(int tag, int offset, uint16_t exclude)
124
+static const int sysinfo_id[] = {
125
+ 0x04, 0x00, 0x00, 0x00, /* PID4..PID7 */
126
+ 0x58, 0xb8, 0x0b, 0x00, /* PID0..PID3 */
127
+ 0x0d, 0xf0, 0x05, 0xb1, /* CID0..CID3 */
128
+};
129
+
130
+static uint64_t iotkit_sysinfo_read(void *opaque, hwaddr offset,
131
+ unsigned size)
132
+{
74
+{
133
+ uint64_t r;
75
+ if (exclude == 0xffff) {
134
+
76
+ return 0;
135
+ switch (offset) {
136
+ case A_SYS_VERSION:
137
+ r = 0x41743;
138
+ break;
139
+
140
+ case A_SYS_CONFIG:
141
+ r = 0x31;
142
+ break;
143
+ case A_PID4 ... A_CID3:
144
+ r = sysinfo_id[(offset - A_PID4) / 4];
145
+ break;
146
+ default:
147
+ qemu_log_mask(LOG_GUEST_ERROR,
148
+ "IoTKit SysInfo read: bad offset %x\n", (int)offset);
149
+ r = 0;
150
+ break;
151
+ }
77
+ }
152
+ trace_iotkit_sysinfo_read(offset, r, size);
78
+ if (offset == 0) {
153
+ return r;
79
+ while (exclude & (1 << tag)) {
80
+ tag = (tag + 1) & 15;
81
+ }
82
+ } else {
83
+ do {
84
+ do {
85
+ tag = (tag + 1) & 15;
86
+ } while (exclude & (1 << tag));
87
+ } while (--offset > 0);
88
+ }
89
+ return tag;
154
+}
90
+}
155
+
91
+
156
+static void iotkit_sysinfo_write(void *opaque, hwaddr offset,
92
+uint64_t HELPER(irg)(CPUARMState *env, uint64_t rn, uint64_t rm)
157
+ uint64_t value, unsigned size)
158
+{
93
+{
159
+ trace_iotkit_sysinfo_write(offset, value, size);
94
+ int rtag;
160
+
95
+
161
+ qemu_log_mask(LOG_GUEST_ERROR,
96
+ /*
162
+ "IoTKit SysInfo: write to RO offset 0x%x\n", (int)offset);
97
+ * Our IMPDEF choice for GCR_EL1.RRND==1 is to behave as if
98
+ * GCR_EL1.RRND==0, always producing deterministic results.
99
+ */
100
+ uint16_t exclude = extract32(rm | env->cp15.gcr_el1, 0, 16);
101
+ int start = extract32(env->cp15.rgsr_el1, 0, 4);
102
+ int seed = extract32(env->cp15.rgsr_el1, 8, 16);
103
+ int offset, i;
104
+
105
+ /* RandomTag */
106
+ for (i = offset = 0; i < 4; ++i) {
107
+ /* NextRandomTagBit */
108
+ int top = (extract32(seed, 5, 1) ^ extract32(seed, 3, 1) ^
109
+ extract32(seed, 2, 1) ^ extract32(seed, 0, 1));
110
+ seed = (top << 15) | (seed >> 1);
111
+ offset |= top << i;
112
+ }
113
+ rtag = choose_nonexcluded_tag(start, offset, exclude);
114
+ env->cp15.rgsr_el1 = rtag | (seed << 8);
115
+
116
+ return address_with_allocation_tag(rn, rtag);
117
+}
118
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
119
index XXXXXXX..XXXXXXX 100644
120
--- a/target/arm/translate-a64.c
121
+++ b/target/arm/translate-a64.c
122
@@ -XXX,XX +XXX,XX @@ static TCGv_i64 clean_data_tbi(DisasContext *s, TCGv_i64 addr)
123
return clean;
124
}
125
126
+/* Insert a zero tag into src, with the result at dst. */
127
+static void gen_address_with_allocation_tag0(TCGv_i64 dst, TCGv_i64 src)
128
+{
129
+ tcg_gen_andi_i64(dst, src, ~MAKE_64BIT_MASK(56, 4));
163
+}
130
+}
164
+
131
+
165
+static const MemoryRegionOps iotkit_sysinfo_ops = {
132
typedef struct DisasCompare64 {
166
+ .read = iotkit_sysinfo_read,
133
TCGCond cond;
167
+ .write = iotkit_sysinfo_write,
134
TCGv_i64 value;
168
+ .endianness = DEVICE_LITTLE_ENDIAN,
135
@@ -XXX,XX +XXX,XX @@ static void disas_data_proc_2src(DisasContext *s, uint32_t insn)
169
+ /* byte/halfword accesses are just zero-padded on reads and writes */
136
case 3: /* SDIV */
170
+ .impl.min_access_size = 4,
137
handle_div(s, true, sf, rm, rn, rd);
171
+ .impl.max_access_size = 4,
138
break;
172
+ .valid.min_access_size = 1,
139
+ case 4: /* IRG */
173
+ .valid.max_access_size = 4,
140
+ if (sf == 0 || !dc_isar_feature(aa64_mte_insn_reg, s)) {
174
+};
141
+ goto do_unallocated;
175
+
142
+ }
176
+static void iotkit_sysinfo_init(Object *obj)
143
+ if (s->ata) {
177
+{
144
+ gen_helper_irg(cpu_reg_sp(s, rd), cpu_env,
178
+ SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
145
+ cpu_reg_sp(s, rn), cpu_reg(s, rm));
179
+ IoTKitSysInfo *s = IOTKIT_SYSINFO(obj);
146
+ } else {
180
+
147
+ gen_address_with_allocation_tag0(cpu_reg_sp(s, rd),
181
+ memory_region_init_io(&s->iomem, obj, &iotkit_sysinfo_ops,
148
+ cpu_reg_sp(s, rn));
182
+ s, "iotkit-sysinfo", 0x1000);
149
+ }
183
+ sysbus_init_mmio(sbd, &s->iomem);
150
+ break;
184
+}
151
case 8: /* LSLV */
185
+
152
handle_shift_reg(s, A64_SHIFT_TYPE_LSL, sf, rm, rn, rd);
186
+static void iotkit_sysinfo_class_init(ObjectClass *klass, void *data)
153
break;
187
+{
154
diff --git a/target/arm/Makefile.objs b/target/arm/Makefile.objs
188
+ /*
189
+ * This device has no guest-modifiable state and so it
190
+ * does not need a reset function or VMState.
191
+ */
192
+}
193
+
194
+static const TypeInfo iotkit_sysinfo_info = {
195
+ .name = TYPE_IOTKIT_SYSINFO,
196
+ .parent = TYPE_SYS_BUS_DEVICE,
197
+ .instance_size = sizeof(IoTKitSysInfo),
198
+ .instance_init = iotkit_sysinfo_init,
199
+ .class_init = iotkit_sysinfo_class_init,
200
+};
201
+
202
+static void iotkit_sysinfo_register_types(void)
203
+{
204
+ type_register_static(&iotkit_sysinfo_info);
205
+}
206
+
207
+type_init(iotkit_sysinfo_register_types);
208
diff --git a/MAINTAINERS b/MAINTAINERS
209
index XXXXXXX..XXXXXXX 100644
155
index XXXXXXX..XXXXXXX 100644
210
--- a/MAINTAINERS
156
--- a/target/arm/Makefile.objs
211
+++ b/MAINTAINERS
157
+++ b/target/arm/Makefile.objs
212
@@ -XXX,XX +XXX,XX @@ F: hw/arm/iotkit.c
158
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_SOFTMMU) += psci.o
213
F: include/hw/arm/iotkit.h
159
obj-$(TARGET_AARCH64) += translate-a64.o helper-a64.o
214
F: hw/misc/iotkit-sysctl.c
160
obj-$(TARGET_AARCH64) += translate-sve.o sve_helper.o
215
F: include/hw/misc/iotkit-sysctl.h
161
obj-$(TARGET_AARCH64) += pauth_helper.o
216
+F: hw/misc/iotkit-sysinfo.c
162
+obj-$(TARGET_AARCH64) += mte_helper.o
217
+F: include/hw/misc/iotkit-sysinfo.h
218
219
Musicpal
220
M: Jan Kiszka <jan.kiszka@web.de>
221
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
222
index XXXXXXX..XXXXXXX 100644
223
--- a/default-configs/arm-softmmu.mak
224
+++ b/default-configs/arm-softmmu.mak
225
@@ -XXX,XX +XXX,XX @@ CONFIG_TZ_PPC=y
226
CONFIG_IOTKIT=y
227
CONFIG_IOTKIT_SECCTL=y
228
CONFIG_IOTKIT_SYSCTL=y
229
+CONFIG_IOTKIT_SYSINFO=y
230
231
CONFIG_VERSATILE=y
232
CONFIG_VERSATILE_PCI=y
233
--
163
--
234
2.18.0
164
2.20.1
235
165
236
166
diff view generated by jsdifflib
1
Connect the VIRQ and VFIQ lines from the GIC to the CPU;
1
From: Richard Henderson <richard.henderson@linaro.org>
2
these exist always for both CPU and GIC whether the
3
virtualization extensions are enabled or not, so we
4
can just unconditionally connect them.
5
2
3
The current Arm ARM has adjusted the official decode of
4
"Add/subtract (immediate)" so that the shift field is only bit 22,
5
and bit 23 is part of the op1 field of the parent category
6
"Data processing - immediate".
7
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
10
Message-id: 20200626033144.790098-11-richard.henderson@linaro.org
11
Suggested-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180821132811.17675-7-peter.maydell@linaro.org
9
---
14
---
10
hw/arm/fsl-imx7.c | 4 ++++
15
target/arm/translate-a64.c | 23 ++++++++---------------
11
1 file changed, 4 insertions(+)
16
1 file changed, 8 insertions(+), 15 deletions(-)
12
17
13
diff --git a/hw/arm/fsl-imx7.c b/hw/arm/fsl-imx7.c
18
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
14
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/fsl-imx7.c
20
--- a/target/arm/translate-a64.c
16
+++ b/hw/arm/fsl-imx7.c
21
+++ b/target/arm/translate-a64.c
17
@@ -XXX,XX +XXX,XX @@ static void fsl_imx7_realize(DeviceState *dev, Error **errp)
22
@@ -XXX,XX +XXX,XX @@ static void disas_pc_rel_adr(DisasContext *s, uint32_t insn)
18
sysbus_connect_irq(sbd, i, irq);
23
/*
19
irq = qdev_get_gpio_in(d, ARM_CPU_FIQ);
24
* Add/subtract (immediate)
20
sysbus_connect_irq(sbd, i + smp_cpus, irq);
25
*
21
+ irq = qdev_get_gpio_in(d, ARM_CPU_VIRQ);
26
- * 31 30 29 28 24 23 22 21 10 9 5 4 0
22
+ sysbus_connect_irq(sbd, i + 2 * smp_cpus, irq);
27
- * +--+--+--+-----------+-----+-------------+-----+-----+
23
+ irq = qdev_get_gpio_in(d, ARM_CPU_VFIQ);
28
- * |sf|op| S| 1 0 0 0 1 |shift| imm12 | Rn | Rd |
24
+ sysbus_connect_irq(sbd, i + 3 * smp_cpus, irq);
29
- * +--+--+--+-----------+-----+-------------+-----+-----+
30
+ * 31 30 29 28 23 22 21 10 9 5 4 0
31
+ * +--+--+--+-------------+--+-------------+-----+-----+
32
+ * |sf|op| S| 1 0 0 0 1 0 |sh| imm12 | Rn | Rd |
33
+ * +--+--+--+-------------+--+-------------+-----+-----+
34
*
35
* sf: 0 -> 32bit, 1 -> 64bit
36
* op: 0 -> add , 1 -> sub
37
* S: 1 -> set flags
38
- * shift: 00 -> LSL imm by 0, 01 -> LSL imm by 12
39
+ * sh: 1 -> LSL imm by 12
40
*/
41
static void disas_add_sub_imm(DisasContext *s, uint32_t insn)
42
{
43
int rd = extract32(insn, 0, 5);
44
int rn = extract32(insn, 5, 5);
45
uint64_t imm = extract32(insn, 10, 12);
46
- int shift = extract32(insn, 22, 2);
47
+ bool shift = extract32(insn, 22, 1);
48
bool setflags = extract32(insn, 29, 1);
49
bool sub_op = extract32(insn, 30, 1);
50
bool is_64bit = extract32(insn, 31, 1);
51
@@ -XXX,XX +XXX,XX @@ static void disas_add_sub_imm(DisasContext *s, uint32_t insn)
52
TCGv_i64 tcg_rd = setflags ? cpu_reg(s, rd) : cpu_reg_sp(s, rd);
53
TCGv_i64 tcg_result;
54
55
- switch (shift) {
56
- case 0x0:
57
- break;
58
- case 0x1:
59
+ if (shift) {
60
imm <<= 12;
61
- break;
62
- default:
63
- unallocated_encoding(s);
64
- return;
25
}
65
}
26
66
27
/*
67
tcg_result = tcg_temp_new_i64();
68
@@ -XXX,XX +XXX,XX @@ static void disas_data_proc_imm(DisasContext *s, uint32_t insn)
69
case 0x20: case 0x21: /* PC-rel. addressing */
70
disas_pc_rel_adr(s, insn);
71
break;
72
- case 0x22: case 0x23: /* Add/subtract (immediate) */
73
+ case 0x22: /* Add/subtract (immediate) */
74
disas_add_sub_imm(s, insn);
75
break;
76
case 0x24: /* Logical (immediate) */
28
--
77
--
29
2.18.0
78
2.20.1
30
79
31
80
diff view generated by jsdifflib
1
The IoTKit includes three different instances of the
1
From: Richard Henderson <richard.henderson@linaro.org>
2
CMSDK APB watchdog; create and wire them up.
3
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-12-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180820141116.9118-7-peter.maydell@linaro.org
8
---
7
---
9
include/hw/arm/iotkit.h | 6 +++++
8
target/arm/helper-a64.h | 1 +
10
hw/arm/iotkit.c | 58 ++++++++++++++++++++++++++++++++++++++---
9
target/arm/internals.h | 9 +++++++
11
2 files changed, 61 insertions(+), 3 deletions(-)
10
target/arm/mte_helper.c | 10 ++++++++
11
target/arm/translate-a64.c | 51 ++++++++++++++++++++++++++++++++++++++
12
4 files changed, 71 insertions(+)
12
13
13
diff --git a/include/hw/arm/iotkit.h b/include/hw/arm/iotkit.h
14
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
14
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
15
--- a/include/hw/arm/iotkit.h
16
--- a/target/arm/helper-a64.h
16
+++ b/include/hw/arm/iotkit.h
17
+++ b/target/arm/helper-a64.h
17
@@ -XXX,XX +XXX,XX @@
18
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(xpaci, TCG_CALL_NO_RWG_SE, i64, env, i64)
18
#include "hw/misc/tz-mpc.h"
19
DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
19
#include "hw/timer/cmsdk-apb-timer.h"
20
20
#include "hw/timer/cmsdk-apb-dualtimer.h"
21
DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
21
+#include "hw/watchdog/cmsdk-apb-watchdog.h"
22
+DEF_HELPER_FLAGS_4(addsubg, TCG_CALL_NO_RWG_SE, i64, env, i64, s32, i32)
22
#include "hw/misc/unimp.h"
23
diff --git a/target/arm/internals.h b/target/arm/internals.h
23
#include "hw/or-irq.h"
24
index XXXXXXX..XXXXXXX 100644
24
#include "hw/core/split-irq.h"
25
--- a/target/arm/internals.h
25
@@ -XXX,XX +XXX,XX @@ typedef struct IoTKit {
26
+++ b/target/arm/internals.h
26
SplitIRQ ppc_irq_splitter[NUM_PPCS];
27
@@ -XXX,XX +XXX,XX @@ void arm_log_exception(int idx);
27
SplitIRQ mpc_irq_splitter[IOTS_NUM_EXP_MPC + IOTS_NUM_MPC];
28
*/
28
qemu_or_irq mpc_irq_orgate;
29
#define GMID_EL1_BS 6
29
+ qemu_or_irq nmi_orgate;
30
30
31
+/* We associate one allocation tag per 16 bytes, the minimum. */
31
CMSDKAPBDualTimer dualtimer;
32
+#define LOG2_TAG_GRANULE 4
32
UnimplementedDeviceState s32ktimer;
33
+#define TAG_GRANULE (1 << LOG2_TAG_GRANULE)
33
34
+ CMSDKAPBWatchdog s32kwatchdog;
35
+ CMSDKAPBWatchdog nswatchdog;
36
+ CMSDKAPBWatchdog swatchdog;
37
+
34
+
38
MemoryRegion container;
35
+static inline int allocation_tag_from_addr(uint64_t ptr)
39
MemoryRegion alias1;
36
+{
40
MemoryRegion alias2;
37
+ return extract64(ptr, 56, 4);
41
diff --git a/hw/arm/iotkit.c b/hw/arm/iotkit.c
38
+}
39
+
40
static inline uint64_t address_with_allocation_tag(uint64_t ptr, int rtag)
41
{
42
return deposit64(ptr, 56, 4, rtag);
43
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
42
index XXXXXXX..XXXXXXX 100644
44
index XXXXXXX..XXXXXXX 100644
43
--- a/hw/arm/iotkit.c
45
--- a/target/arm/mte_helper.c
44
+++ b/hw/arm/iotkit.c
46
+++ b/target/arm/mte_helper.c
45
@@ -XXX,XX +XXX,XX @@
47
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(irg)(CPUARMState *env, uint64_t rn, uint64_t rm)
46
#include "hw/misc/unimp.h"
48
47
#include "hw/arm/arm.h"
49
return address_with_allocation_tag(rn, rtag);
48
50
}
49
+/* Clock frequency in HZ of the 32KHz "slow clock" */
50
+#define S32KCLK (32 * 1000)
51
+
51
+
52
/* Create an alias region of @size bytes starting at @base
52
+uint64_t HELPER(addsubg)(CPUARMState *env, uint64_t ptr,
53
* which mirrors the memory starting at @orig.
53
+ int32_t offset, uint32_t tag_offset)
54
*/
54
+{
55
@@ -XXX,XX +XXX,XX @@ static void iotkit_init(Object *obj)
55
+ int start_tag = allocation_tag_from_addr(ptr);
56
TYPE_CMSDK_APB_TIMER);
56
+ uint16_t exclude = extract32(env->cp15.gcr_el1, 0, 16);
57
sysbus_init_child_obj(obj, "dualtimer", &s->dualtimer, sizeof(s->dualtimer),
57
+ int rtag = choose_nonexcluded_tag(start_tag, tag_offset, exclude);
58
TYPE_CMSDK_APB_DUALTIMER);
59
+ sysbus_init_child_obj(obj, "s32kwatchdog", &s->s32kwatchdog,
60
+ sizeof(s->s32kwatchdog), TYPE_CMSDK_APB_WATCHDOG);
61
+ sysbus_init_child_obj(obj, "nswatchdog", &s->nswatchdog,
62
+ sizeof(s->nswatchdog), TYPE_CMSDK_APB_WATCHDOG);
63
+ sysbus_init_child_obj(obj, "swatchdog", &s->swatchdog,
64
+ sizeof(s->swatchdog), TYPE_CMSDK_APB_WATCHDOG);
65
+ object_initialize_child(obj, "nmi-orgate", &s->nmi_orgate,
66
+ sizeof(s->nmi_orgate), TYPE_OR_IRQ,
67
+ &error_abort, NULL);
68
object_initialize_child(obj, "ppc-irq-orgate", &s->ppc_irq_orgate,
69
sizeof(s->ppc_irq_orgate), TYPE_OR_IRQ,
70
&error_abort, NULL);
71
@@ -XXX,XX +XXX,XX @@ static void iotkit_realize(DeviceState *dev, Error **errp)
72
create_unimplemented_device("SYSINFO", 0x40020000, 0x1000);
73
74
create_unimplemented_device("SYSCONTROL", 0x50021000, 0x1000);
75
- create_unimplemented_device("S32KWATCHDOG", 0x5002e000, 0x1000);
76
+
58
+
77
+ /* This OR gate wires together outputs from the secure watchdogs to NMI */
59
+ return address_with_allocation_tag(ptr + offset, rtag);
78
+ object_property_set_int(OBJECT(&s->nmi_orgate), 2, "num-lines", &err);
60
+}
79
+ if (err) {
61
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
80
+ error_propagate(errp, err);
62
index XXXXXXX..XXXXXXX 100644
63
--- a/target/arm/translate-a64.c
64
+++ b/target/arm/translate-a64.c
65
@@ -XXX,XX +XXX,XX @@ static void disas_add_sub_imm(DisasContext *s, uint32_t insn)
66
tcg_temp_free_i64(tcg_result);
67
}
68
69
+/*
70
+ * Add/subtract (immediate, with tags)
71
+ *
72
+ * 31 30 29 28 23 22 21 16 14 10 9 5 4 0
73
+ * +--+--+--+-------------+--+---------+--+-------+-----+-----+
74
+ * |sf|op| S| 1 0 0 0 1 1 |o2| uimm6 |o3| uimm4 | Rn | Rd |
75
+ * +--+--+--+-------------+--+---------+--+-------+-----+-----+
76
+ *
77
+ * op: 0 -> add, 1 -> sub
78
+ */
79
+static void disas_add_sub_imm_with_tags(DisasContext *s, uint32_t insn)
80
+{
81
+ int rd = extract32(insn, 0, 5);
82
+ int rn = extract32(insn, 5, 5);
83
+ int uimm4 = extract32(insn, 10, 4);
84
+ int uimm6 = extract32(insn, 16, 6);
85
+ bool sub_op = extract32(insn, 30, 1);
86
+ TCGv_i64 tcg_rn, tcg_rd;
87
+ int imm;
88
+
89
+ /* Test all of sf=1, S=0, o2=0, o3=0. */
90
+ if ((insn & 0xa040c000u) != 0x80000000u ||
91
+ !dc_isar_feature(aa64_mte_insn_reg, s)) {
92
+ unallocated_encoding(s);
81
+ return;
93
+ return;
82
+ }
94
+ }
83
+ object_property_set_bool(OBJECT(&s->nmi_orgate), true, "realized", &err);
95
+
84
+ if (err) {
96
+ imm = uimm6 << LOG2_TAG_GRANULE;
85
+ error_propagate(errp, err);
97
+ if (sub_op) {
86
+ return;
98
+ imm = -imm;
87
+ }
99
+ }
88
+ qdev_connect_gpio_out(DEVICE(&s->nmi_orgate), 0,
89
+ qdev_get_gpio_in_named(DEVICE(&s->armv7m), "NMI", 0));
90
+
100
+
91
+ qdev_prop_set_uint32(DEVICE(&s->s32kwatchdog), "wdogclk-frq", S32KCLK);
101
+ tcg_rn = cpu_reg_sp(s, rn);
92
+ object_property_set_bool(OBJECT(&s->s32kwatchdog), true, "realized", &err);
102
+ tcg_rd = cpu_reg_sp(s, rd);
93
+ if (err) {
103
+
94
+ error_propagate(errp, err);
104
+ if (s->ata) {
95
+ return;
105
+ TCGv_i32 offset = tcg_const_i32(imm);
106
+ TCGv_i32 tag_offset = tcg_const_i32(uimm4);
107
+
108
+ gen_helper_addsubg(tcg_rd, cpu_env, tcg_rn, offset, tag_offset);
109
+ tcg_temp_free_i32(tag_offset);
110
+ tcg_temp_free_i32(offset);
111
+ } else {
112
+ tcg_gen_addi_i64(tcg_rd, tcg_rn, imm);
113
+ gen_address_with_allocation_tag0(tcg_rd, tcg_rd);
96
+ }
114
+ }
97
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->s32kwatchdog), 0,
115
+}
98
+ qdev_get_gpio_in(DEVICE(&s->nmi_orgate), 0));
99
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->s32kwatchdog), 0, 0x5002e000);
100
101
/* 0x40080000 .. 0x4008ffff : IoTKit second Base peripheral region */
102
103
- create_unimplemented_device("NS watchdog", 0x40081000, 0x1000);
104
- create_unimplemented_device("S watchdog", 0x50081000, 0x1000);
105
+ qdev_prop_set_uint32(DEVICE(&s->nswatchdog), "wdogclk-frq", s->mainclk_frq);
106
+ object_property_set_bool(OBJECT(&s->nswatchdog), true, "realized", &err);
107
+ if (err) {
108
+ error_propagate(errp, err);
109
+ return;
110
+ }
111
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->nswatchdog), 0,
112
+ qdev_get_gpio_in(DEVICE(&s->armv7m), 1));
113
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->nswatchdog), 0, 0x40081000);
114
+
116
+
115
+ qdev_prop_set_uint32(DEVICE(&s->swatchdog), "wdogclk-frq", s->mainclk_frq);
117
/* The input should be a value in the bottom e bits (with higher
116
+ object_property_set_bool(OBJECT(&s->swatchdog), true, "realized", &err);
118
* bits zero); returns that value replicated into every element
117
+ if (err) {
119
* of size e in a 64 bit integer.
118
+ error_propagate(errp, err);
120
@@ -XXX,XX +XXX,XX @@ static void disas_data_proc_imm(DisasContext *s, uint32_t insn)
119
+ return;
121
case 0x22: /* Add/subtract (immediate) */
120
+ }
122
disas_add_sub_imm(s, insn);
121
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->swatchdog), 0,
123
break;
122
+ qdev_get_gpio_in(DEVICE(&s->nmi_orgate), 1));
124
+ case 0x23: /* Add/subtract (immediate, with tags) */
123
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->swatchdog), 0, 0x50081000);
125
+ disas_add_sub_imm_with_tags(s, insn);
124
126
+ break;
125
for (i = 0; i < ARRAY_SIZE(s->ppc_irq_splitter); i++) {
127
case 0x24: /* Logical (immediate) */
126
Object *splitter = OBJECT(&s->ppc_irq_splitter[i]);
128
disas_logic_imm(s, insn);
129
break;
127
--
130
--
128
2.18.0
131
2.20.1
129
132
130
133
diff view generated by jsdifflib
1
Untabify the arm iwmmxt_helper.c. This affects only the iwMMXt code.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
We've never touched that code in years, so it's not going to get
3
fixed up by our "change when touched" process, and a bulk change is
4
not going to be too disruptive.
5
2
6
This commit was produced using Emacs "untabify" (plus one
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
by-hand removal of a space to fix a checkpatch nit); it is
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
a whitespace-only change.
5
Message-id: 20200626033144.790098-13-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
8
target/arm/translate-a64.c | 15 +++++++++++++++
9
1 file changed, 15 insertions(+)
9
10
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
11
Message-id: 20180821165215.29069-3-peter.maydell@linaro.org
12
---
13
target/arm/iwmmxt_helper.c | 234 ++++++++++++++++++-------------------
14
1 file changed, 117 insertions(+), 117 deletions(-)
15
16
diff --git a/target/arm/iwmmxt_helper.c b/target/arm/iwmmxt_helper.c
17
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
18
--- a/target/arm/iwmmxt_helper.c
13
--- a/target/arm/translate-a64.c
19
+++ b/target/arm/iwmmxt_helper.c
14
+++ b/target/arm/translate-a64.c
20
@@ -XXX,XX +XXX,XX @@
15
@@ -XXX,XX +XXX,XX @@ static void disas_data_proc_2src(DisasContext *s, uint32_t insn)
21
/* iwMMXt macros extracted from GNU gdb. */
16
cpu_reg_sp(s, rn));
22
17
}
23
/* Set the SIMD wCASF flags for 8, 16, 32 or 64-bit operations. */
18
break;
24
-#define SIMD8_SET( v, n, b)    ((v != 0) << ((((b) + 1) * 4) + (n)))
19
+ case 5: /* GMI */
25
-#define SIMD16_SET(v, n, h)    ((v != 0) << ((((h) + 1) * 8) + (n)))
20
+ if (sf == 0 || !dc_isar_feature(aa64_mte_insn_reg, s)) {
26
-#define SIMD32_SET(v, n, w)    ((v != 0) << ((((w) + 1) * 16) + (n)))
21
+ goto do_unallocated;
27
-#define SIMD64_SET(v, n)    ((v != 0) << (32 + (n)))
22
+ } else {
28
+#define SIMD8_SET(v, n, b) ((v != 0) << ((((b) + 1) * 4) + (n)))
23
+ TCGv_i64 t1 = tcg_const_i64(1);
29
+#define SIMD16_SET(v, n, h) ((v != 0) << ((((h) + 1) * 8) + (n)))
24
+ TCGv_i64 t2 = tcg_temp_new_i64();
30
+#define SIMD32_SET(v, n, w) ((v != 0) << ((((w) + 1) * 16) + (n)))
25
+
31
+#define SIMD64_SET(v, n) ((v != 0) << (32 + (n)))
26
+ tcg_gen_extract_i64(t2, cpu_reg_sp(s, rn), 56, 4);
32
/* Flags to pass as "n" above. */
27
+ tcg_gen_shl_i64(t1, t1, t2);
33
-#define SIMD_NBIT    -1
28
+ tcg_gen_or_i64(cpu_reg(s, rd), cpu_reg(s, rm), t1);
34
-#define SIMD_ZBIT    -2
29
+
35
-#define SIMD_CBIT    -3
30
+ tcg_temp_free_i64(t1);
36
-#define SIMD_VBIT    -4
31
+ tcg_temp_free_i64(t2);
37
+#define SIMD_NBIT -1
32
+ }
38
+#define SIMD_ZBIT -2
33
+ break;
39
+#define SIMD_CBIT -3
34
case 8: /* LSLV */
40
+#define SIMD_VBIT -4
35
handle_shift_reg(s, A64_SHIFT_TYPE_LSL, sf, rm, rn, rd);
41
/* Various status bit macros. */
36
break;
42
-#define NBIT8(x)    ((x) & 0x80)
43
-#define NBIT16(x)    ((x) & 0x8000)
44
-#define NBIT32(x)    ((x) & 0x80000000)
45
-#define NBIT64(x)    ((x) & 0x8000000000000000ULL)
46
-#define ZBIT8(x)    (((x) & 0xff) == 0)
47
-#define ZBIT16(x)    (((x) & 0xffff) == 0)
48
-#define ZBIT32(x)    (((x) & 0xffffffff) == 0)
49
-#define ZBIT64(x)    (x == 0)
50
+#define NBIT8(x) ((x) & 0x80)
51
+#define NBIT16(x) ((x) & 0x8000)
52
+#define NBIT32(x) ((x) & 0x80000000)
53
+#define NBIT64(x) ((x) & 0x8000000000000000ULL)
54
+#define ZBIT8(x) (((x) & 0xff) == 0)
55
+#define ZBIT16(x) (((x) & 0xffff) == 0)
56
+#define ZBIT32(x) (((x) & 0xffffffff) == 0)
57
+#define ZBIT64(x) (x == 0)
58
/* Sign extension macros. */
59
-#define EXTEND8H(a)    ((uint16_t) (int8_t) (a))
60
-#define EXTEND8(a)    ((uint32_t) (int8_t) (a))
61
-#define EXTEND16(a)    ((uint32_t) (int16_t) (a))
62
-#define EXTEND16S(a)    ((int32_t) (int16_t) (a))
63
-#define EXTEND32(a)    ((uint64_t) (int32_t) (a))
64
+#define EXTEND8H(a) ((uint16_t) (int8_t) (a))
65
+#define EXTEND8(a) ((uint32_t) (int8_t) (a))
66
+#define EXTEND16(a) ((uint32_t) (int16_t) (a))
67
+#define EXTEND16S(a) ((int32_t) (int16_t) (a))
68
+#define EXTEND32(a) ((uint64_t) (int32_t) (a))
69
70
uint64_t HELPER(iwmmxt_maddsq)(uint64_t a, uint64_t b)
71
{
72
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(iwmmxt_macuw)(uint64_t a, uint64_t b)
73
#define NZBIT64(x) \
74
SIMD64_SET(NBIT64(x), SIMD_NBIT) | \
75
SIMD64_SET(ZBIT64(x), SIMD_ZBIT)
76
-#define IWMMXT_OP_UNPACK(S, SH0, SH1, SH2, SH3)            \
77
+#define IWMMXT_OP_UNPACK(S, SH0, SH1, SH2, SH3) \
78
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, b)))(CPUARMState *env, \
79
uint64_t a, uint64_t b) \
80
-{                                \
81
- a =                             \
82
- (((a >> SH0) & 0xff) << 0) | (((b >> SH0) & 0xff) << 8) |    \
83
- (((a >> SH1) & 0xff) << 16) | (((b >> SH1) & 0xff) << 24) |    \
84
- (((a >> SH2) & 0xff) << 32) | (((b >> SH2) & 0xff) << 40) |    \
85
- (((a >> SH3) & 0xff) << 48) | (((b >> SH3) & 0xff) << 56);    \
86
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
87
- NZBIT8(a >> 0, 0) | NZBIT8(a >> 8, 1) |         \
88
- NZBIT8(a >> 16, 2) | NZBIT8(a >> 24, 3) |        \
89
- NZBIT8(a >> 32, 4) | NZBIT8(a >> 40, 5) |        \
90
- NZBIT8(a >> 48, 6) | NZBIT8(a >> 56, 7);        \
91
+{ \
92
+ a = \
93
+ (((a >> SH0) & 0xff) << 0) | (((b >> SH0) & 0xff) << 8) | \
94
+ (((a >> SH1) & 0xff) << 16) | (((b >> SH1) & 0xff) << 24) | \
95
+ (((a >> SH2) & 0xff) << 32) | (((b >> SH2) & 0xff) << 40) | \
96
+ (((a >> SH3) & 0xff) << 48) | (((b >> SH3) & 0xff) << 56); \
97
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
98
+ NZBIT8(a >> 0, 0) | NZBIT8(a >> 8, 1) | \
99
+ NZBIT8(a >> 16, 2) | NZBIT8(a >> 24, 3) | \
100
+ NZBIT8(a >> 32, 4) | NZBIT8(a >> 40, 5) | \
101
+ NZBIT8(a >> 48, 6) | NZBIT8(a >> 56, 7); \
102
return a; \
103
-}                                \
104
+} \
105
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, w)))(CPUARMState *env, \
106
uint64_t a, uint64_t b) \
107
-{                                \
108
- a =                             \
109
- (((a >> SH0) & 0xffff) << 0) |                \
110
- (((b >> SH0) & 0xffff) << 16) |             \
111
- (((a >> SH2) & 0xffff) << 32) |             \
112
- (((b >> SH2) & 0xffff) << 48);                \
113
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
114
- NZBIT8(a >> 0, 0) | NZBIT8(a >> 16, 1) |        \
115
- NZBIT8(a >> 32, 2) | NZBIT8(a >> 48, 3);        \
116
+{ \
117
+ a = \
118
+ (((a >> SH0) & 0xffff) << 0) | \
119
+ (((b >> SH0) & 0xffff) << 16) | \
120
+ (((a >> SH2) & 0xffff) << 32) | \
121
+ (((b >> SH2) & 0xffff) << 48); \
122
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
123
+ NZBIT8(a >> 0, 0) | NZBIT8(a >> 16, 1) | \
124
+ NZBIT8(a >> 32, 2) | NZBIT8(a >> 48, 3); \
125
return a; \
126
-}                                \
127
+} \
128
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, l)))(CPUARMState *env, \
129
uint64_t a, uint64_t b) \
130
-{                                \
131
- a =                             \
132
- (((a >> SH0) & 0xffffffff) << 0) |            \
133
- (((b >> SH0) & 0xffffffff) << 32);            \
134
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
135
- NZBIT32(a >> 0, 0) | NZBIT32(a >> 32, 1);        \
136
+{ \
137
+ a = \
138
+ (((a >> SH0) & 0xffffffff) << 0) | \
139
+ (((b >> SH0) & 0xffffffff) << 32); \
140
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
141
+ NZBIT32(a >> 0, 0) | NZBIT32(a >> 32, 1); \
142
return a; \
143
-}                                \
144
+} \
145
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, ub)))(CPUARMState *env, \
146
uint64_t x) \
147
-{                                \
148
- x =                             \
149
- (((x >> SH0) & 0xff) << 0) |                \
150
- (((x >> SH1) & 0xff) << 16) |                \
151
- (((x >> SH2) & 0xff) << 32) |                \
152
- (((x >> SH3) & 0xff) << 48);                \
153
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
154
- NZBIT16(x >> 0, 0) | NZBIT16(x >> 16, 1) |        \
155
- NZBIT16(x >> 32, 2) | NZBIT16(x >> 48, 3);        \
156
+{ \
157
+ x = \
158
+ (((x >> SH0) & 0xff) << 0) | \
159
+ (((x >> SH1) & 0xff) << 16) | \
160
+ (((x >> SH2) & 0xff) << 32) | \
161
+ (((x >> SH3) & 0xff) << 48); \
162
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
163
+ NZBIT16(x >> 0, 0) | NZBIT16(x >> 16, 1) | \
164
+ NZBIT16(x >> 32, 2) | NZBIT16(x >> 48, 3); \
165
return x; \
166
-}                                \
167
+} \
168
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, uw)))(CPUARMState *env, \
169
uint64_t x) \
170
-{                                \
171
- x =                             \
172
- (((x >> SH0) & 0xffff) << 0) |                \
173
- (((x >> SH2) & 0xffff) << 32);                \
174
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
175
- NZBIT32(x >> 0, 0) | NZBIT32(x >> 32, 1);        \
176
+{ \
177
+ x = \
178
+ (((x >> SH0) & 0xffff) << 0) | \
179
+ (((x >> SH2) & 0xffff) << 32); \
180
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
181
+ NZBIT32(x >> 0, 0) | NZBIT32(x >> 32, 1); \
182
return x; \
183
-}                                \
184
+} \
185
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, ul)))(CPUARMState *env, \
186
uint64_t x) \
187
-{                                \
188
- x = (((x >> SH0) & 0xffffffff) << 0);            \
189
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = NZBIT64(x >> 0);    \
190
+{ \
191
+ x = (((x >> SH0) & 0xffffffff) << 0); \
192
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = NZBIT64(x >> 0); \
193
return x; \
194
-}                                \
195
+} \
196
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, sb)))(CPUARMState *env, \
197
uint64_t x) \
198
-{                                \
199
- x =                             \
200
- ((uint64_t) EXTEND8H((x >> SH0) & 0xff) << 0) |     \
201
- ((uint64_t) EXTEND8H((x >> SH1) & 0xff) << 16) |    \
202
- ((uint64_t) EXTEND8H((x >> SH2) & 0xff) << 32) |    \
203
- ((uint64_t) EXTEND8H((x >> SH3) & 0xff) << 48);     \
204
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
205
- NZBIT16(x >> 0, 0) | NZBIT16(x >> 16, 1) |        \
206
- NZBIT16(x >> 32, 2) | NZBIT16(x >> 48, 3);        \
207
+{ \
208
+ x = \
209
+ ((uint64_t) EXTEND8H((x >> SH0) & 0xff) << 0) | \
210
+ ((uint64_t) EXTEND8H((x >> SH1) & 0xff) << 16) | \
211
+ ((uint64_t) EXTEND8H((x >> SH2) & 0xff) << 32) | \
212
+ ((uint64_t) EXTEND8H((x >> SH3) & 0xff) << 48); \
213
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
214
+ NZBIT16(x >> 0, 0) | NZBIT16(x >> 16, 1) | \
215
+ NZBIT16(x >> 32, 2) | NZBIT16(x >> 48, 3); \
216
return x; \
217
-}                                \
218
+} \
219
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, sw)))(CPUARMState *env, \
220
uint64_t x) \
221
-{                                \
222
- x =                             \
223
- ((uint64_t) EXTEND16((x >> SH0) & 0xffff) << 0) |    \
224
- ((uint64_t) EXTEND16((x >> SH2) & 0xffff) << 32);    \
225
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
226
- NZBIT32(x >> 0, 0) | NZBIT32(x >> 32, 1);        \
227
+{ \
228
+ x = \
229
+ ((uint64_t) EXTEND16((x >> SH0) & 0xffff) << 0) | \
230
+ ((uint64_t) EXTEND16((x >> SH2) & 0xffff) << 32); \
231
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
232
+ NZBIT32(x >> 0, 0) | NZBIT32(x >> 32, 1); \
233
return x; \
234
-}                                \
235
+} \
236
uint64_t HELPER(glue(iwmmxt_unpack, glue(S, sl)))(CPUARMState *env, \
237
uint64_t x) \
238
-{                                \
239
- x = EXTEND32((x >> SH0) & 0xffffffff);            \
240
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = NZBIT64(x >> 0);    \
241
+{ \
242
+ x = EXTEND32((x >> SH0) & 0xffffffff); \
243
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = NZBIT64(x >> 0); \
244
return x; \
245
}
246
IWMMXT_OP_UNPACK(l, 0, 8, 16, 24)
247
IWMMXT_OP_UNPACK(h, 32, 40, 48, 56)
248
249
-#define IWMMXT_OP_CMP(SUFF, Tb, Tw, Tl, O)            \
250
+#define IWMMXT_OP_CMP(SUFF, Tb, Tw, Tl, O) \
251
uint64_t HELPER(glue(iwmmxt_, glue(SUFF, b)))(CPUARMState *env, \
252
uint64_t a, uint64_t b) \
253
-{                                \
254
- a =                             \
255
- CMP(0, Tb, O, 0xff) | CMP(8, Tb, O, 0xff) |        \
256
- CMP(16, Tb, O, 0xff) | CMP(24, Tb, O, 0xff) |        \
257
- CMP(32, Tb, O, 0xff) | CMP(40, Tb, O, 0xff) |        \
258
- CMP(48, Tb, O, 0xff) | CMP(56, Tb, O, 0xff);        \
259
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
260
- NZBIT8(a >> 0, 0) | NZBIT8(a >> 8, 1) |         \
261
- NZBIT8(a >> 16, 2) | NZBIT8(a >> 24, 3) |        \
262
- NZBIT8(a >> 32, 4) | NZBIT8(a >> 40, 5) |        \
263
- NZBIT8(a >> 48, 6) | NZBIT8(a >> 56, 7);        \
264
+{ \
265
+ a = \
266
+ CMP(0, Tb, O, 0xff) | CMP(8, Tb, O, 0xff) | \
267
+ CMP(16, Tb, O, 0xff) | CMP(24, Tb, O, 0xff) | \
268
+ CMP(32, Tb, O, 0xff) | CMP(40, Tb, O, 0xff) | \
269
+ CMP(48, Tb, O, 0xff) | CMP(56, Tb, O, 0xff); \
270
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
271
+ NZBIT8(a >> 0, 0) | NZBIT8(a >> 8, 1) | \
272
+ NZBIT8(a >> 16, 2) | NZBIT8(a >> 24, 3) | \
273
+ NZBIT8(a >> 32, 4) | NZBIT8(a >> 40, 5) | \
274
+ NZBIT8(a >> 48, 6) | NZBIT8(a >> 56, 7); \
275
return a; \
276
-}                                \
277
+} \
278
uint64_t HELPER(glue(iwmmxt_, glue(SUFF, w)))(CPUARMState *env, \
279
uint64_t a, uint64_t b) \
280
-{                                \
281
- a = CMP(0, Tw, O, 0xffff) | CMP(16, Tw, O, 0xffff) |    \
282
- CMP(32, Tw, O, 0xffff) | CMP(48, Tw, O, 0xffff);    \
283
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
284
- NZBIT16(a >> 0, 0) | NZBIT16(a >> 16, 1) |        \
285
- NZBIT16(a >> 32, 2) | NZBIT16(a >> 48, 3);        \
286
+{ \
287
+ a = CMP(0, Tw, O, 0xffff) | CMP(16, Tw, O, 0xffff) | \
288
+ CMP(32, Tw, O, 0xffff) | CMP(48, Tw, O, 0xffff); \
289
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
290
+ NZBIT16(a >> 0, 0) | NZBIT16(a >> 16, 1) | \
291
+ NZBIT16(a >> 32, 2) | NZBIT16(a >> 48, 3); \
292
return a; \
293
-}                                \
294
+} \
295
uint64_t HELPER(glue(iwmmxt_, glue(SUFF, l)))(CPUARMState *env, \
296
uint64_t a, uint64_t b) \
297
-{                                \
298
- a = CMP(0, Tl, O, 0xffffffff) |                \
299
- CMP(32, Tl, O, 0xffffffff);                \
300
- env->iwmmxt.cregs[ARM_IWMMXT_wCASF] =            \
301
- NZBIT32(a >> 0, 0) | NZBIT32(a >> 32, 1);        \
302
+{ \
303
+ a = CMP(0, Tl, O, 0xffffffff) | \
304
+ CMP(32, Tl, O, 0xffffffff); \
305
+ env->iwmmxt.cregs[ARM_IWMMXT_wCASF] = \
306
+ NZBIT32(a >> 0, 0) | NZBIT32(a >> 32, 1); \
307
return a; \
308
}
309
#define CMP(SHR, TYPE, OPER, MASK) ((((TYPE) ((a >> SHR) & MASK) OPER \
310
--
37
--
311
2.18.0
38
2.20.1
312
39
313
40
diff view generated by jsdifflib
1
Now we have a model of the CMSDK dual timer, we can wire it
1
From: Richard Henderson <richard.henderson@linaro.org>
2
up in the IoTKit.
3
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-14-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180820141116.9118-5-peter.maydell@linaro.org
8
---
7
---
9
include/hw/arm/iotkit.h | 3 ++-
8
target/arm/translate-a64.c | 24 ++++++++++++++++++++++--
10
hw/arm/iotkit.c | 8 +++++---
9
1 file changed, 22 insertions(+), 2 deletions(-)
11
2 files changed, 7 insertions(+), 4 deletions(-)
12
10
13
diff --git a/include/hw/arm/iotkit.h b/include/hw/arm/iotkit.h
11
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
14
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
15
--- a/include/hw/arm/iotkit.h
13
--- a/target/arm/translate-a64.c
16
+++ b/include/hw/arm/iotkit.h
14
+++ b/target/arm/translate-a64.c
17
@@ -XXX,XX +XXX,XX @@
15
@@ -XXX,XX +XXX,XX @@ static void handle_crc32(DisasContext *s,
18
#include "hw/misc/tz-ppc.h"
16
*/
19
#include "hw/misc/tz-mpc.h"
17
static void disas_data_proc_2src(DisasContext *s, uint32_t insn)
20
#include "hw/timer/cmsdk-apb-timer.h"
18
{
21
+#include "hw/timer/cmsdk-apb-dualtimer.h"
19
- unsigned int sf, rm, opcode, rn, rd;
22
#include "hw/misc/unimp.h"
20
+ unsigned int sf, rm, opcode, rn, rd, setflag;
23
#include "hw/or-irq.h"
21
sf = extract32(insn, 31, 1);
24
#include "hw/core/split-irq.h"
22
+ setflag = extract32(insn, 29, 1);
25
@@ -XXX,XX +XXX,XX @@ typedef struct IoTKit {
23
rm = extract32(insn, 16, 5);
26
SplitIRQ mpc_irq_splitter[IOTS_NUM_EXP_MPC + IOTS_NUM_MPC];
24
opcode = extract32(insn, 10, 6);
27
qemu_or_irq mpc_irq_orgate;
25
rn = extract32(insn, 5, 5);
28
26
rd = extract32(insn, 0, 5);
29
- UnimplementedDeviceState dualtimer;
27
30
+ CMSDKAPBDualTimer dualtimer;
28
- if (extract32(insn, 29, 1)) {
31
UnimplementedDeviceState s32ktimer;
29
+ if (setflag && opcode != 0) {
32
30
unallocated_encoding(s);
33
MemoryRegion container;
34
diff --git a/hw/arm/iotkit.c b/hw/arm/iotkit.c
35
index XXXXXXX..XXXXXXX 100644
36
--- a/hw/arm/iotkit.c
37
+++ b/hw/arm/iotkit.c
38
@@ -XXX,XX +XXX,XX @@ static void iotkit_init(Object *obj)
39
sysbus_init_child_obj(obj, "timer1", &s->timer1, sizeof(s->timer1),
40
TYPE_CMSDK_APB_TIMER);
41
sysbus_init_child_obj(obj, "dualtimer", &s->dualtimer, sizeof(s->dualtimer),
42
- TYPE_UNIMPLEMENTED_DEVICE);
43
+ TYPE_CMSDK_APB_DUALTIMER);
44
object_initialize_child(obj, "ppc-irq-orgate", &s->ppc_irq_orgate,
45
sizeof(s->ppc_irq_orgate), TYPE_OR_IRQ,
46
&error_abort, NULL);
47
@@ -XXX,XX +XXX,XX @@ static void iotkit_realize(DeviceState *dev, Error **errp)
48
return;
31
return;
49
}
32
}
50
33
51
- qdev_prop_set_string(DEVICE(&s->dualtimer), "name", "Dual timer");
34
switch (opcode) {
52
- qdev_prop_set_uint64(DEVICE(&s->dualtimer), "size", 0x1000);
35
+ case 0: /* SUBP(S) */
36
+ if (sf == 0 || !dc_isar_feature(aa64_mte_insn_reg, s)) {
37
+ goto do_unallocated;
38
+ } else {
39
+ TCGv_i64 tcg_n, tcg_m, tcg_d;
53
+
40
+
54
+ qdev_prop_set_uint32(DEVICE(&s->dualtimer), "pclk-frq", s->mainclk_frq);
41
+ tcg_n = read_cpu_reg_sp(s, rn, true);
55
object_property_set_bool(OBJECT(&s->dualtimer), true, "realized", &err);
42
+ tcg_m = read_cpu_reg_sp(s, rm, true);
56
if (err) {
43
+ tcg_gen_sextract_i64(tcg_n, tcg_n, 0, 56);
57
error_propagate(errp, err);
44
+ tcg_gen_sextract_i64(tcg_m, tcg_m, 0, 56);
58
return;
45
+ tcg_d = cpu_reg(s, rd);
59
}
46
+
60
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->dualtimer), 0,
47
+ if (setflag) {
61
+ qdev_get_gpio_in(DEVICE(&s->armv7m), 5));
48
+ gen_sub_CC(true, tcg_d, tcg_n, tcg_m);
62
mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(&s->dualtimer), 0);
49
+ } else {
63
object_property_set_link(OBJECT(&s->apb_ppc0), OBJECT(mr), "port[2]", &err);
50
+ tcg_gen_sub_i64(tcg_d, tcg_n, tcg_m);
64
if (err) {
51
+ }
52
+ }
53
+ break;
54
case 2: /* UDIV */
55
handle_div(s, false, sf, rm, rn, rd);
56
break;
65
--
57
--
66
2.18.0
58
2.20.1
67
59
68
60
diff view generated by jsdifflib
1
Move from the legacy SysBusDevice::init method to using
1
From: Richard Henderson <richard.henderson@linaro.org>
2
DeviceState::realize.
3
2
3
Use the same code as system mode, so that we generate the same
4
exception + syndrome for the unaligned access.
5
6
For the moment, if MTE is enabled so that this path is reachable,
7
this would generate a SIGSEGV in the user-only cpu_loop. Decoding
8
the syndrome to produce the proper SIGBUS will be done later.
9
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
12
Message-id: 20200626033144.790098-15-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Message-id: 20180820141116.9118-19-peter.maydell@linaro.org
7
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
---
14
---
9
hw/ssi/pl022.c | 8 +++-----
15
target/arm/cpu.c | 2 +-
10
1 file changed, 3 insertions(+), 5 deletions(-)
16
target/arm/tlb_helper.c | 4 ++--
17
2 files changed, 3 insertions(+), 3 deletions(-)
11
18
12
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
19
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
13
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/ssi/pl022.c
21
--- a/target/arm/cpu.c
15
+++ b/hw/ssi/pl022.c
22
+++ b/target/arm/cpu.c
16
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_pl022 = {
23
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_class_init(ObjectClass *oc, void *data)
17
}
24
cc->tlb_fill = arm_cpu_tlb_fill;
18
};
25
cc->debug_excp_handler = arm_debug_excp_handler;
19
26
cc->debug_check_watchpoint = arm_debug_check_watchpoint;
20
-static int pl022_init(SysBusDevice *sbd)
27
-#if !defined(CONFIG_USER_ONLY)
21
+static void pl022_realize(DeviceState *dev, Error **errp)
28
cc->do_unaligned_access = arm_cpu_do_unaligned_access;
22
{
29
+#if !defined(CONFIG_USER_ONLY)
23
- DeviceState *dev = DEVICE(sbd);
30
cc->do_transaction_failed = arm_cpu_do_transaction_failed;
24
+ SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
31
cc->adjust_watchpoint_address = arm_adjust_watchpoint_address;
25
PL022State *s = PL022(dev);
32
#endif /* CONFIG_TCG && !CONFIG_USER_ONLY */
26
33
diff --git a/target/arm/tlb_helper.c b/target/arm/tlb_helper.c
27
memory_region_init_io(&s->iomem, OBJECT(s), &pl022_ops, s, "pl022", 0x1000);
34
index XXXXXXX..XXXXXXX 100644
28
sysbus_init_mmio(sbd, &s->iomem);
35
--- a/target/arm/tlb_helper.c
29
sysbus_init_irq(sbd, &s->irq);
36
+++ b/target/arm/tlb_helper.c
30
s->ssi = ssi_create_bus(dev, "ssi");
37
@@ -XXX,XX +XXX,XX @@
31
- return 0;
38
#include "internals.h"
39
#include "exec/exec-all.h"
40
41
-#if !defined(CONFIG_USER_ONLY)
42
-
43
static inline uint32_t merge_syn_data_abort(uint32_t template_syn,
44
unsigned int target_el,
45
bool same_el, bool ea,
46
@@ -XXX,XX +XXX,XX @@ void arm_cpu_do_unaligned_access(CPUState *cs, vaddr vaddr,
47
arm_deliver_fault(cpu, vaddr, access_type, mmu_idx, &fi);
32
}
48
}
33
49
34
static void pl022_class_init(ObjectClass *klass, void *data)
50
+#if !defined(CONFIG_USER_ONLY)
35
{
51
+
36
- SysBusDeviceClass *sdc = SYS_BUS_DEVICE_CLASS(klass);
52
/*
37
DeviceClass *dc = DEVICE_CLASS(klass);
53
* arm_cpu_do_transaction_failed: handle a memory system error response
38
54
* (eg "no device/memory present at address") by raising an external abort
39
- sdc->init = pl022_init;
40
dc->reset = pl022_reset;
41
dc->vmsd = &vmstate_pl022;
42
+ dc->realize = pl022_realize;
43
}
44
45
static const TypeInfo pl022_info = {
46
--
55
--
47
2.18.0
56
2.20.1
48
57
49
58
diff view generated by jsdifflib
1
Wire up the system control element's register banks
1
From: Richard Henderson <richard.henderson@linaro.org>
2
(sysctl and sysinfo).
3
2
4
This is the last of the previously completely unimplemented
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
components in the IoTKit.
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-16-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
8
target/arm/helper-a64.h | 7 ++
9
target/arm/helper.h | 2 +
10
target/arm/mte_helper.c | 194 +++++++++++++++++++++++++++++++++++++
11
target/arm/op_helper.c | 16 +++
12
target/arm/translate-a64.c | 172 +++++++++++++++++++++++++++++++-
13
5 files changed, 386 insertions(+), 5 deletions(-)
6
14
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Message-id: 20180820141116.9118-11-peter.maydell@linaro.org
11
---
12
include/hw/arm/iotkit.h | 6 +++++-
13
hw/arm/iotkit.c | 26 ++++++++++++++++++--------
14
2 files changed, 23 insertions(+), 9 deletions(-)
15
16
diff --git a/include/hw/arm/iotkit.h b/include/hw/arm/iotkit.h
17
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/arm/iotkit.h
17
--- a/target/arm/helper-a64.h
19
+++ b/include/hw/arm/iotkit.h
18
+++ b/target/arm/helper-a64.h
20
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
21
#include "hw/timer/cmsdk-apb-timer.h"
20
22
#include "hw/timer/cmsdk-apb-dualtimer.h"
21
DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
23
#include "hw/watchdog/cmsdk-apb-watchdog.h"
22
DEF_HELPER_FLAGS_4(addsubg, TCG_CALL_NO_RWG_SE, i64, env, i64, s32, i32)
24
-#include "hw/misc/unimp.h"
23
+DEF_HELPER_FLAGS_3(ldg, TCG_CALL_NO_WG, i64, env, i64, i64)
25
+#include "hw/misc/iotkit-sysctl.h"
24
+DEF_HELPER_FLAGS_3(stg, TCG_CALL_NO_WG, void, env, i64, i64)
26
+#include "hw/misc/iotkit-sysinfo.h"
25
+DEF_HELPER_FLAGS_3(stg_parallel, TCG_CALL_NO_WG, void, env, i64, i64)
27
#include "hw/or-irq.h"
26
+DEF_HELPER_FLAGS_2(stg_stub, TCG_CALL_NO_WG, void, env, i64)
28
#include "hw/core/split-irq.h"
27
+DEF_HELPER_FLAGS_3(st2g, TCG_CALL_NO_WG, void, env, i64, i64)
29
28
+DEF_HELPER_FLAGS_3(st2g_parallel, TCG_CALL_NO_WG, void, env, i64, i64)
30
@@ -XXX,XX +XXX,XX @@ typedef struct IoTKit {
29
+DEF_HELPER_FLAGS_2(st2g_stub, TCG_CALL_NO_WG, void, env, i64)
31
CMSDKAPBWatchdog nswatchdog;
30
diff --git a/target/arm/helper.h b/target/arm/helper.h
32
CMSDKAPBWatchdog swatchdog;
33
34
+ IoTKitSysCtl sysctl;
35
+ IoTKitSysCtl sysinfo;
36
+
37
MemoryRegion container;
38
MemoryRegion alias1;
39
MemoryRegion alias2;
40
diff --git a/hw/arm/iotkit.c b/hw/arm/iotkit.c
41
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
42
--- a/hw/arm/iotkit.c
32
--- a/target/arm/helper.h
43
+++ b/hw/arm/iotkit.c
33
+++ b/target/arm/helper.h
44
@@ -XXX,XX +XXX,XX @@
34
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_1(rebuild_hflags_a32_newel, TCG_CALL_NO_RWG, void, env)
45
#include "hw/sysbus.h"
35
DEF_HELPER_FLAGS_2(rebuild_hflags_a32, TCG_CALL_NO_RWG, void, env, int)
46
#include "hw/registerfields.h"
36
DEF_HELPER_FLAGS_2(rebuild_hflags_a64, TCG_CALL_NO_RWG, void, env, int)
47
#include "hw/arm/iotkit.h"
37
48
-#include "hw/misc/unimp.h"
38
+DEF_HELPER_FLAGS_5(probe_access, TCG_CALL_NO_WG, void, env, tl, i32, i32, i32)
49
#include "hw/arm/arm.h"
39
+
50
40
DEF_HELPER_1(vfp_get_fpscr, i32, env)
51
/* Clock frequency in HZ of the 32KHz "slow clock" */
41
DEF_HELPER_2(vfp_set_fpscr, void, env, i32)
52
@@ -XXX,XX +XXX,XX @@ static void iotkit_init(Object *obj)
42
53
sizeof(s->nswatchdog), TYPE_CMSDK_APB_WATCHDOG);
43
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
54
sysbus_init_child_obj(obj, "swatchdog", &s->swatchdog,
44
index XXXXXXX..XXXXXXX 100644
55
sizeof(s->swatchdog), TYPE_CMSDK_APB_WATCHDOG);
45
--- a/target/arm/mte_helper.c
56
+ sysbus_init_child_obj(obj, "iotkit-sysctl", &s->sysctl,
46
+++ b/target/arm/mte_helper.c
57
+ sizeof(s->sysctl), TYPE_IOTKIT_SYSCTL);
47
@@ -XXX,XX +XXX,XX @@ static int choose_nonexcluded_tag(int tag, int offset, uint16_t exclude)
58
+ sysbus_init_child_obj(obj, "iotkit-sysinfo", &s->sysinfo,
48
return tag;
59
+ sizeof(s->sysinfo), TYPE_IOTKIT_SYSINFO);
49
}
60
object_initialize_child(obj, "nmi-orgate", &s->nmi_orgate,
50
61
sizeof(s->nmi_orgate), TYPE_OR_IRQ,
51
+/**
62
&error_abort, NULL);
52
+ * allocation_tag_mem:
63
@@ -XXX,XX +XXX,XX @@ static void iotkit_realize(DeviceState *dev, Error **errp)
53
+ * @env: the cpu environment
64
qdev_get_gpio_in_named(dev_apb_ppc1,
54
+ * @ptr_mmu_idx: the addressing regime to use for the virtual address
65
"cfg_sec_resp", 0));
55
+ * @ptr: the virtual address for which to look up tag memory
66
56
+ * @ptr_access: the access to use for the virtual address
67
- /* Using create_unimplemented_device() maps the stub into the
57
+ * @ptr_size: the number of bytes in the normal memory access
68
- * system address space rather than into our container, but the
58
+ * @tag_access: the access to use for the tag memory
69
- * overall effect to the guest is the same.
59
+ * @tag_size: the number of bytes in the tag memory access
70
- */
60
+ * @ra: the return address for exception handling
71
- create_unimplemented_device("SYSINFO", 0x40020000, 0x1000);
61
+ *
72
-
62
+ * Our tag memory is formatted as a sequence of little-endian nibbles.
73
- create_unimplemented_device("SYSCONTROL", 0x50021000, 0x1000);
63
+ * That is, the byte at (addr >> (LOG2_TAG_GRANULE + 1)) contains two
74
+ object_property_set_bool(OBJECT(&s->sysinfo), true, "realized", &err);
64
+ * tags, with the tag at [3:0] for the lower addr and the tag at [7:4]
75
+ if (err) {
65
+ * for the higher addr.
76
+ error_propagate(errp, err);
66
+ *
67
+ * Here, resolve the physical address from the virtual address, and return
68
+ * a pointer to the corresponding tag byte. Exit with exception if the
69
+ * virtual address is not accessible for @ptr_access.
70
+ *
71
+ * The @ptr_size and @tag_size values may not have an obvious relation
72
+ * due to the alignment of @ptr, and the number of tag checks required.
73
+ *
74
+ * If there is no tag storage corresponding to @ptr, return NULL.
75
+ */
76
+static uint8_t *allocation_tag_mem(CPUARMState *env, int ptr_mmu_idx,
77
+ uint64_t ptr, MMUAccessType ptr_access,
78
+ int ptr_size, MMUAccessType tag_access,
79
+ int tag_size, uintptr_t ra)
80
+{
81
+ /* Tag storage not implemented. */
82
+ return NULL;
83
+}
84
+
85
uint64_t HELPER(irg)(CPUARMState *env, uint64_t rn, uint64_t rm)
86
{
87
int rtag;
88
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(addsubg)(CPUARMState *env, uint64_t ptr,
89
90
return address_with_allocation_tag(ptr + offset, rtag);
91
}
92
+
93
+static int load_tag1(uint64_t ptr, uint8_t *mem)
94
+{
95
+ int ofs = extract32(ptr, LOG2_TAG_GRANULE, 1) * 4;
96
+ return extract32(*mem, ofs, 4);
97
+}
98
+
99
+uint64_t HELPER(ldg)(CPUARMState *env, uint64_t ptr, uint64_t xt)
100
+{
101
+ int mmu_idx = cpu_mmu_index(env, false);
102
+ uint8_t *mem;
103
+ int rtag = 0;
104
+
105
+ /* Trap if accessing an invalid page. */
106
+ mem = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_LOAD, 1,
107
+ MMU_DATA_LOAD, 1, GETPC());
108
+
109
+ /* Load if page supports tags. */
110
+ if (mem) {
111
+ rtag = load_tag1(ptr, mem);
112
+ }
113
+
114
+ return address_with_allocation_tag(xt, rtag);
115
+}
116
+
117
+static void check_tag_aligned(CPUARMState *env, uint64_t ptr, uintptr_t ra)
118
+{
119
+ if (unlikely(!QEMU_IS_ALIGNED(ptr, TAG_GRANULE))) {
120
+ arm_cpu_do_unaligned_access(env_cpu(env), ptr, MMU_DATA_STORE,
121
+ cpu_mmu_index(env, false), ra);
122
+ g_assert_not_reached();
123
+ }
124
+}
125
+
126
+/* For use in a non-parallel context, store to the given nibble. */
127
+static void store_tag1(uint64_t ptr, uint8_t *mem, int tag)
128
+{
129
+ int ofs = extract32(ptr, LOG2_TAG_GRANULE, 1) * 4;
130
+ *mem = deposit32(*mem, ofs, 4, tag);
131
+}
132
+
133
+/* For use in a parallel context, atomically store to the given nibble. */
134
+static void store_tag1_parallel(uint64_t ptr, uint8_t *mem, int tag)
135
+{
136
+ int ofs = extract32(ptr, LOG2_TAG_GRANULE, 1) * 4;
137
+ uint8_t old = atomic_read(mem);
138
+
139
+ while (1) {
140
+ uint8_t new = deposit32(old, ofs, 4, tag);
141
+ uint8_t cmp = atomic_cmpxchg(mem, old, new);
142
+ if (likely(cmp == old)) {
143
+ return;
144
+ }
145
+ old = cmp;
146
+ }
147
+}
148
+
149
+typedef void stg_store1(uint64_t, uint8_t *, int);
150
+
151
+static inline void do_stg(CPUARMState *env, uint64_t ptr, uint64_t xt,
152
+ uintptr_t ra, stg_store1 store1)
153
+{
154
+ int mmu_idx = cpu_mmu_index(env, false);
155
+ uint8_t *mem;
156
+
157
+ check_tag_aligned(env, ptr, ra);
158
+
159
+ /* Trap if accessing an invalid page. */
160
+ mem = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_STORE, TAG_GRANULE,
161
+ MMU_DATA_STORE, 1, ra);
162
+
163
+ /* Store if page supports tags. */
164
+ if (mem) {
165
+ store1(ptr, mem, allocation_tag_from_addr(xt));
166
+ }
167
+}
168
+
169
+void HELPER(stg)(CPUARMState *env, uint64_t ptr, uint64_t xt)
170
+{
171
+ do_stg(env, ptr, xt, GETPC(), store_tag1);
172
+}
173
+
174
+void HELPER(stg_parallel)(CPUARMState *env, uint64_t ptr, uint64_t xt)
175
+{
176
+ do_stg(env, ptr, xt, GETPC(), store_tag1_parallel);
177
+}
178
+
179
+void HELPER(stg_stub)(CPUARMState *env, uint64_t ptr)
180
+{
181
+ int mmu_idx = cpu_mmu_index(env, false);
182
+ uintptr_t ra = GETPC();
183
+
184
+ check_tag_aligned(env, ptr, ra);
185
+ probe_write(env, ptr, TAG_GRANULE, mmu_idx, ra);
186
+}
187
+
188
+static inline void do_st2g(CPUARMState *env, uint64_t ptr, uint64_t xt,
189
+ uintptr_t ra, stg_store1 store1)
190
+{
191
+ int mmu_idx = cpu_mmu_index(env, false);
192
+ int tag = allocation_tag_from_addr(xt);
193
+ uint8_t *mem1, *mem2;
194
+
195
+ check_tag_aligned(env, ptr, ra);
196
+
197
+ /*
198
+ * Trap if accessing an invalid page(s).
199
+ * This takes priority over !allocation_tag_access_enabled.
200
+ */
201
+ if (ptr & TAG_GRANULE) {
202
+ /* Two stores unaligned mod TAG_GRANULE*2 -- modify two bytes. */
203
+ mem1 = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_STORE,
204
+ TAG_GRANULE, MMU_DATA_STORE, 1, ra);
205
+ mem2 = allocation_tag_mem(env, mmu_idx, ptr + TAG_GRANULE,
206
+ MMU_DATA_STORE, TAG_GRANULE,
207
+ MMU_DATA_STORE, 1, ra);
208
+
209
+ /* Store if page(s) support tags. */
210
+ if (mem1) {
211
+ store1(TAG_GRANULE, mem1, tag);
212
+ }
213
+ if (mem2) {
214
+ store1(0, mem2, tag);
215
+ }
216
+ } else {
217
+ /* Two stores aligned mod TAG_GRANULE*2 -- modify one byte. */
218
+ mem1 = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_STORE,
219
+ 2 * TAG_GRANULE, MMU_DATA_STORE, 1, ra);
220
+ if (mem1) {
221
+ tag |= tag << 4;
222
+ atomic_set(mem1, tag);
223
+ }
224
+ }
225
+}
226
+
227
+void HELPER(st2g)(CPUARMState *env, uint64_t ptr, uint64_t xt)
228
+{
229
+ do_st2g(env, ptr, xt, GETPC(), store_tag1);
230
+}
231
+
232
+void HELPER(st2g_parallel)(CPUARMState *env, uint64_t ptr, uint64_t xt)
233
+{
234
+ do_st2g(env, ptr, xt, GETPC(), store_tag1_parallel);
235
+}
236
+
237
+void HELPER(st2g_stub)(CPUARMState *env, uint64_t ptr)
238
+{
239
+ int mmu_idx = cpu_mmu_index(env, false);
240
+ uintptr_t ra = GETPC();
241
+ int in_page = -(ptr | TARGET_PAGE_MASK);
242
+
243
+ check_tag_aligned(env, ptr, ra);
244
+
245
+ if (likely(in_page >= 2 * TAG_GRANULE)) {
246
+ probe_write(env, ptr, 2 * TAG_GRANULE, mmu_idx, ra);
247
+ } else {
248
+ probe_write(env, ptr, TAG_GRANULE, mmu_idx, ra);
249
+ probe_write(env, ptr + TAG_GRANULE, TAG_GRANULE, mmu_idx, ra);
250
+ }
251
+}
252
diff --git a/target/arm/op_helper.c b/target/arm/op_helper.c
253
index XXXXXXX..XXXXXXX 100644
254
--- a/target/arm/op_helper.c
255
+++ b/target/arm/op_helper.c
256
@@ -XXX,XX +XXX,XX @@ uint32_t HELPER(ror_cc)(CPUARMState *env, uint32_t x, uint32_t i)
257
return ((uint32_t)x >> shift) | (x << (32 - shift));
258
}
259
}
260
+
261
+void HELPER(probe_access)(CPUARMState *env, target_ulong ptr,
262
+ uint32_t access_type, uint32_t mmu_idx,
263
+ uint32_t size)
264
+{
265
+ uint32_t in_page = -((uint32_t)ptr | TARGET_PAGE_SIZE);
266
+ uintptr_t ra = GETPC();
267
+
268
+ if (likely(size <= in_page)) {
269
+ probe_access(env, ptr, size, access_type, mmu_idx, ra);
270
+ } else {
271
+ probe_access(env, ptr, in_page, access_type, mmu_idx, ra);
272
+ probe_access(env, ptr + in_page, size - in_page,
273
+ access_type, mmu_idx, ra);
274
+ }
275
+}
276
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
277
index XXXXXXX..XXXXXXX 100644
278
--- a/target/arm/translate-a64.c
279
+++ b/target/arm/translate-a64.c
280
@@ -XXX,XX +XXX,XX @@ static void gen_address_with_allocation_tag0(TCGv_i64 dst, TCGv_i64 src)
281
tcg_gen_andi_i64(dst, src, ~MAKE_64BIT_MASK(56, 4));
282
}
283
284
+static void gen_probe_access(DisasContext *s, TCGv_i64 ptr,
285
+ MMUAccessType acc, int log2_size)
286
+{
287
+ TCGv_i32 t_acc = tcg_const_i32(acc);
288
+ TCGv_i32 t_idx = tcg_const_i32(get_mem_index(s));
289
+ TCGv_i32 t_size = tcg_const_i32(1 << log2_size);
290
+
291
+ gen_helper_probe_access(cpu_env, ptr, t_acc, t_idx, t_size);
292
+ tcg_temp_free_i32(t_acc);
293
+ tcg_temp_free_i32(t_idx);
294
+ tcg_temp_free_i32(t_size);
295
+}
296
+
297
typedef struct DisasCompare64 {
298
TCGCond cond;
299
TCGv_i64 value;
300
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
301
}
302
}
303
304
+/*
305
+ * Load/Store memory tags
306
+ *
307
+ * 31 30 29 24 22 21 12 10 5 0
308
+ * +-----+-------------+-----+---+------+-----+------+------+
309
+ * | 1 1 | 0 1 1 0 0 1 | op1 | 1 | imm9 | op2 | Rn | Rt |
310
+ * +-----+-------------+-----+---+------+-----+------+------+
311
+ */
312
+static void disas_ldst_tag(DisasContext *s, uint32_t insn)
313
+{
314
+ int rt = extract32(insn, 0, 5);
315
+ int rn = extract32(insn, 5, 5);
316
+ uint64_t offset = sextract64(insn, 12, 9) << LOG2_TAG_GRANULE;
317
+ int op2 = extract32(insn, 10, 2);
318
+ int op1 = extract32(insn, 22, 2);
319
+ bool is_load = false, is_pair = false, is_zero = false;
320
+ int index = 0;
321
+ TCGv_i64 addr, clean_addr, tcg_rt;
322
+
323
+ /* We checked insn bits [29:24,21] in the caller. */
324
+ if (extract32(insn, 30, 2) != 3) {
325
+ goto do_unallocated;
326
+ }
327
+
328
+ /*
329
+ * @index is a tri-state variable which has 3 states:
330
+ * < 0 : post-index, writeback
331
+ * = 0 : signed offset
332
+ * > 0 : pre-index, writeback
333
+ */
334
+ switch (op1) {
335
+ case 0:
336
+ if (op2 != 0) {
337
+ /* STG */
338
+ index = op2 - 2;
339
+ break;
340
+ }
341
+ goto do_unallocated;
342
+ case 1:
343
+ if (op2 != 0) {
344
+ /* STZG */
345
+ is_zero = true;
346
+ index = op2 - 2;
347
+ } else {
348
+ /* LDG */
349
+ is_load = true;
350
+ }
351
+ break;
352
+ case 2:
353
+ if (op2 != 0) {
354
+ /* ST2G */
355
+ is_pair = true;
356
+ index = op2 - 2;
357
+ break;
358
+ }
359
+ goto do_unallocated;
360
+ case 3:
361
+ if (op2 != 0) {
362
+ /* STZ2G */
363
+ is_pair = is_zero = true;
364
+ index = op2 - 2;
365
+ break;
366
+ }
367
+ goto do_unallocated;
368
+
369
+ default:
370
+ do_unallocated:
371
+ unallocated_encoding(s);
77
+ return;
372
+ return;
78
+ }
373
+ }
79
+ /* System information registers */
374
+
80
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->sysinfo), 0, 0x40020000);
375
+ if (!dc_isar_feature(aa64_mte_insn_reg, s)) {
81
+ /* System control registers */
376
+ goto do_unallocated;
82
+ object_property_set_bool(OBJECT(&s->sysctl), true, "realized", &err);
377
+ }
83
+ if (err) {
378
+
84
+ error_propagate(errp, err);
379
+ if (rn == 31) {
85
+ return;
380
+ gen_check_sp_alignment(s);
86
+ }
381
+ }
87
+ sysbus_mmio_map(SYS_BUS_DEVICE(&s->sysctl), 0, 0x50021000);
382
+
88
383
+ addr = read_cpu_reg_sp(s, rn, true);
89
/* This OR gate wires together outputs from the secure watchdogs to NMI */
384
+ if (index >= 0) {
90
object_property_set_int(OBJECT(&s->nmi_orgate), 2, "num-lines", &err);
385
+ /* pre-index or signed offset */
386
+ tcg_gen_addi_i64(addr, addr, offset);
387
+ }
388
+
389
+ if (is_load) {
390
+ tcg_gen_andi_i64(addr, addr, -TAG_GRANULE);
391
+ tcg_rt = cpu_reg(s, rt);
392
+ if (s->ata) {
393
+ gen_helper_ldg(tcg_rt, cpu_env, addr, tcg_rt);
394
+ } else {
395
+ clean_addr = clean_data_tbi(s, addr);
396
+ gen_probe_access(s, clean_addr, MMU_DATA_LOAD, MO_8);
397
+ gen_address_with_allocation_tag0(tcg_rt, addr);
398
+ }
399
+ } else {
400
+ tcg_rt = cpu_reg_sp(s, rt);
401
+ if (!s->ata) {
402
+ /*
403
+ * For STG and ST2G, we need to check alignment and probe memory.
404
+ * TODO: For STZG and STZ2G, we could rely on the stores below,
405
+ * at least for system mode; user-only won't enforce alignment.
406
+ */
407
+ if (is_pair) {
408
+ gen_helper_st2g_stub(cpu_env, addr);
409
+ } else {
410
+ gen_helper_stg_stub(cpu_env, addr);
411
+ }
412
+ } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
413
+ if (is_pair) {
414
+ gen_helper_st2g_parallel(cpu_env, addr, tcg_rt);
415
+ } else {
416
+ gen_helper_stg_parallel(cpu_env, addr, tcg_rt);
417
+ }
418
+ } else {
419
+ if (is_pair) {
420
+ gen_helper_st2g(cpu_env, addr, tcg_rt);
421
+ } else {
422
+ gen_helper_stg(cpu_env, addr, tcg_rt);
423
+ }
424
+ }
425
+ }
426
+
427
+ if (is_zero) {
428
+ TCGv_i64 clean_addr = clean_data_tbi(s, addr);
429
+ TCGv_i64 tcg_zero = tcg_const_i64(0);
430
+ int mem_index = get_mem_index(s);
431
+ int i, n = (1 + is_pair) << LOG2_TAG_GRANULE;
432
+
433
+ tcg_gen_qemu_st_i64(tcg_zero, clean_addr, mem_index,
434
+ MO_Q | MO_ALIGN_16);
435
+ for (i = 8; i < n; i += 8) {
436
+ tcg_gen_addi_i64(clean_addr, clean_addr, 8);
437
+ tcg_gen_qemu_st_i64(tcg_zero, clean_addr, mem_index, MO_Q);
438
+ }
439
+ tcg_temp_free_i64(tcg_zero);
440
+ }
441
+
442
+ if (index != 0) {
443
+ /* pre-index or post-index */
444
+ if (index < 0) {
445
+ /* post-index */
446
+ tcg_gen_addi_i64(addr, addr, offset);
447
+ }
448
+ tcg_gen_mov_i64(cpu_reg_sp(s, rn), addr);
449
+ }
450
+}
451
+
452
/* Loads and stores */
453
static void disas_ldst(DisasContext *s, uint32_t insn)
454
{
455
@@ -XXX,XX +XXX,XX @@ static void disas_ldst(DisasContext *s, uint32_t insn)
456
case 0x0d: /* AdvSIMD load/store single structure */
457
disas_ldst_single_struct(s, insn);
458
break;
459
- case 0x19: /* LDAPR/STLR (unscaled immediate) */
460
- if (extract32(insn, 10, 2) != 0 ||
461
- extract32(insn, 21, 1) != 0) {
462
+ case 0x19:
463
+ if (extract32(insn, 21, 1) != 0) {
464
+ disas_ldst_tag(s, insn);
465
+ } else if (extract32(insn, 10, 2) == 0) {
466
+ disas_ldst_ldapr_stlr(s, insn);
467
+ } else {
468
unallocated_encoding(s);
469
- break;
470
}
471
- disas_ldst_ldapr_stlr(s, insn);
472
break;
473
default:
474
unallocated_encoding(s);
91
--
475
--
92
2.18.0
476
2.20.1
93
477
94
478
diff view generated by jsdifflib
1
Connect the VIRQ and VFIQ lines from the GIC to the CPU;
1
From: Richard Henderson <richard.henderson@linaro.org>
2
these exist always for both CPU and GIC whether the
3
virtualization extensions are enabled or not, so we
4
can just unconditionally connect them.
5
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-17-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180821132811.17675-6-peter.maydell@linaro.org
9
---
7
---
10
hw/arm/fsl-imx6ul.c | 4 ++++
8
target/arm/translate-a64.c | 29 ++++++++++++++++++++++++++---
11
1 file changed, 4 insertions(+)
9
1 file changed, 26 insertions(+), 3 deletions(-)
12
10
13
diff --git a/hw/arm/fsl-imx6ul.c b/hw/arm/fsl-imx6ul.c
11
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
14
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/fsl-imx6ul.c
13
--- a/target/arm/translate-a64.c
16
+++ b/hw/arm/fsl-imx6ul.c
14
+++ b/target/arm/translate-a64.c
17
@@ -XXX,XX +XXX,XX @@ static void fsl_imx6ul_realize(DeviceState *dev, Error **errp)
15
@@ -XXX,XX +XXX,XX @@ static void disas_ld_lit(DisasContext *s, uint32_t insn)
18
irq = qdev_get_gpio_in(d, ARM_CPU_IRQ);
16
* +-----+-------+---+---+-------+---+-------+-------+------+------+
19
sysbus_connect_irq(sbd, i, irq);
17
*
20
sysbus_connect_irq(sbd, i + smp_cpus, qdev_get_gpio_in(d, ARM_CPU_FIQ));
18
* opc: LDP/STP/LDNP/STNP 00 -> 32 bit, 10 -> 64 bit
21
+ sysbus_connect_irq(sbd, i + 2 * smp_cpus,
19
- * LDPSW 01
22
+ qdev_get_gpio_in(d, ARM_CPU_VIRQ));
20
+ * LDPSW/STGP 01
23
+ sysbus_connect_irq(sbd, i + 3 * smp_cpus,
21
* LDP/STP/LDNP/STNP (SIMD) 00 -> 32 bit, 01 -> 64 bit, 10 -> 128 bit
24
+ qdev_get_gpio_in(d, ARM_CPU_VFIQ));
22
* V: 0 -> GPR, 1 -> Vector
23
* idx: 00 -> signed offset with non-temporal hint, 01 -> post-index,
24
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pair(DisasContext *s, uint32_t insn)
25
bool is_signed = false;
26
bool postindex = false;
27
bool wback = false;
28
+ bool set_tag = false;
29
30
TCGv_i64 clean_addr, dirty_addr;
31
32
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pair(DisasContext *s, uint32_t insn)
33
34
if (is_vector) {
35
size = 2 + opc;
36
+ } else if (opc == 1 && !is_load) {
37
+ /* STGP */
38
+ if (!dc_isar_feature(aa64_mte_insn_reg, s) || index == 0) {
39
+ unallocated_encoding(s);
40
+ return;
41
+ }
42
+ size = 3;
43
+ set_tag = true;
44
} else {
45
size = 2 + extract32(opc, 1, 1);
46
is_signed = extract32(opc, 0, 1);
47
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pair(DisasContext *s, uint32_t insn)
48
return;
25
}
49
}
26
50
27
/*
51
- offset <<= size;
52
+ offset <<= (set_tag ? LOG2_TAG_GRANULE : size);
53
54
if (rn == 31) {
55
gen_check_sp_alignment(s);
56
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pair(DisasContext *s, uint32_t insn)
57
if (!postindex) {
58
tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
59
}
60
- clean_addr = clean_data_tbi(s, dirty_addr);
61
62
+ if (set_tag) {
63
+ if (!s->ata) {
64
+ /*
65
+ * TODO: We could rely on the stores below, at least for
66
+ * system mode, if we arrange to add MO_ALIGN_16.
67
+ */
68
+ gen_helper_stg_stub(cpu_env, dirty_addr);
69
+ } else if (tb_cflags(s->base.tb) & CF_PARALLEL) {
70
+ gen_helper_stg_parallel(cpu_env, dirty_addr, dirty_addr);
71
+ } else {
72
+ gen_helper_stg(cpu_env, dirty_addr, dirty_addr);
73
+ }
74
+ }
75
+
76
+ clean_addr = clean_data_tbi(s, dirty_addr);
77
if (is_vector) {
78
if (is_load) {
79
do_fp_ld(s, rt, clean_addr, size);
28
--
80
--
29
2.18.0
81
2.20.1
30
82
31
83
diff view generated by jsdifflib
1
The kernel booting specification for an AArch32 kernel requires that
1
From: Richard Henderson <richard.henderson@linaro.org>
2
it is booted in Hyp mode if available; otherwise the kernel can't
3
enable KVM. We were incorrectly leaving the kernel in SVC mode.
4
If we're booting an AArch32 kernel in the Nonsecure state and Hyp
5
mode is available, start in it.
6
2
3
We can simplify our DC_ZVA if we recognize that the largest BS
4
that we actually use in system mode is 64. Let us just assert
5
that it fits within TARGET_PAGE_SIZE.
6
7
For DC_GVA and STZGM, we want to be able to write whole bytes
8
of tag memory, so assert that BS is >= 2 * TAG_GRANULE, or 32.
9
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
12
Message-id: 20200626033144.790098-18-richard.henderson@linaro.org
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
9
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
10
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
11
Message-id: 20180820153020.21478-7-peter.maydell@linaro.org
12
---
14
---
13
hw/arm/boot.c | 11 +++++++++++
15
target/arm/cpu.c | 24 ++++++++++++++++++++++++
14
1 file changed, 11 insertions(+)
16
1 file changed, 24 insertions(+)
15
17
16
diff --git a/hw/arm/boot.c b/hw/arm/boot.c
18
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
17
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/boot.c
20
--- a/target/arm/cpu.c
19
+++ b/hw/arm/boot.c
21
+++ b/target/arm/cpu.c
20
@@ -XXX,XX +XXX,XX @@ static void do_cpu_reset(void *opaque)
22
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
21
}
23
}
22
}
24
#endif
23
25
24
+ if (!env->aarch64 && !info->secure_boot &&
26
+ if (tcg_enabled()) {
25
+ arm_feature(env, ARM_FEATURE_EL2)) {
27
+ int dcz_blocklen = 4 << cpu->dcz_blocksize;
26
+ /*
27
+ * This is an AArch32 boot not to Secure state, and
28
+ * we have Hyp mode available, so boot the kernel into
29
+ * Hyp mode. This is not how the CPU comes out of reset,
30
+ * so we need to manually put it there.
31
+ */
32
+ cpsr_write(env, ARM_CPU_MODE_HYP, CPSR_M, CPSRWriteRaw);
33
+ }
34
+
28
+
35
if (cs == first_cpu) {
29
+ /*
36
AddressSpace *as = arm_boot_address_space(cpu, info);
30
+ * We only support DCZ blocklen that fits on one page.
31
+ *
32
+ * Architectually this is always true. However TARGET_PAGE_SIZE
33
+ * is variable and, for compatibility with -machine virt-2.7,
34
+ * is only 1KiB, as an artifact of legacy ARMv5 subpage support.
35
+ * But even then, while the largest architectural DCZ blocklen
36
+ * is 2KiB, no cpu actually uses such a large blocklen.
37
+ */
38
+ assert(dcz_blocklen <= TARGET_PAGE_SIZE);
39
+
40
+ /*
41
+ * We only support DCZ blocksize >= 2*TAG_GRANULE, which is to say
42
+ * both nibbles of each byte storing tag data may be written at once.
43
+ * Since TAG_GRANULE is 16, this means that blocklen must be >= 32.
44
+ */
45
+ if (cpu_isar_feature(aa64_mte, cpu)) {
46
+ assert(dcz_blocklen >= 2 * TAG_GRANULE);
47
+ }
48
+ }
49
+
50
qemu_init_vcpu(cs);
51
cpu_reset(cs);
37
52
38
--
53
--
39
2.18.0
54
2.20.1
40
55
41
56
diff view generated by jsdifflib
1
Reduce the size of the per-cpu GICH memory regions from 0x1000
1
From: Richard Henderson <richard.henderson@linaro.org>
2
to 0x200. The registers only cover 0x200 bytes, and the Cortex-A15
3
wants to map them at a spacing of 0x200 bytes apart. Having the
4
region be too large interferes with mapping them like that, so
5
reduce it.
6
2
3
Now that we know that the operation is on a single page,
4
we need not loop over pages while probing.
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-19-richard.henderson@linaro.org
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20180821132811.17675-3-peter.maydell@linaro.org
10
---
10
---
11
hw/intc/arm_gic.c | 2 +-
11
target/arm/helper-a64.c | 94 +++++++++++------------------------------
12
1 file changed, 1 insertion(+), 1 deletion(-)
12
1 file changed, 25 insertions(+), 69 deletions(-)
13
13
14
diff --git a/hw/intc/arm_gic.c b/hw/intc/arm_gic.c
14
diff --git a/target/arm/helper-a64.c b/target/arm/helper-a64.c
15
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/intc/arm_gic.c
16
--- a/target/arm/helper-a64.c
17
+++ b/hw/intc/arm_gic.c
17
+++ b/target/arm/helper-a64.c
18
@@ -XXX,XX +XXX,XX @@ static void arm_gic_realize(DeviceState *dev, Error **errp)
18
@@ -XXX,XX +XXX,XX @@ void HELPER(dc_zva)(CPUARMState *env, uint64_t vaddr_in)
19
for (i = 0; i < s->num_cpu; i++) {
19
* (which matches the usual QEMU behaviour of not implementing either
20
memory_region_init_io(&s->vifaceiomem[i + 1], OBJECT(s),
20
* alignment faults or any memory attribute handling).
21
&gic_viface_ops, &s->backref[i],
21
*/
22
- "gic_viface", 0x1000);
22
-
23
+ "gic_viface", 0x200);
23
- ARMCPU *cpu = env_archcpu(env);
24
sysbus_init_mmio(sbd, &s->vifaceiomem[i + 1]);
24
- uint64_t blocklen = 4 << cpu->dcz_blocksize;
25
+ int blocklen = 4 << env_archcpu(env)->dcz_blocksize;
26
uint64_t vaddr = vaddr_in & ~(blocklen - 1);
27
+ int mmu_idx = cpu_mmu_index(env, false);
28
+ void *mem;
29
+
30
+ /*
31
+ * Trapless lookup. In addition to actual invalid page, may
32
+ * return NULL for I/O, watchpoints, clean pages, etc.
33
+ */
34
+ mem = tlb_vaddr_to_host(env, vaddr, MMU_DATA_STORE, mmu_idx);
35
36
#ifndef CONFIG_USER_ONLY
37
- {
38
+ if (unlikely(!mem)) {
39
+ uintptr_t ra = GETPC();
40
+
41
/*
42
- * Slightly awkwardly, QEMU's TARGET_PAGE_SIZE may be less than
43
- * the block size so we might have to do more than one TLB lookup.
44
- * We know that in fact for any v8 CPU the page size is at least 4K
45
- * and the block size must be 2K or less, but TARGET_PAGE_SIZE is only
46
- * 1K as an artefact of legacy v5 subpage support being present in the
47
- * same QEMU executable. So in practice the hostaddr[] array has
48
- * two entries, given the current setting of TARGET_PAGE_BITS_MIN.
49
+ * Trap if accessing an invalid page. DC_ZVA requires that we supply
50
+ * the original pointer for an invalid page. But watchpoints require
51
+ * that we probe the actual space. So do both.
52
*/
53
- int maxidx = DIV_ROUND_UP(blocklen, TARGET_PAGE_SIZE);
54
- void *hostaddr[DIV_ROUND_UP(2 * KiB, 1 << TARGET_PAGE_BITS_MIN)];
55
- int try, i;
56
- unsigned mmu_idx = cpu_mmu_index(env, false);
57
- TCGMemOpIdx oi = make_memop_idx(MO_UB, mmu_idx);
58
+ (void) probe_write(env, vaddr_in, 1, mmu_idx, ra);
59
+ mem = probe_write(env, vaddr, blocklen, mmu_idx, ra);
60
61
- assert(maxidx <= ARRAY_SIZE(hostaddr));
62
-
63
- for (try = 0; try < 2; try++) {
64
-
65
- for (i = 0; i < maxidx; i++) {
66
- hostaddr[i] = tlb_vaddr_to_host(env,
67
- vaddr + TARGET_PAGE_SIZE * i,
68
- 1, mmu_idx);
69
- if (!hostaddr[i]) {
70
- break;
71
- }
72
- }
73
- if (i == maxidx) {
74
- /*
75
- * If it's all in the TLB it's fair game for just writing to;
76
- * we know we don't need to update dirty status, etc.
77
- */
78
- for (i = 0; i < maxidx - 1; i++) {
79
- memset(hostaddr[i], 0, TARGET_PAGE_SIZE);
80
- }
81
- memset(hostaddr[i], 0, blocklen - (i * TARGET_PAGE_SIZE));
82
- return;
83
- }
84
+ if (unlikely(!mem)) {
85
/*
86
- * OK, try a store and see if we can populate the tlb. This
87
- * might cause an exception if the memory isn't writable,
88
- * in which case we will longjmp out of here. We must for
89
- * this purpose use the actual register value passed to us
90
- * so that we get the fault address right.
91
+ * The only remaining reason for mem == NULL is I/O.
92
+ * Just do a series of byte writes as the architecture demands.
93
*/
94
- helper_ret_stb_mmu(env, vaddr_in, 0, oi, GETPC());
95
- /* Now we can populate the other TLB entries, if any */
96
- for (i = 0; i < maxidx; i++) {
97
- uint64_t va = vaddr + TARGET_PAGE_SIZE * i;
98
- if (va != (vaddr_in & TARGET_PAGE_MASK)) {
99
- helper_ret_stb_mmu(env, va, 0, oi, GETPC());
100
- }
101
+ for (int i = 0; i < blocklen; i++) {
102
+ cpu_stb_mmuidx_ra(env, vaddr + i, 0, mmu_idx, ra);
103
}
104
- }
105
-
106
- /*
107
- * Slow path (probably attempt to do this to an I/O device or
108
- * similar, or clearing of a block of code we have translations
109
- * cached for). Just do a series of byte writes as the architecture
110
- * demands. It's not worth trying to use a cpu_physical_memory_map(),
111
- * memset(), unmap() sequence here because:
112
- * + we'd need to account for the blocksize being larger than a page
113
- * + the direct-RAM access case is almost always going to be dealt
114
- * with in the fastpath code above, so there's no speed benefit
115
- * + we would have to deal with the map returning NULL because the
116
- * bounce buffer was in use
117
- */
118
- for (i = 0; i < blocklen; i++) {
119
- helper_ret_stb_mmu(env, vaddr + i, 0, oi, GETPC());
120
+ return;
25
}
121
}
26
}
122
}
123
-#else
124
- memset(g2h(vaddr), 0, blocklen);
125
#endif
126
+
127
+ memset(mem, 0, blocklen);
128
}
27
--
129
--
28
2.18.0
130
2.20.1
29
131
30
132
diff view generated by jsdifflib
1
Abstract out the calculation of the pitch and size of the
1
From: Richard Henderson <richard.henderson@linaro.org>
2
framebuffer into functions that operate on the BCM2835FBConfig
2
3
struct -- these are about to get a little more complicated
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
when we add support for virtual and physical sizes differing.
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
5
Message-id: 20200626033144.790098-20-richard.henderson@linaro.org
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>
8
Message-id: 20180814144436.679-6-peter.maydell@linaro.org
9
---
7
---
10
include/hw/display/bcm2835_fb.h | 22 ++++++++++++++++++++++
8
target/arm/helper-a64.h | 3 ++
11
hw/display/bcm2835_fb.c | 6 +++---
9
target/arm/translate.h | 2 +
12
hw/misc/bcm2835_property.c | 4 ++--
10
target/arm/mte_helper.c | 84 ++++++++++++++++++++++++++++++++++++++
13
3 files changed, 27 insertions(+), 5 deletions(-)
11
target/arm/translate-a64.c | 72 ++++++++++++++++++++++++++++----
14
12
4 files changed, 153 insertions(+), 8 deletions(-)
15
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
13
16
index XXXXXXX..XXXXXXX 100644
14
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
17
--- a/include/hw/display/bcm2835_fb.h
15
index XXXXXXX..XXXXXXX 100644
18
+++ b/include/hw/display/bcm2835_fb.h
16
--- a/target/arm/helper-a64.h
19
@@ -XXX,XX +XXX,XX @@ typedef struct {
17
+++ b/target/arm/helper-a64.h
20
18
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(stg_stub, TCG_CALL_NO_WG, void, env, i64)
21
void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig);
19
DEF_HELPER_FLAGS_3(st2g, TCG_CALL_NO_WG, void, env, i64, i64)
22
20
DEF_HELPER_FLAGS_3(st2g_parallel, TCG_CALL_NO_WG, void, env, i64, i64)
23
+/**
21
DEF_HELPER_FLAGS_2(st2g_stub, TCG_CALL_NO_WG, void, env, i64)
24
+ * bcm2835_fb_get_pitch: return number of bytes per line of the framebuffer
22
+DEF_HELPER_FLAGS_2(ldgm, TCG_CALL_NO_WG, i64, env, i64)
25
+ * @config: configuration info for the framebuffer
23
+DEF_HELPER_FLAGS_3(stgm, TCG_CALL_NO_WG, void, env, i64, i64)
26
+ *
24
+DEF_HELPER_FLAGS_3(stzgm_tags, TCG_CALL_NO_WG, void, env, i64, i64)
27
+ * Return the number of bytes per line of the framebuffer, ie the number
25
diff --git a/target/arm/translate.h b/target/arm/translate.h
28
+ * that must be added to a pixel address to get the address of the pixel
26
index XXXXXXX..XXXXXXX 100644
29
+ * directly below it on screen.
27
--- a/target/arm/translate.h
30
+ */
28
+++ b/target/arm/translate.h
31
+static inline uint32_t bcm2835_fb_get_pitch(BCM2835FBConfig *config)
29
@@ -XXX,XX +XXX,XX @@ typedef struct DisasContext {
30
* < 0, set by the current instruction.
31
*/
32
int8_t btype;
33
+ /* A copy of cpu->dcz_blocksize. */
34
+ uint8_t dcz_blocksize;
35
/* True if this page is guarded. */
36
bool guarded_page;
37
/* Bottom two bits of XScale c15_cpar coprocessor access control reg */
38
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
39
index XXXXXXX..XXXXXXX 100644
40
--- a/target/arm/mte_helper.c
41
+++ b/target/arm/mte_helper.c
42
@@ -XXX,XX +XXX,XX @@ void HELPER(st2g_stub)(CPUARMState *env, uint64_t ptr)
43
probe_write(env, ptr + TAG_GRANULE, TAG_GRANULE, mmu_idx, ra);
44
}
45
}
46
+
47
+#define LDGM_STGM_SIZE (4 << GMID_EL1_BS)
48
+
49
+uint64_t HELPER(ldgm)(CPUARMState *env, uint64_t ptr)
32
+{
50
+{
33
+ return config->xres * (config->bpp >> 3);
51
+ int mmu_idx = cpu_mmu_index(env, false);
52
+ uintptr_t ra = GETPC();
53
+ void *tag_mem;
54
+
55
+ ptr = QEMU_ALIGN_DOWN(ptr, LDGM_STGM_SIZE);
56
+
57
+ /* Trap if accessing an invalid page. */
58
+ tag_mem = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_LOAD,
59
+ LDGM_STGM_SIZE, MMU_DATA_LOAD,
60
+ LDGM_STGM_SIZE / (2 * TAG_GRANULE), ra);
61
+
62
+ /* The tag is squashed to zero if the page does not support tags. */
63
+ if (!tag_mem) {
64
+ return 0;
65
+ }
66
+
67
+ QEMU_BUILD_BUG_ON(GMID_EL1_BS != 6);
68
+ /*
69
+ * We are loading 64-bits worth of tags. The ordering of elements
70
+ * within the word corresponds to a 64-bit little-endian operation.
71
+ */
72
+ return ldq_le_p(tag_mem);
34
+}
73
+}
35
+
74
+
36
+/**
75
+void HELPER(stgm)(CPUARMState *env, uint64_t ptr, uint64_t val)
37
+ * bcm2835_fb_get_size: return total size of framebuffer in bytes
38
+ * @config: configuration info for the framebuffer
39
+ */
40
+static inline uint32_t bcm2835_fb_get_size(BCM2835FBConfig *config)
41
+{
76
+{
42
+ return config->yres * bcm2835_fb_get_pitch(config);
77
+ int mmu_idx = cpu_mmu_index(env, false);
78
+ uintptr_t ra = GETPC();
79
+ void *tag_mem;
80
+
81
+ ptr = QEMU_ALIGN_DOWN(ptr, LDGM_STGM_SIZE);
82
+
83
+ /* Trap if accessing an invalid page. */
84
+ tag_mem = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_STORE,
85
+ LDGM_STGM_SIZE, MMU_DATA_LOAD,
86
+ LDGM_STGM_SIZE / (2 * TAG_GRANULE), ra);
87
+
88
+ /*
89
+ * Tag store only happens if the page support tags,
90
+ * and if the OS has enabled access to the tags.
91
+ */
92
+ if (!tag_mem) {
93
+ return;
94
+ }
95
+
96
+ QEMU_BUILD_BUG_ON(GMID_EL1_BS != 6);
97
+ /*
98
+ * We are storing 64-bits worth of tags. The ordering of elements
99
+ * within the word corresponds to a 64-bit little-endian operation.
100
+ */
101
+ stq_le_p(tag_mem, val);
43
+}
102
+}
44
+
103
+
45
#endif
104
+void HELPER(stzgm_tags)(CPUARMState *env, uint64_t ptr, uint64_t val)
46
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
105
+{
47
index XXXXXXX..XXXXXXX 100644
106
+ uintptr_t ra = GETPC();
48
--- a/hw/display/bcm2835_fb.c
107
+ int mmu_idx = cpu_mmu_index(env, false);
49
+++ b/hw/display/bcm2835_fb.c
108
+ int log2_dcz_bytes, log2_tag_bytes;
50
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
109
+ intptr_t dcz_bytes, tag_bytes;
110
+ uint8_t *mem;
111
+
112
+ /*
113
+ * In arm_cpu_realizefn, we assert that dcz > LOG2_TAG_GRANULE+1,
114
+ * i.e. 32 bytes, which is an unreasonably small dcz anyway,
115
+ * to make sure that we can access one complete tag byte here.
116
+ */
117
+ log2_dcz_bytes = env_archcpu(env)->dcz_blocksize + 2;
118
+ log2_tag_bytes = log2_dcz_bytes - (LOG2_TAG_GRANULE + 1);
119
+ dcz_bytes = (intptr_t)1 << log2_dcz_bytes;
120
+ tag_bytes = (intptr_t)1 << log2_tag_bytes;
121
+ ptr &= -dcz_bytes;
122
+
123
+ mem = allocation_tag_mem(env, mmu_idx, ptr, MMU_DATA_STORE, dcz_bytes,
124
+ MMU_DATA_STORE, tag_bytes, ra);
125
+ if (mem) {
126
+ int tag_pair = (val & 0xf) * 0x11;
127
+ memset(mem, tag_pair, tag_bytes);
128
+ }
129
+}
130
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
131
index XXXXXXX..XXXXXXX 100644
132
--- a/target/arm/translate-a64.c
133
+++ b/target/arm/translate-a64.c
134
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
135
uint64_t offset = sextract64(insn, 12, 9) << LOG2_TAG_GRANULE;
136
int op2 = extract32(insn, 10, 2);
137
int op1 = extract32(insn, 22, 2);
138
- bool is_load = false, is_pair = false, is_zero = false;
139
+ bool is_load = false, is_pair = false, is_zero = false, is_mult = false;
140
int index = 0;
141
TCGv_i64 addr, clean_addr, tcg_rt;
142
143
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
144
if (op2 != 0) {
145
/* STG */
146
index = op2 - 2;
147
- break;
148
+ } else {
149
+ /* STZGM */
150
+ if (s->current_el == 0 || offset != 0) {
151
+ goto do_unallocated;
152
+ }
153
+ is_mult = is_zero = true;
154
}
155
- goto do_unallocated;
156
+ break;
157
case 1:
158
if (op2 != 0) {
159
/* STZG */
160
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
161
/* ST2G */
162
is_pair = true;
163
index = op2 - 2;
164
- break;
165
+ } else {
166
+ /* STGM */
167
+ if (s->current_el == 0 || offset != 0) {
168
+ goto do_unallocated;
169
+ }
170
+ is_mult = true;
171
}
172
- goto do_unallocated;
173
+ break;
174
case 3:
175
if (op2 != 0) {
176
/* STZ2G */
177
is_pair = is_zero = true;
178
index = op2 - 2;
179
- break;
180
+ } else {
181
+ /* LDGM */
182
+ if (s->current_el == 0 || offset != 0) {
183
+ goto do_unallocated;
184
+ }
185
+ is_mult = is_load = true;
186
}
187
- goto do_unallocated;
188
+ break;
189
190
default:
191
do_unallocated:
192
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
51
return;
193
return;
52
}
194
}
53
195
54
- src_width = s->config.xres * (s->config.bpp >> 3);
196
- if (!dc_isar_feature(aa64_mte_insn_reg, s)) {
55
+ src_width = bcm2835_fb_get_pitch(&s->config);
197
+ if (is_mult
56
dest_width = s->config.xres;
198
+ ? !dc_isar_feature(aa64_mte, s)
57
199
+ : !dc_isar_feature(aa64_mte_insn_reg, s)) {
58
switch (surface_bits_per_pixel(surface)) {
200
goto do_unallocated;
59
@@ -XXX,XX +XXX,XX @@ static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
201
}
60
202
61
/* TODO - Manage properly virtual resolution */
203
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_tag(DisasContext *s, uint32_t insn)
62
204
tcg_gen_addi_i64(addr, addr, offset);
63
- pitch = s->config.xres * (s->config.bpp >> 3);
205
}
64
- size = s->config.yres * pitch;
206
65
+ pitch = bcm2835_fb_get_pitch(&s->config);
207
+ if (is_mult) {
66
+ size = bcm2835_fb_get_size(&s->config);
208
+ tcg_rt = cpu_reg(s, rt);
67
209
+
68
stl_le_phys(&s->dma_as, value + 16, pitch);
210
+ if (is_zero) {
69
stl_le_phys(&s->dma_as, value + 32, s->config.base);
211
+ int size = 4 << s->dcz_blocksize;
70
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
212
+
71
index XXXXXXX..XXXXXXX 100644
213
+ if (s->ata) {
72
--- a/hw/misc/bcm2835_property.c
214
+ gen_helper_stzgm_tags(cpu_env, addr, tcg_rt);
73
+++ b/hw/misc/bcm2835_property.c
215
+ }
74
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
216
+ /*
75
case 0x00040001: /* Allocate buffer */
217
+ * The non-tags portion of STZGM is mostly like DC_ZVA,
76
stl_le_phys(&s->dma_as, value + 12, fbconfig.base);
218
+ * except the alignment happens before the access.
77
stl_le_phys(&s->dma_as, value + 16,
219
+ */
78
- fbconfig.xres * fbconfig.yres * fbconfig.bpp / 8);
220
+ clean_addr = clean_data_tbi(s, addr);
79
+ bcm2835_fb_get_size(&fbconfig));
221
+ tcg_gen_andi_i64(clean_addr, clean_addr, -size);
80
resplen = 8;
222
+ gen_helper_dc_zva(cpu_env, clean_addr);
81
break;
223
+ } else if (s->ata) {
82
case 0x00048001: /* Release buffer */
224
+ if (is_load) {
83
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
225
+ gen_helper_ldgm(tcg_rt, cpu_env, addr);
84
break;
226
+ } else {
85
case 0x00040008: /* Get pitch */
227
+ gen_helper_stgm(cpu_env, addr, tcg_rt);
86
stl_le_phys(&s->dma_as, value + 12,
228
+ }
87
- fbconfig.xres * fbconfig.bpp / 8);
229
+ } else {
88
+ bcm2835_fb_get_pitch(&fbconfig));
230
+ MMUAccessType acc = is_load ? MMU_DATA_LOAD : MMU_DATA_STORE;
89
resplen = 4;
231
+ int size = 4 << GMID_EL1_BS;
90
break;
232
+
91
case 0x00040009: /* Get virtual offset */
233
+ clean_addr = clean_data_tbi(s, addr);
234
+ tcg_gen_andi_i64(clean_addr, clean_addr, -size);
235
+ gen_probe_access(s, clean_addr, acc, size);
236
+
237
+ if (is_load) {
238
+ /* The result tags are zeros. */
239
+ tcg_gen_movi_i64(tcg_rt, 0);
240
+ }
241
+ }
242
+ return;
243
+ }
244
+
245
if (is_load) {
246
tcg_gen_andi_i64(addr, addr, -TAG_GRANULE);
247
tcg_rt = cpu_reg(s, rt);
248
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_init_disas_context(DisasContextBase *dcbase,
249
dc->vec_stride = 0;
250
dc->cp_regs = arm_cpu->cp_regs;
251
dc->features = env->features;
252
+ dc->dcz_blocksize = arm_cpu->dcz_blocksize;
253
254
/* Single step state. The code-generation logic here is:
255
* SS_ACTIVE == 0:
92
--
256
--
93
2.18.0
257
2.20.1
94
258
95
259
diff view generated by jsdifflib
1
The AArch32 HCR and HCR2 registers alias HCR_EL2
1
From: Richard Henderson <richard.henderson@linaro.org>
2
bits [31:0] and [63:32]; implement them.
3
2
4
Since HCR2 exists in ARMv8 but not ARMv7, we need new
3
Like the regular data cache flushes, these are nops within qemu.
5
regdef arrays for "we have EL3, not EL2, we're ARMv8"
6
and "we have EL2, we're ARMv8" to hold the definitions.
7
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-21-richard.henderson@linaro.org
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
11
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
12
Message-id: 20180820153020.21478-3-peter.maydell@linaro.org
13
---
9
---
14
target/arm/helper.c | 54 +++++++++++++++++++++++++++++++++++++++++----
10
target/arm/helper.c | 65 +++++++++++++++++++++++++++++++++++++++++++++
15
1 file changed, 50 insertions(+), 4 deletions(-)
11
1 file changed, 65 insertions(+)
16
12
17
diff --git a/target/arm/helper.c b/target/arm/helper.c
13
diff --git a/target/arm/helper.c b/target/arm/helper.c
18
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
19
--- a/target/arm/helper.c
15
--- a/target/arm/helper.c
20
+++ b/target/arm/helper.c
16
+++ b/target/arm/helper.c
21
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo el3_no_el2_cp_reginfo[] = {
17
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo mte_reginfo[] = {
22
.opc0 = 3, .opc1 = 4, .crn = 12, .crm = 0, .opc2 = 0,
18
.opc0 = 3, .opc1 = 3, .crn = 4, .crm = 2, .opc2 = 7,
23
.access = PL2_RW,
24
.readfn = arm_cp_read_zero, .writefn = arm_cp_write_ignore },
25
- { .name = "HCR_EL2", .state = ARM_CP_STATE_AA64,
26
+ { .name = "HCR_EL2", .state = ARM_CP_STATE_BOTH,
27
.type = ARM_CP_NO_RAW,
19
.type = ARM_CP_NO_RAW,
28
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 0,
20
.access = PL0_RW, .readfn = tco_read, .writefn = tco_write },
29
.access = PL2_RW,
21
+ { .name = "DC_IGVAC", .state = ARM_CP_STATE_AA64,
30
- .readfn = arm_cp_read_zero, .writefn = arm_cp_write_ignore },
22
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 6, .opc2 = 3,
31
+ .type = ARM_CP_CONST, .resetvalue = 0 },
23
+ .type = ARM_CP_NOP, .access = PL1_W,
32
{ .name = "ESR_EL2", .state = ARM_CP_STATE_BOTH,
24
+ .accessfn = aa64_cacheop_poc_access },
33
.opc0 = 3, .opc1 = 4, .crn = 5, .crm = 2, .opc2 = 0,
25
+ { .name = "DC_IGSW", .state = ARM_CP_STATE_AA64,
34
.access = PL2_RW,
26
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 6, .opc2 = 4,
35
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo el3_no_el2_cp_reginfo[] = {
27
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
28
+ { .name = "DC_IGDVAC", .state = ARM_CP_STATE_AA64,
29
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 6, .opc2 = 5,
30
+ .type = ARM_CP_NOP, .access = PL1_W,
31
+ .accessfn = aa64_cacheop_poc_access },
32
+ { .name = "DC_IGDSW", .state = ARM_CP_STATE_AA64,
33
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 6, .opc2 = 6,
34
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
35
+ { .name = "DC_CGSW", .state = ARM_CP_STATE_AA64,
36
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 10, .opc2 = 4,
37
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
38
+ { .name = "DC_CGDSW", .state = ARM_CP_STATE_AA64,
39
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 10, .opc2 = 6,
40
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
41
+ { .name = "DC_CIGSW", .state = ARM_CP_STATE_AA64,
42
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 14, .opc2 = 4,
43
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
44
+ { .name = "DC_CIGDSW", .state = ARM_CP_STATE_AA64,
45
+ .opc0 = 1, .opc1 = 0, .crn = 7, .crm = 14, .opc2 = 6,
46
+ .type = ARM_CP_NOP, .access = PL1_W, .accessfn = access_tsw },
36
REGINFO_SENTINEL
47
REGINFO_SENTINEL
37
};
48
};
38
49
39
+/* Ditto, but for registers which exist in ARMv8 but not v7 */
50
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo mte_tco_ro_reginfo[] = {
40
+static const ARMCPRegInfo el3_no_el2_v8_cp_reginfo[] = {
51
.type = ARM_CP_CONST, .access = PL0_RW, },
41
+ { .name = "HCR2", .state = ARM_CP_STATE_AA32,
52
REGINFO_SENTINEL
42
+ .cp = 15, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 4,
53
};
43
+ .access = PL2_RW,
54
+
44
+ .type = ARM_CP_CONST, .resetvalue = 0 },
55
+static const ARMCPRegInfo mte_el0_cacheop_reginfo[] = {
56
+ { .name = "DC_CGVAC", .state = ARM_CP_STATE_AA64,
57
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 10, .opc2 = 3,
58
+ .type = ARM_CP_NOP, .access = PL0_W,
59
+ .accessfn = aa64_cacheop_poc_access },
60
+ { .name = "DC_CGDVAC", .state = ARM_CP_STATE_AA64,
61
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 10, .opc2 = 5,
62
+ .type = ARM_CP_NOP, .access = PL0_W,
63
+ .accessfn = aa64_cacheop_poc_access },
64
+ { .name = "DC_CGVAP", .state = ARM_CP_STATE_AA64,
65
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 12, .opc2 = 3,
66
+ .type = ARM_CP_NOP, .access = PL0_W,
67
+ .accessfn = aa64_cacheop_poc_access },
68
+ { .name = "DC_CGDVAP", .state = ARM_CP_STATE_AA64,
69
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 12, .opc2 = 5,
70
+ .type = ARM_CP_NOP, .access = PL0_W,
71
+ .accessfn = aa64_cacheop_poc_access },
72
+ { .name = "DC_CGVADP", .state = ARM_CP_STATE_AA64,
73
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 13, .opc2 = 3,
74
+ .type = ARM_CP_NOP, .access = PL0_W,
75
+ .accessfn = aa64_cacheop_poc_access },
76
+ { .name = "DC_CGDVADP", .state = ARM_CP_STATE_AA64,
77
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 13, .opc2 = 5,
78
+ .type = ARM_CP_NOP, .access = PL0_W,
79
+ .accessfn = aa64_cacheop_poc_access },
80
+ { .name = "DC_CIGVAC", .state = ARM_CP_STATE_AA64,
81
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 14, .opc2 = 3,
82
+ .type = ARM_CP_NOP, .access = PL0_W,
83
+ .accessfn = aa64_cacheop_poc_access },
84
+ { .name = "DC_CIGDVAC", .state = ARM_CP_STATE_AA64,
85
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 14, .opc2 = 5,
86
+ .type = ARM_CP_NOP, .access = PL0_W,
87
+ .accessfn = aa64_cacheop_poc_access },
45
+ REGINFO_SENTINEL
88
+ REGINFO_SENTINEL
46
+};
89
+};
47
+
90
+
48
static void hcr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
91
#endif
49
{
92
50
ARMCPU *cpu = arm_env_get_cpu(env);
93
static CPAccessResult access_predinv(CPUARMState *env, const ARMCPRegInfo *ri,
51
@@ -XXX,XX +XXX,XX @@ static void hcr_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
94
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
52
* HCR_PTW forbids certain page-table setups
53
* HCR_DC Disables stage1 and enables stage2 translation
54
*/
95
*/
55
- if ((raw_read(env, ri) ^ value) & (HCR_VM | HCR_PTW | HCR_DC)) {
96
if (cpu_isar_feature(aa64_mte, cpu)) {
56
+ if ((env->cp15.hcr_el2 ^ value) & (HCR_VM | HCR_PTW | HCR_DC)) {
97
define_arm_cp_regs(cpu, mte_reginfo);
57
tlb_flush(CPU(cpu));
98
+ define_arm_cp_regs(cpu, mte_el0_cacheop_reginfo);
99
} else if (cpu_isar_feature(aa64_mte_insn_reg, cpu)) {
100
define_arm_cp_regs(cpu, mte_tco_ro_reginfo);
101
+ define_arm_cp_regs(cpu, mte_el0_cacheop_reginfo);
58
}
102
}
59
- raw_write(env, ri, value);
103
#endif
60
+ env->cp15.hcr_el2 = value;
104
61
+}
62
+
63
+static void hcr_writehigh(CPUARMState *env, const ARMCPRegInfo *ri,
64
+ uint64_t value)
65
+{
66
+ /* Handle HCR2 write, i.e. write to high half of HCR_EL2 */
67
+ value = deposit64(env->cp15.hcr_el2, 32, 32, value);
68
+ hcr_write(env, NULL, value);
69
+}
70
+
71
+static void hcr_writelow(CPUARMState *env, const ARMCPRegInfo *ri,
72
+ uint64_t value)
73
+{
74
+ /* Handle HCR write, i.e. write to low half of HCR_EL2 */
75
+ value = deposit64(env->cp15.hcr_el2, 0, 32, value);
76
+ hcr_write(env, NULL, value);
77
}
78
79
static const ARMCPRegInfo el2_cp_reginfo[] = {
80
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
81
.opc0 = 3, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 0,
82
.access = PL2_RW, .fieldoffset = offsetof(CPUARMState, cp15.hcr_el2),
83
.writefn = hcr_write },
84
+ { .name = "HCR", .state = ARM_CP_STATE_AA32,
85
+ .type = ARM_CP_ALIAS,
86
+ .cp = 15, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 0,
87
+ .access = PL2_RW, .fieldoffset = offsetof(CPUARMState, cp15.hcr_el2),
88
+ .writefn = hcr_writelow },
89
{ .name = "ELR_EL2", .state = ARM_CP_STATE_AA64,
90
.type = ARM_CP_ALIAS,
91
.opc0 = 3, .opc1 = 4, .crn = 4, .crm = 0, .opc2 = 1,
92
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo el2_cp_reginfo[] = {
93
REGINFO_SENTINEL
94
};
95
96
+static const ARMCPRegInfo el2_v8_cp_reginfo[] = {
97
+ { .name = "HCR2", .state = ARM_CP_STATE_AA32,
98
+ .type = ARM_CP_ALIAS,
99
+ .cp = 15, .opc1 = 4, .crn = 1, .crm = 1, .opc2 = 4,
100
+ .access = PL2_RW,
101
+ .fieldoffset = offsetofhigh32(CPUARMState, cp15.hcr_el2),
102
+ .writefn = hcr_writehigh },
103
+ REGINFO_SENTINEL
104
+};
105
+
106
static CPAccessResult nsacr_access(CPUARMState *env, const ARMCPRegInfo *ri,
107
bool isread)
108
{
109
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
110
};
111
define_arm_cp_regs(cpu, vpidr_regs);
112
define_arm_cp_regs(cpu, el2_cp_reginfo);
113
+ if (arm_feature(env, ARM_FEATURE_V8)) {
114
+ define_arm_cp_regs(cpu, el2_v8_cp_reginfo);
115
+ }
116
/* RVBAR_EL2 is only implemented if EL2 is the highest EL */
117
if (!arm_feature(env, ARM_FEATURE_EL3)) {
118
ARMCPRegInfo rvbar = {
119
@@ -XXX,XX +XXX,XX @@ void register_cp_regs_for_features(ARMCPU *cpu)
120
};
121
define_arm_cp_regs(cpu, vpidr_regs);
122
define_arm_cp_regs(cpu, el3_no_el2_cp_reginfo);
123
+ if (arm_feature(env, ARM_FEATURE_V8)) {
124
+ define_arm_cp_regs(cpu, el3_no_el2_v8_cp_reginfo);
125
+ }
126
}
127
}
128
if (arm_feature(env, ARM_FEATURE_EL3)) {
129
--
105
--
130
2.18.0
106
2.20.1
131
107
132
108
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Richard Henderson <richard.henderson@linaro.org>
2
2
3
We will shortly need this in mte_helper.c as well.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
4
Message-id: 20180814002653.12828-5-richard.henderson@linaro.org
7
Message-id: 20200626033144.790098-22-richard.henderson@linaro.org
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
9
---
8
target/arm/helper.c | 101 ++++++++++++++++++++++----------------------
10
target/arm/internals.h | 36 ++++++++++++++++++++++++++++++++++++
9
1 file changed, 51 insertions(+), 50 deletions(-)
11
target/arm/helper.c | 36 ------------------------------------
12
2 files changed, 36 insertions(+), 36 deletions(-)
10
13
14
diff --git a/target/arm/internals.h b/target/arm/internals.h
15
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/internals.h
17
+++ b/target/arm/internals.h
18
@@ -XXX,XX +XXX,XX @@ static inline bool regime_is_pan(CPUARMState *env, ARMMMUIdx mmu_idx)
19
}
20
}
21
22
+/* Return the exception level which controls this address translation regime */
23
+static inline uint32_t regime_el(CPUARMState *env, ARMMMUIdx mmu_idx)
24
+{
25
+ switch (mmu_idx) {
26
+ case ARMMMUIdx_E20_0:
27
+ case ARMMMUIdx_E20_2:
28
+ case ARMMMUIdx_E20_2_PAN:
29
+ case ARMMMUIdx_Stage2:
30
+ case ARMMMUIdx_E2:
31
+ return 2;
32
+ case ARMMMUIdx_SE3:
33
+ return 3;
34
+ case ARMMMUIdx_SE10_0:
35
+ return arm_el_is_aa64(env, 3) ? 1 : 3;
36
+ case ARMMMUIdx_SE10_1:
37
+ case ARMMMUIdx_SE10_1_PAN:
38
+ case ARMMMUIdx_Stage1_E0:
39
+ case ARMMMUIdx_Stage1_E1:
40
+ case ARMMMUIdx_Stage1_E1_PAN:
41
+ case ARMMMUIdx_E10_0:
42
+ case ARMMMUIdx_E10_1:
43
+ case ARMMMUIdx_E10_1_PAN:
44
+ case ARMMMUIdx_MPrivNegPri:
45
+ case ARMMMUIdx_MUserNegPri:
46
+ case ARMMMUIdx_MPriv:
47
+ case ARMMMUIdx_MUser:
48
+ case ARMMMUIdx_MSPrivNegPri:
49
+ case ARMMMUIdx_MSUserNegPri:
50
+ case ARMMMUIdx_MSPriv:
51
+ case ARMMMUIdx_MSUser:
52
+ return 1;
53
+ default:
54
+ g_assert_not_reached();
55
+ }
56
+}
57
+
58
/* Return the FSR value for a debug exception (watchpoint, hardware
59
* breakpoint or BKPT insn) targeting the specified exception level.
60
*/
11
diff --git a/target/arm/helper.c b/target/arm/helper.c
61
diff --git a/target/arm/helper.c b/target/arm/helper.c
12
index XXXXXXX..XXXXXXX 100644
62
index XXXXXXX..XXXXXXX 100644
13
--- a/target/arm/helper.c
63
--- a/target/arm/helper.c
14
+++ b/target/arm/helper.c
64
+++ b/target/arm/helper.c
15
@@ -XXX,XX +XXX,XX @@ float##fsz HELPER(vfp_##name##to##p)(uint##isz##_t x, uint32_t shift, \
65
@@ -XXX,XX +XXX,XX @@ void arm_cpu_do_interrupt(CPUState *cs)
16
void *fpstp) \
17
{ return itype##_to_##float##fsz##_scalbn(x, -shift, fpstp); }
18
19
-/* Notice that we want only input-denormal exception flags from the
20
- * scalbn operation: the other possible flags (overflow+inexact if
21
- * we overflow to infinity, output-denormal) aren't correct for the
22
- * complete scale-and-convert operation.
23
- */
24
-#define VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, round) \
25
-uint##isz##_t HELPER(vfp_to##name##p##round)(float##fsz x, \
26
- uint32_t shift, \
27
- void *fpstp) \
28
-{ \
29
- float_status *fpst = fpstp; \
30
- int old_exc_flags = get_float_exception_flags(fpst); \
31
- float##fsz tmp; \
32
- if (float##fsz##_is_any_nan(x)) { \
33
- float_raise(float_flag_invalid, fpst); \
34
- return 0; \
35
- } \
36
- tmp = float##fsz##_scalbn(x, shift, fpst); \
37
- old_exc_flags |= get_float_exception_flags(fpst) \
38
- & float_flag_input_denormal; \
39
- set_float_exception_flags(old_exc_flags, fpst); \
40
- return float##fsz##_to_##itype##round(tmp, fpst); \
41
+#define VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, ROUND, suff) \
42
+uint##isz##_t HELPER(vfp_to##name##p##suff)(float##fsz x, uint32_t shift, \
43
+ void *fpst) \
44
+{ \
45
+ if (unlikely(float##fsz##_is_any_nan(x))) { \
46
+ float_raise(float_flag_invalid, fpst); \
47
+ return 0; \
48
+ } \
49
+ return float##fsz##_to_##itype##_scalbn(x, ROUND, shift, fpst); \
50
}
66
}
51
67
#endif /* !CONFIG_USER_ONLY */
52
#define VFP_CONV_FIX(name, p, fsz, isz, itype) \
68
53
VFP_CONV_FIX_FLOAT(name, p, fsz, isz, itype) \
69
-/* Return the exception level which controls this address translation regime */
54
-VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, _round_to_zero) \
70
-static uint32_t regime_el(CPUARMState *env, ARMMMUIdx mmu_idx)
55
-VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, )
56
+VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, \
57
+ float_round_to_zero, _round_to_zero) \
58
+VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, \
59
+ get_float_rounding_mode(fpst), )
60
61
#define VFP_CONV_FIX_A64(name, p, fsz, isz, itype) \
62
VFP_CONV_FIX_FLOAT(name, p, fsz, isz, itype) \
63
-VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, )
64
+VFP_CONV_FLOAT_FIX_ROUND(name, p, fsz, isz, itype, \
65
+ get_float_rounding_mode(fpst), )
66
67
VFP_CONV_FIX(sh, d, 64, 64, int16)
68
VFP_CONV_FIX(sl, d, 64, 64, int32)
69
@@ -XXX,XX +XXX,XX @@ uint32_t HELPER(vfp_uqtoh)(uint64_t x, uint32_t shift, void *fpst)
70
return uint64_to_float16_scalbn(x, -shift, fpst);
71
}
72
73
-static float64 do_prescale_fp16(float16 f, int shift, float_status *fpst)
74
-{
71
-{
75
- if (unlikely(float16_is_any_nan(f))) {
72
- switch (mmu_idx) {
76
- float_raise(float_flag_invalid, fpst);
73
- case ARMMMUIdx_E20_0:
77
- return 0;
74
- case ARMMMUIdx_E20_2:
78
- } else {
75
- case ARMMMUIdx_E20_2_PAN:
79
- int old_exc_flags = get_float_exception_flags(fpst);
76
- case ARMMMUIdx_Stage2:
80
- float64 ret;
77
- case ARMMMUIdx_E2:
81
-
78
- return 2;
82
- ret = float16_to_float64(f, true, fpst);
79
- case ARMMMUIdx_SE3:
83
- ret = float64_scalbn(ret, shift, fpst);
80
- return 3;
84
- old_exc_flags |= get_float_exception_flags(fpst)
81
- case ARMMMUIdx_SE10_0:
85
- & float_flag_input_denormal;
82
- return arm_el_is_aa64(env, 3) ? 1 : 3;
86
- set_float_exception_flags(old_exc_flags, fpst);
83
- case ARMMMUIdx_SE10_1:
87
-
84
- case ARMMMUIdx_SE10_1_PAN:
88
- return ret;
85
- case ARMMMUIdx_Stage1_E0:
86
- case ARMMMUIdx_Stage1_E1:
87
- case ARMMMUIdx_Stage1_E1_PAN:
88
- case ARMMMUIdx_E10_0:
89
- case ARMMMUIdx_E10_1:
90
- case ARMMMUIdx_E10_1_PAN:
91
- case ARMMMUIdx_MPrivNegPri:
92
- case ARMMMUIdx_MUserNegPri:
93
- case ARMMMUIdx_MPriv:
94
- case ARMMMUIdx_MUser:
95
- case ARMMMUIdx_MSPrivNegPri:
96
- case ARMMMUIdx_MSUserNegPri:
97
- case ARMMMUIdx_MSPriv:
98
- case ARMMMUIdx_MSUser:
99
- return 1;
100
- default:
101
- g_assert_not_reached();
89
- }
102
- }
90
-}
103
-}
91
-
104
-
92
uint32_t HELPER(vfp_toshh)(uint32_t x, uint32_t shift, void *fpst)
105
uint64_t arm_sctlr(CPUARMState *env, int el)
93
{
106
{
94
- return float64_to_int16(do_prescale_fp16(x, shift, fpst), fpst);
107
/* Only EL0 needs to be adjusted for EL1&0 or EL2&0. */
95
+ if (unlikely(float16_is_any_nan(x))) {
96
+ float_raise(float_flag_invalid, fpst);
97
+ return 0;
98
+ }
99
+ return float16_to_int16_scalbn(x, get_float_rounding_mode(fpst),
100
+ shift, fpst);
101
}
102
103
uint32_t HELPER(vfp_touhh)(uint32_t x, uint32_t shift, void *fpst)
104
{
105
- return float64_to_uint16(do_prescale_fp16(x, shift, fpst), fpst);
106
+ if (unlikely(float16_is_any_nan(x))) {
107
+ float_raise(float_flag_invalid, fpst);
108
+ return 0;
109
+ }
110
+ return float16_to_uint16_scalbn(x, get_float_rounding_mode(fpst),
111
+ shift, fpst);
112
}
113
114
uint32_t HELPER(vfp_toslh)(uint32_t x, uint32_t shift, void *fpst)
115
{
116
- return float64_to_int32(do_prescale_fp16(x, shift, fpst), fpst);
117
+ if (unlikely(float16_is_any_nan(x))) {
118
+ float_raise(float_flag_invalid, fpst);
119
+ return 0;
120
+ }
121
+ return float16_to_int32_scalbn(x, get_float_rounding_mode(fpst),
122
+ shift, fpst);
123
}
124
125
uint32_t HELPER(vfp_toulh)(uint32_t x, uint32_t shift, void *fpst)
126
{
127
- return float64_to_uint32(do_prescale_fp16(x, shift, fpst), fpst);
128
+ if (unlikely(float16_is_any_nan(x))) {
129
+ float_raise(float_flag_invalid, fpst);
130
+ return 0;
131
+ }
132
+ return float16_to_uint32_scalbn(x, get_float_rounding_mode(fpst),
133
+ shift, fpst);
134
}
135
136
uint64_t HELPER(vfp_tosqh)(uint32_t x, uint32_t shift, void *fpst)
137
{
138
- return float64_to_int64(do_prescale_fp16(x, shift, fpst), fpst);
139
+ if (unlikely(float16_is_any_nan(x))) {
140
+ float_raise(float_flag_invalid, fpst);
141
+ return 0;
142
+ }
143
+ return float16_to_int64_scalbn(x, get_float_rounding_mode(fpst),
144
+ shift, fpst);
145
}
146
147
uint64_t HELPER(vfp_touqh)(uint32_t x, uint32_t shift, void *fpst)
148
{
149
- return float64_to_uint64(do_prescale_fp16(x, shift, fpst), fpst);
150
+ if (unlikely(float16_is_any_nan(x))) {
151
+ float_raise(float_flag_invalid, fpst);
152
+ return 0;
153
+ }
154
+ return float16_to_uint64_scalbn(x, get_float_rounding_mode(fpst),
155
+ shift, fpst);
156
}
157
158
/* Set the current fp rounding mode and return the old one.
159
--
108
--
160
2.18.0
109
2.20.1
161
110
162
111
diff view generated by jsdifflib
1
Connect the VIRQ and VFIQ lines from the GIC to the CPU;
1
From: Richard Henderson <richard.henderson@linaro.org>
2
these exist always for both CPU and GIC whether the
3
virtualization extensions are enabled or not, so we
4
can just unconditionally connect them.
5
2
3
We will shortly need this in mte_helper.c as well.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-23-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180821132811.17675-4-peter.maydell@linaro.org
9
---
9
---
10
hw/arm/vexpress.c | 4 ++++
10
target/arm/internals.h | 9 +++++++++
11
1 file changed, 4 insertions(+)
11
target/arm/helper.c | 9 ---------
12
2 files changed, 9 insertions(+), 9 deletions(-)
12
13
13
diff --git a/hw/arm/vexpress.c b/hw/arm/vexpress.c
14
diff --git a/target/arm/internals.h b/target/arm/internals.h
14
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/vexpress.c
16
--- a/target/arm/internals.h
16
+++ b/hw/arm/vexpress.c
17
+++ b/target/arm/internals.h
17
@@ -XXX,XX +XXX,XX @@ static void init_cpus(const char *cpu_type, const char *privdev,
18
@@ -XXX,XX +XXX,XX @@ static inline uint32_t regime_el(CPUARMState *env, ARMMMUIdx mmu_idx)
18
sysbus_connect_irq(busdev, n, qdev_get_gpio_in(cpudev, ARM_CPU_IRQ));
19
sysbus_connect_irq(busdev, n + smp_cpus,
20
qdev_get_gpio_in(cpudev, ARM_CPU_FIQ));
21
+ sysbus_connect_irq(busdev, n + 2 * smp_cpus,
22
+ qdev_get_gpio_in(cpudev, ARM_CPU_VIRQ));
23
+ sysbus_connect_irq(busdev, n + 3 * smp_cpus,
24
+ qdev_get_gpio_in(cpudev, ARM_CPU_VFIQ));
25
}
19
}
26
}
20
}
27
21
22
+/* Return the TCR controlling this translation regime */
23
+static inline TCR *regime_tcr(CPUARMState *env, ARMMMUIdx mmu_idx)
24
+{
25
+ if (mmu_idx == ARMMMUIdx_Stage2) {
26
+ return &env->cp15.vtcr_el2;
27
+ }
28
+ return &env->cp15.tcr_el[regime_el(env, mmu_idx)];
29
+}
30
+
31
/* Return the FSR value for a debug exception (watchpoint, hardware
32
* breakpoint or BKPT insn) targeting the specified exception level.
33
*/
34
diff --git a/target/arm/helper.c b/target/arm/helper.c
35
index XXXXXXX..XXXXXXX 100644
36
--- a/target/arm/helper.c
37
+++ b/target/arm/helper.c
38
@@ -XXX,XX +XXX,XX @@ static inline uint64_t regime_ttbr(CPUARMState *env, ARMMMUIdx mmu_idx,
39
40
#endif /* !CONFIG_USER_ONLY */
41
42
-/* Return the TCR controlling this translation regime */
43
-static inline TCR *regime_tcr(CPUARMState *env, ARMMMUIdx mmu_idx)
44
-{
45
- if (mmu_idx == ARMMMUIdx_Stage2) {
46
- return &env->cp15.vtcr_el2;
47
- }
48
- return &env->cp15.tcr_el[regime_el(env, mmu_idx)];
49
-}
50
-
51
/* Convert a possible stage1+2 MMU index into the appropriate
52
* stage 1 MMU index
53
*/
28
--
54
--
29
2.18.0
55
2.20.1
30
56
31
57
diff view generated by jsdifflib
1
Refactor bcm2835_fb_mbox_push() to work by calling
1
From: Richard Henderson <richard.henderson@linaro.org>
2
bcm2835_fb_validate_config() and bcm2835_fb_reconfigure(),
2
3
so that config set this way is also validated.
3
Replace existing uses of check_data_tbi in translate-a64.c that
4
4
perform a single logical memory access. Leave the helper blank
5
for now to reduce the patch size.
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-24-richard.henderson@linaro.org
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180814144436.679-9-peter.maydell@linaro.org
8
---
11
---
9
hw/display/bcm2835_fb.c | 63 ++++++++++++++++++++---------------------
12
target/arm/helper-a64.h | 1 +
10
1 file changed, 31 insertions(+), 32 deletions(-)
13
target/arm/internals.h | 8 +++
11
14
target/arm/translate-a64.h | 2 +
12
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
15
target/arm/mte_helper.c | 8 +++
13
index XXXXXXX..XXXXXXX 100644
16
target/arm/translate-a64.c | 100 ++++++++++++++++++++++++++++---------
14
--- a/hw/display/bcm2835_fb.c
17
5 files changed, 95 insertions(+), 24 deletions(-)
15
+++ b/hw/display/bcm2835_fb.c
18
16
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_validate_config(BCM2835FBConfig *config)
19
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
20
index XXXXXXX..XXXXXXX 100644
21
--- a/target/arm/helper-a64.h
22
+++ b/target/arm/helper-a64.h
23
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_3(autdb, TCG_CALL_NO_WG, i64, env, i64, i64)
24
DEF_HELPER_FLAGS_2(xpaci, TCG_CALL_NO_RWG_SE, i64, env, i64)
25
DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
26
27
+DEF_HELPER_FLAGS_3(mte_check1, TCG_CALL_NO_WG, i64, env, i32, i64)
28
DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
29
DEF_HELPER_FLAGS_4(addsubg, TCG_CALL_NO_RWG_SE, i64, env, i64, s32, i32)
30
DEF_HELPER_FLAGS_3(ldg, TCG_CALL_NO_WG, i64, env, i64, i64)
31
diff --git a/target/arm/internals.h b/target/arm/internals.h
32
index XXXXXXX..XXXXXXX 100644
33
--- a/target/arm/internals.h
34
+++ b/target/arm/internals.h
35
@@ -XXX,XX +XXX,XX @@ void arm_log_exception(int idx);
36
#define LOG2_TAG_GRANULE 4
37
#define TAG_GRANULE (1 << LOG2_TAG_GRANULE)
38
39
+/* Bits within a descriptor passed to the helper_mte_check* functions. */
40
+FIELD(MTEDESC, MIDX, 0, 4)
41
+FIELD(MTEDESC, TBI, 4, 2)
42
+FIELD(MTEDESC, TCMA, 6, 2)
43
+FIELD(MTEDESC, WRITE, 8, 1)
44
+FIELD(MTEDESC, ESIZE, 9, 5)
45
+FIELD(MTEDESC, TSIZE, 14, 10) /* mte_checkN only */
46
+
47
static inline int allocation_tag_from_addr(uint64_t ptr)
48
{
49
return extract64(ptr, 56, 4);
50
diff --git a/target/arm/translate-a64.h b/target/arm/translate-a64.h
51
index XXXXXXX..XXXXXXX 100644
52
--- a/target/arm/translate-a64.h
53
+++ b/target/arm/translate-a64.h
54
@@ -XXX,XX +XXX,XX @@ TCGv_ptr get_fpstatus_ptr(bool);
55
bool logic_imm_decode_wmask(uint64_t *result, unsigned int immn,
56
unsigned int imms, unsigned int immr);
57
bool sve_access_check(DisasContext *s);
58
+TCGv_i64 gen_mte_check1(DisasContext *s, TCGv_i64 addr, bool is_write,
59
+ bool tag_checked, int log2_size);
60
61
/* We should have at some point before trying to access an FP register
62
* done the necessary access check, so assert that
63
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
64
index XXXXXXX..XXXXXXX 100644
65
--- a/target/arm/mte_helper.c
66
+++ b/target/arm/mte_helper.c
67
@@ -XXX,XX +XXX,XX @@ void HELPER(stzgm_tags)(CPUARMState *env, uint64_t ptr, uint64_t val)
68
memset(mem, tag_pair, tag_bytes);
17
}
69
}
18
}
70
}
19
71
+
20
-static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
72
+/*
21
-{
73
+ * Perform an MTE checked access for a single logical or atomic access.
22
- uint32_t pitch;
74
+ */
23
- uint32_t size;
75
+uint64_t HELPER(mte_check1)(CPUARMState *env, uint32_t desc, uint64_t ptr)
24
-
76
+{
25
- value &= ~0xf;
77
+ return ptr;
26
-
78
+}
27
- s->lock = true;
79
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
28
-
80
index XXXXXXX..XXXXXXX 100644
29
- s->config.xres = ldl_le_phys(&s->dma_as, value);
81
--- a/target/arm/translate-a64.c
30
- s->config.yres = ldl_le_phys(&s->dma_as, value + 4);
82
+++ b/target/arm/translate-a64.c
31
- s->config.xres_virtual = ldl_le_phys(&s->dma_as, value + 8);
83
@@ -XXX,XX +XXX,XX @@ static void gen_a64_set_pc(DisasContext *s, TCGv_i64 src)
32
- s->config.yres_virtual = ldl_le_phys(&s->dma_as, value + 12);
84
}
33
- s->config.bpp = ldl_le_phys(&s->dma_as, value + 20);
85
34
- s->config.xoffset = ldl_le_phys(&s->dma_as, value + 24);
86
/*
35
- s->config.yoffset = ldl_le_phys(&s->dma_as, value + 28);
87
- * Return a "clean" address for ADDR according to TBID.
36
-
88
- * This is always a fresh temporary, as we need to be able to
37
- s->config.base = s->vcram_base | (value & 0xc0000000);
89
- * increment this independently of a dirty write-back address.
38
- s->config.base += BCM2835_FB_OFFSET;
90
+ * Handle MTE and/or TBI.
39
-
91
+ *
40
- pitch = bcm2835_fb_get_pitch(&s->config);
92
+ * For TBI, ideally, we would do nothing. Proper behaviour on fault is
41
- size = bcm2835_fb_get_size(&s->config);
93
+ * for the tag to be present in the FAR_ELx register. But for user-only
42
-
94
+ * mode we do not have a TLB with which to implement this, so we must
43
- stl_le_phys(&s->dma_as, value + 16, pitch);
95
+ * remove the top byte now.
44
- stl_le_phys(&s->dma_as, value + 32, s->config.base);
96
+ *
45
- stl_le_phys(&s->dma_as, value + 36, size);
97
+ * Always return a fresh temporary that we can increment independently
46
-
98
+ * of the write-back address.
47
- s->invalidate = true;
99
*/
48
- qemu_console_resize(s->con, s->config.xres, s->config.yres);
100
+
49
- s->lock = false;
101
static TCGv_i64 clean_data_tbi(DisasContext *s, TCGv_i64 addr)
50
-}
51
-
52
void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
53
{
102
{
54
s->lock = true;
103
TCGv_i64 clean = new_tmp_a64(s);
55
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
104
- /*
56
s->lock = false;
105
- * In order to get the correct value in the FAR_ELx register,
106
- * we must present the memory subsystem with the "dirty" address
107
- * including the TBI. In system mode we can make this work via
108
- * the TLB, dropping the TBI during translation. But for user-only
109
- * mode we don't have that option, and must remove the top byte now.
110
- */
111
#ifdef CONFIG_USER_ONLY
112
gen_top_byte_ignore(s, clean, addr, s->tbid);
113
#else
114
@@ -XXX,XX +XXX,XX @@ static void gen_probe_access(DisasContext *s, TCGv_i64 ptr,
115
tcg_temp_free_i32(t_size);
57
}
116
}
58
117
59
+static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
118
+/*
119
+ * For MTE, check a single logical or atomic access. This probes a single
120
+ * address, the exact one specified. The size and alignment of the access
121
+ * is not relevant to MTE, per se, but watchpoints do require the size,
122
+ * and we want to recognize those before making any other changes to state.
123
+ */
124
+static TCGv_i64 gen_mte_check1_mmuidx(DisasContext *s, TCGv_i64 addr,
125
+ bool is_write, bool tag_checked,
126
+ int log2_size, bool is_unpriv,
127
+ int core_idx)
60
+{
128
+{
61
+ uint32_t pitch;
129
+ if (tag_checked && s->mte_active[is_unpriv]) {
62
+ uint32_t size;
130
+ TCGv_i32 tcg_desc;
63
+ BCM2835FBConfig newconf;
131
+ TCGv_i64 ret;
64
+
132
+ int desc = 0;
65
+ value &= ~0xf;
133
+
66
+
134
+ desc = FIELD_DP32(desc, MTEDESC, MIDX, core_idx);
67
+ newconf.xres = ldl_le_phys(&s->dma_as, value);
135
+ desc = FIELD_DP32(desc, MTEDESC, TBI, s->tbid);
68
+ newconf.yres = ldl_le_phys(&s->dma_as, value + 4);
136
+ desc = FIELD_DP32(desc, MTEDESC, TCMA, s->tcma);
69
+ newconf.xres_virtual = ldl_le_phys(&s->dma_as, value + 8);
137
+ desc = FIELD_DP32(desc, MTEDESC, WRITE, is_write);
70
+ newconf.yres_virtual = ldl_le_phys(&s->dma_as, value + 12);
138
+ desc = FIELD_DP32(desc, MTEDESC, ESIZE, 1 << log2_size);
71
+ newconf.bpp = ldl_le_phys(&s->dma_as, value + 20);
139
+ tcg_desc = tcg_const_i32(desc);
72
+ newconf.xoffset = ldl_le_phys(&s->dma_as, value + 24);
140
+
73
+ newconf.yoffset = ldl_le_phys(&s->dma_as, value + 28);
141
+ ret = new_tmp_a64(s);
74
+
142
+ gen_helper_mte_check1(ret, cpu_env, tcg_desc, addr);
75
+ newconf.base = s->vcram_base | (value & 0xc0000000);
143
+ tcg_temp_free_i32(tcg_desc);
76
+ newconf.base += BCM2835_FB_OFFSET;
144
+
77
+
145
+ return ret;
78
+ bcm2835_fb_validate_config(&newconf);
146
+ }
79
+
147
+ return clean_data_tbi(s, addr);
80
+ pitch = bcm2835_fb_get_pitch(&newconf);
81
+ size = bcm2835_fb_get_size(&newconf);
82
+
83
+ stl_le_phys(&s->dma_as, value + 16, pitch);
84
+ stl_le_phys(&s->dma_as, value + 32, newconf.base);
85
+ stl_le_phys(&s->dma_as, value + 36, size);
86
+
87
+ bcm2835_fb_reconfigure(s, &newconf);
88
+}
148
+}
89
+
149
+
90
static uint64_t bcm2835_fb_read(void *opaque, hwaddr offset, unsigned size)
150
+TCGv_i64 gen_mte_check1(DisasContext *s, TCGv_i64 addr, bool is_write,
91
{
151
+ bool tag_checked, int log2_size)
92
BCM2835FBState *s = opaque;
152
+{
153
+ return gen_mte_check1_mmuidx(s, addr, is_write, tag_checked, log2_size,
154
+ false, get_mem_index(s));
155
+}
156
+
157
typedef struct DisasCompare64 {
158
TCGCond cond;
159
TCGv_i64 value;
160
@@ -XXX,XX +XXX,XX @@ static void gen_compare_and_swap(DisasContext *s, int rs, int rt,
161
if (rn == 31) {
162
gen_check_sp_alignment(s);
163
}
164
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
165
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn), true, rn != 31, size);
166
tcg_gen_atomic_cmpxchg_i64(tcg_rs, clean_addr, tcg_rs, tcg_rt, memidx,
167
size | MO_ALIGN | s->be_data);
168
}
169
@@ -XXX,XX +XXX,XX @@ static void gen_compare_and_swap_pair(DisasContext *s, int rs, int rt,
170
if (rn == 31) {
171
gen_check_sp_alignment(s);
172
}
173
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
174
+
175
+ /* This is a single atomic access, despite the "pair". */
176
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn), true, rn != 31, size + 1);
177
178
if (size == 2) {
179
TCGv_i64 cmp = tcg_temp_new_i64();
180
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
181
if (is_lasr) {
182
tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
183
}
184
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
185
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
186
+ true, rn != 31, size);
187
gen_store_exclusive(s, rs, rt, rt2, clean_addr, size, false);
188
return;
189
190
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
191
if (rn == 31) {
192
gen_check_sp_alignment(s);
193
}
194
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
195
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
196
+ false, rn != 31, size);
197
s->is_ldex = true;
198
gen_load_exclusive(s, rt, rt2, clean_addr, size, false);
199
if (is_lasr) {
200
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
201
gen_check_sp_alignment(s);
202
}
203
tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
204
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
205
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
206
+ true, rn != 31, size);
207
do_gpr_st(s, cpu_reg(s, rt), clean_addr, size, true, rt,
208
disas_ldst_compute_iss_sf(size, false, 0), is_lasr);
209
return;
210
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
211
if (rn == 31) {
212
gen_check_sp_alignment(s);
213
}
214
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
215
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
216
+ false, rn != 31, size);
217
do_gpr_ld(s, cpu_reg(s, rt), clean_addr, size, false, false, true, rt,
218
disas_ldst_compute_iss_sf(size, false, 0), is_lasr);
219
tcg_gen_mb(TCG_MO_ALL | TCG_BAR_LDAQ);
220
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
221
if (is_lasr) {
222
tcg_gen_mb(TCG_MO_ALL | TCG_BAR_STRL);
223
}
224
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
225
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
226
+ true, rn != 31, size);
227
gen_store_exclusive(s, rs, rt, rt2, clean_addr, size, true);
228
return;
229
}
230
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_excl(DisasContext *s, uint32_t insn)
231
if (rn == 31) {
232
gen_check_sp_alignment(s);
233
}
234
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
235
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn),
236
+ false, rn != 31, size);
237
s->is_ldex = true;
238
gen_load_exclusive(s, rt, rt2, clean_addr, size, true);
239
if (is_lasr) {
240
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_imm9(DisasContext *s, uint32_t insn,
241
bool iss_valid = !is_vector;
242
bool post_index;
243
bool writeback;
244
+ int memidx;
245
246
TCGv_i64 clean_addr, dirty_addr;
247
248
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_imm9(DisasContext *s, uint32_t insn,
249
if (!post_index) {
250
tcg_gen_addi_i64(dirty_addr, dirty_addr, imm9);
251
}
252
- clean_addr = clean_data_tbi(s, dirty_addr);
253
+
254
+ memidx = is_unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
255
+ clean_addr = gen_mte_check1_mmuidx(s, dirty_addr, is_store,
256
+ writeback || rn != 31,
257
+ size, is_unpriv, memidx);
258
259
if (is_vector) {
260
if (is_store) {
261
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_imm9(DisasContext *s, uint32_t insn,
262
}
263
} else {
264
TCGv_i64 tcg_rt = cpu_reg(s, rt);
265
- int memidx = is_unpriv ? get_a64_user_mem_index(s) : get_mem_index(s);
266
bool iss_sf = disas_ldst_compute_iss_sf(size, is_signed, opc);
267
268
if (is_store) {
269
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_roffset(DisasContext *s, uint32_t insn,
270
ext_and_shift_reg(tcg_rm, tcg_rm, opt, shift ? size : 0);
271
272
tcg_gen_add_i64(dirty_addr, dirty_addr, tcg_rm);
273
- clean_addr = clean_data_tbi(s, dirty_addr);
274
+ clean_addr = gen_mte_check1(s, dirty_addr, is_store, true, size);
275
276
if (is_vector) {
277
if (is_store) {
278
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_reg_unsigned_imm(DisasContext *s, uint32_t insn,
279
dirty_addr = read_cpu_reg_sp(s, rn, 1);
280
offset = imm12 << size;
281
tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
282
- clean_addr = clean_data_tbi(s, dirty_addr);
283
+ clean_addr = gen_mte_check1(s, dirty_addr, is_store, rn != 31, size);
284
285
if (is_vector) {
286
if (is_store) {
287
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_atomic(DisasContext *s, uint32_t insn,
288
if (rn == 31) {
289
gen_check_sp_alignment(s);
290
}
291
- clean_addr = clean_data_tbi(s, cpu_reg_sp(s, rn));
292
+ clean_addr = gen_mte_check1(s, cpu_reg_sp(s, rn), false, rn != 31, size);
293
294
if (o3_opc == 014) {
295
/*
296
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pac(DisasContext *s, uint32_t insn,
297
tcg_gen_addi_i64(dirty_addr, dirty_addr, offset);
298
299
/* Note that "clean" and "dirty" here refer to TBI not PAC. */
300
- clean_addr = clean_data_tbi(s, dirty_addr);
301
+ clean_addr = gen_mte_check1(s, dirty_addr, false,
302
+ is_wback || rn != 31, size);
303
304
tcg_rt = cpu_reg(s, rt);
305
do_gpr_ld(s, tcg_rt, clean_addr, size, /* is_signed */ false,
93
--
306
--
94
2.18.0
307
2.20.1
95
308
96
309
diff view generated by jsdifflib
New patch
1
1
From: Richard Henderson <richard.henderson@linaro.org>
2
3
Replace existing uses of check_data_tbi in translate-a64.c that
4
perform multiple logical memory access. Leave the helper blank
5
for now to reduce the patch size.
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-25-richard.henderson@linaro.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
target/arm/helper-a64.h | 1 +
13
target/arm/translate-a64.h | 2 ++
14
target/arm/mte_helper.c | 8 +++++
15
target/arm/translate-a64.c | 71 +++++++++++++++++++++++++++++---------
16
4 files changed, 66 insertions(+), 16 deletions(-)
17
18
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
19
index XXXXXXX..XXXXXXX 100644
20
--- a/target/arm/helper-a64.h
21
+++ b/target/arm/helper-a64.h
22
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(xpaci, TCG_CALL_NO_RWG_SE, i64, env, i64)
23
DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
24
25
DEF_HELPER_FLAGS_3(mte_check1, TCG_CALL_NO_WG, i64, env, i32, i64)
26
+DEF_HELPER_FLAGS_3(mte_checkN, TCG_CALL_NO_WG, i64, env, i32, i64)
27
DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
28
DEF_HELPER_FLAGS_4(addsubg, TCG_CALL_NO_RWG_SE, i64, env, i64, s32, i32)
29
DEF_HELPER_FLAGS_3(ldg, TCG_CALL_NO_WG, i64, env, i64, i64)
30
diff --git a/target/arm/translate-a64.h b/target/arm/translate-a64.h
31
index XXXXXXX..XXXXXXX 100644
32
--- a/target/arm/translate-a64.h
33
+++ b/target/arm/translate-a64.h
34
@@ -XXX,XX +XXX,XX @@ bool logic_imm_decode_wmask(uint64_t *result, unsigned int immn,
35
bool sve_access_check(DisasContext *s);
36
TCGv_i64 gen_mte_check1(DisasContext *s, TCGv_i64 addr, bool is_write,
37
bool tag_checked, int log2_size);
38
+TCGv_i64 gen_mte_checkN(DisasContext *s, TCGv_i64 addr, bool is_write,
39
+ bool tag_checked, int count, int log2_esize);
40
41
/* We should have at some point before trying to access an FP register
42
* done the necessary access check, so assert that
43
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
44
index XXXXXXX..XXXXXXX 100644
45
--- a/target/arm/mte_helper.c
46
+++ b/target/arm/mte_helper.c
47
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(mte_check1)(CPUARMState *env, uint32_t desc, uint64_t ptr)
48
{
49
return ptr;
50
}
51
+
52
+/*
53
+ * Perform an MTE checked access for multiple logical accesses.
54
+ */
55
+uint64_t HELPER(mte_checkN)(CPUARMState *env, uint32_t desc, uint64_t ptr)
56
+{
57
+ return ptr;
58
+}
59
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
60
index XXXXXXX..XXXXXXX 100644
61
--- a/target/arm/translate-a64.c
62
+++ b/target/arm/translate-a64.c
63
@@ -XXX,XX +XXX,XX @@ TCGv_i64 gen_mte_check1(DisasContext *s, TCGv_i64 addr, bool is_write,
64
false, get_mem_index(s));
65
}
66
67
+/*
68
+ * For MTE, check multiple logical sequential accesses.
69
+ */
70
+TCGv_i64 gen_mte_checkN(DisasContext *s, TCGv_i64 addr, bool is_write,
71
+ bool tag_checked, int log2_esize, int total_size)
72
+{
73
+ if (tag_checked && s->mte_active[0] && total_size != (1 << log2_esize)) {
74
+ TCGv_i32 tcg_desc;
75
+ TCGv_i64 ret;
76
+ int desc = 0;
77
+
78
+ desc = FIELD_DP32(desc, MTEDESC, MIDX, get_mem_index(s));
79
+ desc = FIELD_DP32(desc, MTEDESC, TBI, s->tbid);
80
+ desc = FIELD_DP32(desc, MTEDESC, TCMA, s->tcma);
81
+ desc = FIELD_DP32(desc, MTEDESC, WRITE, is_write);
82
+ desc = FIELD_DP32(desc, MTEDESC, ESIZE, 1 << log2_esize);
83
+ desc = FIELD_DP32(desc, MTEDESC, TSIZE, total_size);
84
+ tcg_desc = tcg_const_i32(desc);
85
+
86
+ ret = new_tmp_a64(s);
87
+ gen_helper_mte_checkN(ret, cpu_env, tcg_desc, addr);
88
+ tcg_temp_free_i32(tcg_desc);
89
+
90
+ return ret;
91
+ }
92
+ return gen_mte_check1(s, addr, is_write, tag_checked, log2_esize);
93
+}
94
+
95
typedef struct DisasCompare64 {
96
TCGCond cond;
97
TCGv_i64 value;
98
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_pair(DisasContext *s, uint32_t insn)
99
}
100
}
101
102
- clean_addr = clean_data_tbi(s, dirty_addr);
103
+ clean_addr = gen_mte_checkN(s, dirty_addr, !is_load,
104
+ (wback || rn != 31) && !set_tag,
105
+ size, 2 << size);
106
+
107
if (is_vector) {
108
if (is_load) {
109
do_fp_ld(s, rt, clean_addr, size);
110
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
111
TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
112
MemOp endian = s->be_data;
113
114
- int ebytes; /* bytes per element */
115
+ int total; /* total bytes */
116
int elements; /* elements per vector */
117
int rpt; /* num iterations */
118
int selem; /* structure elements */
119
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
120
endian = MO_LE;
121
}
122
123
- /* Consecutive little-endian elements from a single register
124
+ total = rpt * selem * (is_q ? 16 : 8);
125
+ tcg_rn = cpu_reg_sp(s, rn);
126
+
127
+ /*
128
+ * Issue the MTE check vs the logical repeat count, before we
129
+ * promote consecutive little-endian elements below.
130
+ */
131
+ clean_addr = gen_mte_checkN(s, tcg_rn, is_store, is_postidx || rn != 31,
132
+ size, total);
133
+
134
+ /*
135
+ * Consecutive little-endian elements from a single register
136
* can be promoted to a larger little-endian operation.
137
*/
138
if (selem == 1 && endian == MO_LE) {
139
size = 3;
140
}
141
- ebytes = 1 << size;
142
- elements = (is_q ? 16 : 8) / ebytes;
143
-
144
- tcg_rn = cpu_reg_sp(s, rn);
145
- clean_addr = clean_data_tbi(s, tcg_rn);
146
- tcg_ebytes = tcg_const_i64(ebytes);
147
+ elements = (is_q ? 16 : 8) >> size;
148
149
+ tcg_ebytes = tcg_const_i64(1 << size);
150
for (r = 0; r < rpt; r++) {
151
int e;
152
for (e = 0; e < elements; e++) {
153
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_multiple_struct(DisasContext *s, uint32_t insn)
154
155
if (is_postidx) {
156
if (rm == 31) {
157
- tcg_gen_addi_i64(tcg_rn, tcg_rn, rpt * elements * selem * ebytes);
158
+ tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
159
} else {
160
tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, rm));
161
}
162
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
163
int selem = (extract32(opc, 0, 1) << 1 | R) + 1;
164
bool replicate = false;
165
int index = is_q << 3 | S << 2 | size;
166
- int ebytes, xs;
167
+ int xs, total;
168
TCGv_i64 clean_addr, tcg_rn, tcg_ebytes;
169
170
if (extract32(insn, 31, 1)) {
171
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
172
return;
173
}
174
175
- ebytes = 1 << scale;
176
-
177
if (rn == 31) {
178
gen_check_sp_alignment(s);
179
}
180
181
+ total = selem << scale;
182
tcg_rn = cpu_reg_sp(s, rn);
183
- clean_addr = clean_data_tbi(s, tcg_rn);
184
- tcg_ebytes = tcg_const_i64(ebytes);
185
186
+ clean_addr = gen_mte_checkN(s, tcg_rn, !is_load, is_postidx || rn != 31,
187
+ scale, total);
188
+
189
+ tcg_ebytes = tcg_const_i64(1 << scale);
190
for (xs = 0; xs < selem; xs++) {
191
if (replicate) {
192
/* Load and replicate to all elements */
193
@@ -XXX,XX +XXX,XX @@ static void disas_ldst_single_struct(DisasContext *s, uint32_t insn)
194
195
if (is_postidx) {
196
if (rm == 31) {
197
- tcg_gen_addi_i64(tcg_rn, tcg_rn, selem * ebytes);
198
+ tcg_gen_addi_i64(tcg_rn, tcg_rn, total);
199
} else {
200
tcg_gen_add_i64(tcg_rn, tcg_rn, cpu_reg(s, rm));
201
}
202
--
203
2.20.1
204
205
diff view generated by jsdifflib
1
Use the DeviceState vmsd pointer rather than calling vmstate_register()
1
From: Richard Henderson <richard.henderson@linaro.org>
2
directly.
2
3
3
Fill out the stub that was added earlier.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-26-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Message-id: 20180820141116.9118-18-peter.maydell@linaro.org
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
---
9
---
8
hw/ssi/pl022.c | 2 +-
10
target/arm/internals.h | 48 +++++++++++++++
9
1 file changed, 1 insertion(+), 1 deletion(-)
11
target/arm/mte_helper.c | 132 +++++++++++++++++++++++++++++++++++++++-
10
12
2 files changed, 179 insertions(+), 1 deletion(-)
11
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
13
14
diff --git a/target/arm/internals.h b/target/arm/internals.h
12
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
13
--- a/hw/ssi/pl022.c
16
--- a/target/arm/internals.h
14
+++ b/hw/ssi/pl022.c
17
+++ b/target/arm/internals.h
15
@@ -XXX,XX +XXX,XX @@ static int pl022_init(SysBusDevice *sbd)
18
@@ -XXX,XX +XXX,XX @@ FIELD(MTEDESC, WRITE, 8, 1)
16
sysbus_init_mmio(sbd, &s->iomem);
19
FIELD(MTEDESC, ESIZE, 9, 5)
17
sysbus_init_irq(sbd, &s->irq);
20
FIELD(MTEDESC, TSIZE, 14, 10) /* mte_checkN only */
18
s->ssi = ssi_create_bus(dev, "ssi");
21
19
- vmstate_register(dev, -1, &vmstate_pl022, s);
22
+bool mte_probe1(CPUARMState *env, uint32_t desc, uint64_t ptr);
20
return 0;
23
+uint64_t mte_check1(CPUARMState *env, uint32_t desc,
24
+ uint64_t ptr, uintptr_t ra);
25
+
26
static inline int allocation_tag_from_addr(uint64_t ptr)
27
{
28
return extract64(ptr, 56, 4);
29
@@ -XXX,XX +XXX,XX @@ static inline uint64_t address_with_allocation_tag(uint64_t ptr, int rtag)
30
return deposit64(ptr, 56, 4, rtag);
21
}
31
}
22
32
23
@@ -XXX,XX +XXX,XX @@ static void pl022_class_init(ObjectClass *klass, void *data)
33
+/* Return true if tbi bits mean that the access is checked. */
24
34
+static inline bool tbi_check(uint32_t desc, int bit55)
25
sdc->init = pl022_init;
35
+{
26
dc->reset = pl022_reset;
36
+ return (desc >> (R_MTEDESC_TBI_SHIFT + bit55)) & 1;
27
+ dc->vmsd = &vmstate_pl022;
37
+}
38
+
39
+/* Return true if tcma bits mean that the access is unchecked. */
40
+static inline bool tcma_check(uint32_t desc, int bit55, int ptr_tag)
41
+{
42
+ /*
43
+ * We had extracted bit55 and ptr_tag for other reasons, so fold
44
+ * (ptr<59:55> == 00000 || ptr<59:55> == 11111) into a single test.
45
+ */
46
+ bool match = ((ptr_tag + bit55) & 0xf) == 0;
47
+ bool tcma = (desc >> (R_MTEDESC_TCMA_SHIFT + bit55)) & 1;
48
+ return tcma && match;
49
+}
50
+
51
+/*
52
+ * For TBI, ideally, we would do nothing. Proper behaviour on fault is
53
+ * for the tag to be present in the FAR_ELx register. But for user-only
54
+ * mode, we do not have a TLB with which to implement this, so we must
55
+ * remove the top byte.
56
+ */
57
+static inline uint64_t useronly_clean_ptr(uint64_t ptr)
58
+{
59
+ /* TBI is known to be enabled. */
60
+#ifdef CONFIG_USER_ONLY
61
+ ptr = sextract64(ptr, 0, 56);
62
+#endif
63
+ return ptr;
64
+}
65
+
66
+static inline uint64_t useronly_maybe_clean_ptr(uint32_t desc, uint64_t ptr)
67
+{
68
+#ifdef CONFIG_USER_ONLY
69
+ int64_t clean_ptr = sextract64(ptr, 0, 56);
70
+ if (tbi_check(desc, clean_ptr < 0)) {
71
+ ptr = clean_ptr;
72
+ }
73
+#endif
74
+ return ptr;
75
+}
76
+
77
#endif
78
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
79
index XXXXXXX..XXXXXXX 100644
80
--- a/target/arm/mte_helper.c
81
+++ b/target/arm/mte_helper.c
82
@@ -XXX,XX +XXX,XX @@ void HELPER(stzgm_tags)(CPUARMState *env, uint64_t ptr, uint64_t val)
83
}
28
}
84
}
29
85
30
static const TypeInfo pl022_info = {
86
+/* Record a tag check failure. */
87
+static void mte_check_fail(CPUARMState *env, int mmu_idx,
88
+ uint64_t dirty_ptr, uintptr_t ra)
89
+{
90
+ ARMMMUIdx arm_mmu_idx = core_to_aa64_mmu_idx(mmu_idx);
91
+ int el, reg_el, tcf, select;
92
+ uint64_t sctlr;
93
+
94
+ reg_el = regime_el(env, arm_mmu_idx);
95
+ sctlr = env->cp15.sctlr_el[reg_el];
96
+
97
+ switch (arm_mmu_idx) {
98
+ case ARMMMUIdx_E10_0:
99
+ case ARMMMUIdx_E20_0:
100
+ el = 0;
101
+ tcf = extract64(sctlr, 38, 2);
102
+ break;
103
+ default:
104
+ el = reg_el;
105
+ tcf = extract64(sctlr, 40, 2);
106
+ }
107
+
108
+ switch (tcf) {
109
+ case 1:
110
+ /*
111
+ * Tag check fail causes a synchronous exception.
112
+ *
113
+ * In restore_state_to_opc, we set the exception syndrome
114
+ * for the load or store operation. Unwind first so we
115
+ * may overwrite that with the syndrome for the tag check.
116
+ */
117
+ cpu_restore_state(env_cpu(env), ra, true);
118
+ env->exception.vaddress = dirty_ptr;
119
+ raise_exception(env, EXCP_DATA_ABORT,
120
+ syn_data_abort_no_iss(el != 0, 0, 0, 0, 0, 0, 0x11),
121
+ exception_target_el(env));
122
+ /* noreturn, but fall through to the assert anyway */
123
+
124
+ case 0:
125
+ /*
126
+ * Tag check fail does not affect the PE.
127
+ * We eliminate this case by not setting MTE_ACTIVE
128
+ * in tb_flags, so that we never make this runtime call.
129
+ */
130
+ g_assert_not_reached();
131
+
132
+ case 2:
133
+ /* Tag check fail causes asynchronous flag set. */
134
+ mmu_idx = arm_mmu_idx_el(env, el);
135
+ if (regime_has_2_ranges(mmu_idx)) {
136
+ select = extract64(dirty_ptr, 55, 1);
137
+ } else {
138
+ select = 0;
139
+ }
140
+ env->cp15.tfsr_el[el] |= 1 << select;
141
+ break;
142
+
143
+ default:
144
+ /* Case 3: Reserved. */
145
+ qemu_log_mask(LOG_GUEST_ERROR,
146
+ "Tag check failure with SCTLR_EL%d.TCF%s "
147
+ "set to reserved value %d\n",
148
+ reg_el, el ? "" : "0", tcf);
149
+ break;
150
+ }
151
+}
152
+
153
/*
154
* Perform an MTE checked access for a single logical or atomic access.
155
*/
156
+static bool mte_probe1_int(CPUARMState *env, uint32_t desc, uint64_t ptr,
157
+ uintptr_t ra, int bit55)
158
+{
159
+ int mem_tag, mmu_idx, ptr_tag, size;
160
+ MMUAccessType type;
161
+ uint8_t *mem;
162
+
163
+ ptr_tag = allocation_tag_from_addr(ptr);
164
+
165
+ if (tcma_check(desc, bit55, ptr_tag)) {
166
+ return true;
167
+ }
168
+
169
+ mmu_idx = FIELD_EX32(desc, MTEDESC, MIDX);
170
+ type = FIELD_EX32(desc, MTEDESC, WRITE) ? MMU_DATA_STORE : MMU_DATA_LOAD;
171
+ size = FIELD_EX32(desc, MTEDESC, ESIZE);
172
+
173
+ mem = allocation_tag_mem(env, mmu_idx, ptr, type, size,
174
+ MMU_DATA_LOAD, 1, ra);
175
+ if (!mem) {
176
+ return true;
177
+ }
178
+
179
+ mem_tag = load_tag1(ptr, mem);
180
+ return ptr_tag == mem_tag;
181
+}
182
+
183
+/*
184
+ * No-fault version of mte_check1, to be used by SVE for MemSingleNF.
185
+ * Returns false if the access is Checked and the check failed. This
186
+ * is only intended to probe the tag -- the validity of the page must
187
+ * be checked beforehand.
188
+ */
189
+bool mte_probe1(CPUARMState *env, uint32_t desc, uint64_t ptr)
190
+{
191
+ int bit55 = extract64(ptr, 55, 1);
192
+
193
+ /* If TBI is disabled, the access is unchecked. */
194
+ if (unlikely(!tbi_check(desc, bit55))) {
195
+ return true;
196
+ }
197
+
198
+ return mte_probe1_int(env, desc, ptr, 0, bit55);
199
+}
200
+
201
+uint64_t mte_check1(CPUARMState *env, uint32_t desc,
202
+ uint64_t ptr, uintptr_t ra)
203
+{
204
+ int bit55 = extract64(ptr, 55, 1);
205
+
206
+ /* If TBI is disabled, the access is unchecked, and ptr is not dirty. */
207
+ if (unlikely(!tbi_check(desc, bit55))) {
208
+ return ptr;
209
+ }
210
+
211
+ if (unlikely(!mte_probe1_int(env, desc, ptr, ra, bit55))) {
212
+ int mmu_idx = FIELD_EX32(desc, MTEDESC, MIDX);
213
+ mte_check_fail(env, mmu_idx, ptr, ra);
214
+ }
215
+
216
+ return useronly_clean_ptr(ptr);
217
+}
218
+
219
uint64_t HELPER(mte_check1)(CPUARMState *env, uint32_t desc, uint64_t ptr)
220
{
221
- return ptr;
222
+ return mte_check1(env, desc, ptr, GETPC());
223
}
224
225
/*
31
--
226
--
32
2.18.0
227
2.20.1
33
228
34
229
diff view generated by jsdifflib
New patch
1
1
From: Richard Henderson <richard.henderson@linaro.org>
2
3
Fill out the stub that was added earlier.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-27-richard.henderson@linaro.org
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
---
10
target/arm/internals.h | 2 +
11
target/arm/mte_helper.c | 165 +++++++++++++++++++++++++++++++++++++++-
12
2 files changed, 166 insertions(+), 1 deletion(-)
13
14
diff --git a/target/arm/internals.h b/target/arm/internals.h
15
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/internals.h
17
+++ b/target/arm/internals.h
18
@@ -XXX,XX +XXX,XX @@ FIELD(MTEDESC, TSIZE, 14, 10) /* mte_checkN only */
19
bool mte_probe1(CPUARMState *env, uint32_t desc, uint64_t ptr);
20
uint64_t mte_check1(CPUARMState *env, uint32_t desc,
21
uint64_t ptr, uintptr_t ra);
22
+uint64_t mte_checkN(CPUARMState *env, uint32_t desc,
23
+ uint64_t ptr, uintptr_t ra);
24
25
static inline int allocation_tag_from_addr(uint64_t ptr)
26
{
27
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
28
index XXXXXXX..XXXXXXX 100644
29
--- a/target/arm/mte_helper.c
30
+++ b/target/arm/mte_helper.c
31
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(mte_check1)(CPUARMState *env, uint32_t desc, uint64_t ptr)
32
/*
33
* Perform an MTE checked access for multiple logical accesses.
34
*/
35
+
36
+/**
37
+ * checkN:
38
+ * @tag: tag memory to test
39
+ * @odd: true to begin testing at tags at odd nibble
40
+ * @cmp: the tag to compare against
41
+ * @count: number of tags to test
42
+ *
43
+ * Return the number of successful tests.
44
+ * Thus a return value < @count indicates a failure.
45
+ *
46
+ * A note about sizes: count is expected to be small.
47
+ *
48
+ * The most common use will be LDP/STP of two integer registers,
49
+ * which means 16 bytes of memory touching at most 2 tags, but
50
+ * often the access is aligned and thus just 1 tag.
51
+ *
52
+ * Using AdvSIMD LD/ST (multiple), one can access 64 bytes of memory,
53
+ * touching at most 5 tags. SVE LDR/STR (vector) with the default
54
+ * vector length is also 64 bytes; the maximum architectural length
55
+ * is 256 bytes touching at most 9 tags.
56
+ *
57
+ * The loop below uses 7 logical operations and 1 memory operation
58
+ * per tag pair. An implementation that loads an aligned word and
59
+ * uses masking to ignore adjacent tags requires 18 logical operations
60
+ * and thus does not begin to pay off until 6 tags.
61
+ * Which, according to the survey above, is unlikely to be common.
62
+ */
63
+static int checkN(uint8_t *mem, int odd, int cmp, int count)
64
+{
65
+ int n = 0, diff;
66
+
67
+ /* Replicate the test tag and compare. */
68
+ cmp *= 0x11;
69
+ diff = *mem++ ^ cmp;
70
+
71
+ if (odd) {
72
+ goto start_odd;
73
+ }
74
+
75
+ while (1) {
76
+ /* Test even tag. */
77
+ if (unlikely((diff) & 0x0f)) {
78
+ break;
79
+ }
80
+ if (++n == count) {
81
+ break;
82
+ }
83
+
84
+ start_odd:
85
+ /* Test odd tag. */
86
+ if (unlikely((diff) & 0xf0)) {
87
+ break;
88
+ }
89
+ if (++n == count) {
90
+ break;
91
+ }
92
+
93
+ diff = *mem++ ^ cmp;
94
+ }
95
+ return n;
96
+}
97
+
98
+uint64_t mte_checkN(CPUARMState *env, uint32_t desc,
99
+ uint64_t ptr, uintptr_t ra)
100
+{
101
+ int mmu_idx, ptr_tag, bit55;
102
+ uint64_t ptr_last, ptr_end, prev_page, next_page;
103
+ uint64_t tag_first, tag_end;
104
+ uint64_t tag_byte_first, tag_byte_end;
105
+ uint32_t esize, total, tag_count, tag_size, n, c;
106
+ uint8_t *mem1, *mem2;
107
+ MMUAccessType type;
108
+
109
+ bit55 = extract64(ptr, 55, 1);
110
+
111
+ /* If TBI is disabled, the access is unchecked, and ptr is not dirty. */
112
+ if (unlikely(!tbi_check(desc, bit55))) {
113
+ return ptr;
114
+ }
115
+
116
+ ptr_tag = allocation_tag_from_addr(ptr);
117
+
118
+ if (tcma_check(desc, bit55, ptr_tag)) {
119
+ goto done;
120
+ }
121
+
122
+ mmu_idx = FIELD_EX32(desc, MTEDESC, MIDX);
123
+ type = FIELD_EX32(desc, MTEDESC, WRITE) ? MMU_DATA_STORE : MMU_DATA_LOAD;
124
+ esize = FIELD_EX32(desc, MTEDESC, ESIZE);
125
+ total = FIELD_EX32(desc, MTEDESC, TSIZE);
126
+
127
+ /* Find the addr of the end of the access, and of the last element. */
128
+ ptr_end = ptr + total;
129
+ ptr_last = ptr_end - esize;
130
+
131
+ /* Round the bounds to the tag granule, and compute the number of tags. */
132
+ tag_first = QEMU_ALIGN_DOWN(ptr, TAG_GRANULE);
133
+ tag_end = QEMU_ALIGN_UP(ptr_last, TAG_GRANULE);
134
+ tag_count = (tag_end - tag_first) / TAG_GRANULE;
135
+
136
+ /* Round the bounds to twice the tag granule, and compute the bytes. */
137
+ tag_byte_first = QEMU_ALIGN_DOWN(ptr, 2 * TAG_GRANULE);
138
+ tag_byte_end = QEMU_ALIGN_UP(ptr_last, 2 * TAG_GRANULE);
139
+
140
+ /* Locate the page boundaries. */
141
+ prev_page = ptr & TARGET_PAGE_MASK;
142
+ next_page = prev_page + TARGET_PAGE_SIZE;
143
+
144
+ if (likely(tag_end - prev_page <= TARGET_PAGE_SIZE)) {
145
+ /* Memory access stays on one page. */
146
+ tag_size = (tag_byte_end - tag_byte_first) / (2 * TAG_GRANULE);
147
+ mem1 = allocation_tag_mem(env, mmu_idx, ptr, type, total,
148
+ MMU_DATA_LOAD, tag_size, ra);
149
+ if (!mem1) {
150
+ goto done;
151
+ }
152
+ /* Perform all of the comparisons. */
153
+ n = checkN(mem1, ptr & TAG_GRANULE, ptr_tag, tag_count);
154
+ } else {
155
+ /* Memory access crosses to next page. */
156
+ tag_size = (next_page - tag_byte_first) / (2 * TAG_GRANULE);
157
+ mem1 = allocation_tag_mem(env, mmu_idx, ptr, type, next_page - ptr,
158
+ MMU_DATA_LOAD, tag_size, ra);
159
+
160
+ tag_size = (tag_byte_end - next_page) / (2 * TAG_GRANULE);
161
+ mem2 = allocation_tag_mem(env, mmu_idx, next_page, type,
162
+ ptr_end - next_page,
163
+ MMU_DATA_LOAD, tag_size, ra);
164
+
165
+ /*
166
+ * Perform all of the comparisons.
167
+ * Note the possible but unlikely case of the operation spanning
168
+ * two pages that do not both have tagging enabled.
169
+ */
170
+ n = c = (next_page - tag_first) / TAG_GRANULE;
171
+ if (mem1) {
172
+ n = checkN(mem1, ptr & TAG_GRANULE, ptr_tag, c);
173
+ }
174
+ if (n == c) {
175
+ if (!mem2) {
176
+ goto done;
177
+ }
178
+ n += checkN(mem2, 0, ptr_tag, tag_count - c);
179
+ }
180
+ }
181
+
182
+ /*
183
+ * If we failed, we know which granule. Compute the element that
184
+ * is first in that granule, and signal failure on that element.
185
+ */
186
+ if (unlikely(n < tag_count)) {
187
+ uint64_t fail_ofs;
188
+
189
+ fail_ofs = tag_first + n * TAG_GRANULE - ptr;
190
+ fail_ofs = ROUND_UP(fail_ofs, esize);
191
+ mte_check_fail(env, mmu_idx, ptr + fail_ofs, ra);
192
+ }
193
+
194
+ done:
195
+ return useronly_clean_ptr(ptr);
196
+}
197
+
198
uint64_t HELPER(mte_checkN)(CPUARMState *env, uint32_t desc, uint64_t ptr)
199
{
200
- return ptr;
201
+ return mte_checkN(env, desc, ptr, GETPC());
202
}
203
--
204
2.20.1
205
206
diff view generated by jsdifflib
1
Implement a model of the TrustZone Master Securtiy Controller,
1
From: Richard Henderson <richard.henderson@linaro.org>
2
as documented in the Arm CoreLink SIE-200 System IP for
3
Embedded TRM (DDI0571G):
4
https://developer.arm.com/products/architecture/m-profile/docs/ddi0571/g
5
2
6
The MSC is intended to sit in front of a device which can
3
Use a special helper for DC_ZVA, rather than the more
7
be a bus master (eg a DMA controller) and programmably gate
4
general mte_checkN.
8
its transactions. This allows a bus-mastering device to be
9
controlled by non-secure code but still restricted from
10
making accesses to addresses which are secure-only.
11
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-28-richard.henderson@linaro.org
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Message-id: 20180820141116.9118-12-peter.maydell@linaro.org
14
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
15
---
10
---
16
hw/misc/Makefile.objs | 1 +
11
target/arm/helper-a64.h | 1 +
17
include/hw/misc/tz-msc.h | 79 ++++++++
12
target/arm/mte_helper.c | 106 +++++++++++++++++++++++++++++++++++++
18
hw/misc/tz-msc.c | 308 ++++++++++++++++++++++++++++++++
13
target/arm/translate-a64.c | 16 +++++-
19
MAINTAINERS | 2 +
14
3 files changed, 122 insertions(+), 1 deletion(-)
20
default-configs/arm-softmmu.mak | 1 +
21
hw/misc/trace-events | 9 +
22
6 files changed, 400 insertions(+)
23
create mode 100644 include/hw/misc/tz-msc.h
24
create mode 100644 hw/misc/tz-msc.c
25
15
26
diff --git a/hw/misc/Makefile.objs b/hw/misc/Makefile.objs
16
diff --git a/target/arm/helper-a64.h b/target/arm/helper-a64.h
27
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
28
--- a/hw/misc/Makefile.objs
18
--- a/target/arm/helper-a64.h
29
+++ b/hw/misc/Makefile.objs
19
+++ b/target/arm/helper-a64.h
30
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_MPS2_FPGAIO) += mps2-fpgaio.o
20
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(xpacd, TCG_CALL_NO_RWG_SE, i64, env, i64)
31
obj-$(CONFIG_MPS2_SCC) += mps2-scc.o
21
32
22
DEF_HELPER_FLAGS_3(mte_check1, TCG_CALL_NO_WG, i64, env, i32, i64)
33
obj-$(CONFIG_TZ_MPC) += tz-mpc.o
23
DEF_HELPER_FLAGS_3(mte_checkN, TCG_CALL_NO_WG, i64, env, i32, i64)
34
+obj-$(CONFIG_TZ_MSC) += tz-msc.o
24
+DEF_HELPER_FLAGS_3(mte_check_zva, TCG_CALL_NO_WG, i64, env, i32, i64)
35
obj-$(CONFIG_TZ_PPC) += tz-ppc.o
25
DEF_HELPER_FLAGS_3(irg, TCG_CALL_NO_RWG, i64, env, i64, i64)
36
obj-$(CONFIG_IOTKIT_SECCTL) += iotkit-secctl.o
26
DEF_HELPER_FLAGS_4(addsubg, TCG_CALL_NO_RWG_SE, i64, env, i64, s32, i32)
37
obj-$(CONFIG_IOTKIT_SYSCTL) += iotkit-sysctl.o
27
DEF_HELPER_FLAGS_3(ldg, TCG_CALL_NO_WG, i64, env, i64, i64)
38
diff --git a/include/hw/misc/tz-msc.h b/include/hw/misc/tz-msc.h
28
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
39
new file mode 100644
29
index XXXXXXX..XXXXXXX 100644
40
index XXXXXXX..XXXXXXX
30
--- a/target/arm/mte_helper.c
41
--- /dev/null
31
+++ b/target/arm/mte_helper.c
42
+++ b/include/hw/misc/tz-msc.h
32
@@ -XXX,XX +XXX,XX @@ uint64_t HELPER(mte_checkN)(CPUARMState *env, uint32_t desc, uint64_t ptr)
43
@@ -XXX,XX +XXX,XX @@
33
{
44
+/*
34
return mte_checkN(env, desc, ptr, GETPC());
45
+ * ARM TrustZone master security controller emulation
35
}
46
+ *
47
+ * Copyright (c) 2018 Linaro Limited
48
+ * Written by Peter Maydell
49
+ *
50
+ * This program is free software; you can redistribute it and/or modify
51
+ * it under the terms of the GNU General Public License version 2 or
52
+ * (at your option) any later version.
53
+ */
54
+
36
+
55
+/*
37
+/*
56
+ * This is a model of the TrustZone master security controller (MSC).
38
+ * Perform an MTE checked access for DC_ZVA.
57
+ * It is documented in the ARM CoreLink SIE-200 System IP for Embedded TRM
58
+ * (DDI 0571G):
59
+ * https://developer.arm.com/products/architecture/m-profile/docs/ddi0571/g
60
+ *
61
+ * The MSC sits in front of a device which can be a bus master (such as
62
+ * a DMA controller) and allows secure software to configure it to either
63
+ * pass through or reject transactions made by that bus master.
64
+ * Rejected transactions may be configured to either be aborted, or to
65
+ * behave as RAZ/WI. An interrupt can be signalled for a rejected transaction.
66
+ *
67
+ * The MSC has no register interface -- it is configured purely by a
68
+ * collection of input signals from other hardware in the system. Typically
69
+ * they are either hardwired or exposed in an ad-hoc register interface by
70
+ * the SoC that uses the MSC.
71
+ *
72
+ * We don't currently implement the irq_enable GPIO input, because on
73
+ * the MPS2 FPGA images it is always tied high, which is awkward to
74
+ * implement in QEMU.
75
+ *
76
+ * QEMU interface:
77
+ * + Named GPIO input "cfg_nonsec": set to 1 if the bus master should be
78
+ * treated as nonsecure, or 0 for secure
79
+ * + Named GPIO input "cfg_sec_resp": set to 1 if a rejected transaction should
80
+ * result in a transaction error, or 0 for the transaction to RAZ/WI
81
+ * + Named GPIO input "irq_clear": set to 1 to clear a pending interrupt
82
+ * + Named GPIO output "irq": set for a transaction-failed interrupt
83
+ * + Property "downstream": MemoryRegion defining where bus master transactions
84
+ * are made if they are not blocked
85
+ * + Property "idau": an object implementing IDAUInterface, which defines which
86
+ * addresses should be treated as secure and which as non-secure.
87
+ * This need not be the same IDAU as the one used by the CPU.
88
+ * + sysbus MMIO region 0: MemoryRegion defining the upstream end of the MSC;
89
+ * this should be passed to the bus master device as the region it should
90
+ * make memory transactions to
91
+ */
39
+ */
40
+uint64_t HELPER(mte_check_zva)(CPUARMState *env, uint32_t desc, uint64_t ptr)
41
+{
42
+ uintptr_t ra = GETPC();
43
+ int log2_dcz_bytes, log2_tag_bytes;
44
+ int mmu_idx, bit55;
45
+ intptr_t dcz_bytes, tag_bytes, i;
46
+ void *mem;
47
+ uint64_t ptr_tag, mem_tag, align_ptr;
92
+
48
+
93
+#ifndef TZ_MSC_H
49
+ bit55 = extract64(ptr, 55, 1);
94
+#define TZ_MSC_H
95
+
50
+
96
+#include "hw/sysbus.h"
51
+ /* If TBI is disabled, the access is unchecked, and ptr is not dirty. */
97
+#include "target/arm/idau.h"
52
+ if (unlikely(!tbi_check(desc, bit55))) {
98
+
53
+ return ptr;
99
+#define TYPE_TZ_MSC "tz-msc"
100
+#define TZ_MSC(obj) OBJECT_CHECK(TZMSC, (obj), TYPE_TZ_MSC)
101
+
102
+typedef struct TZMSC {
103
+ /*< private >*/
104
+ SysBusDevice parent_obj;
105
+
106
+ /*< public >*/
107
+
108
+ /* State: these just track the values of our input signals */
109
+ bool cfg_nonsec;
110
+ bool cfg_sec_resp;
111
+ bool irq_clear;
112
+ /* State: are we asserting irq ? */
113
+ bool irq_status;
114
+
115
+ qemu_irq irq;
116
+ MemoryRegion *downstream;
117
+ AddressSpace downstream_as;
118
+ MemoryRegion upstream;
119
+ IDAUInterface *idau;
120
+} TZMSC;
121
+
122
+#endif
123
diff --git a/hw/misc/tz-msc.c b/hw/misc/tz-msc.c
124
new file mode 100644
125
index XXXXXXX..XXXXXXX
126
--- /dev/null
127
+++ b/hw/misc/tz-msc.c
128
@@ -XXX,XX +XXX,XX @@
129
+/*
130
+ * ARM TrustZone master security controller emulation
131
+ *
132
+ * Copyright (c) 2018 Linaro Limited
133
+ * Written by Peter Maydell
134
+ *
135
+ * This program is free software; you can redistribute it and/or modify
136
+ * it under the terms of the GNU General Public License version 2 or
137
+ * (at your option) any later version.
138
+ */
139
+
140
+#include "qemu/osdep.h"
141
+#include "qemu/log.h"
142
+#include "qapi/error.h"
143
+#include "trace.h"
144
+#include "hw/sysbus.h"
145
+#include "hw/registerfields.h"
146
+#include "hw/misc/tz-msc.h"
147
+
148
+static void tz_msc_update_irq(TZMSC *s)
149
+{
150
+ bool level = s->irq_status;
151
+
152
+ trace_tz_msc_update_irq(level);
153
+ qemu_set_irq(s->irq, level);
154
+}
155
+
156
+static void tz_msc_cfg_nonsec(void *opaque, int n, int level)
157
+{
158
+ TZMSC *s = TZ_MSC(opaque);
159
+
160
+ trace_tz_msc_cfg_nonsec(level);
161
+ s->cfg_nonsec = level;
162
+}
163
+
164
+static void tz_msc_cfg_sec_resp(void *opaque, int n, int level)
165
+{
166
+ TZMSC *s = TZ_MSC(opaque);
167
+
168
+ trace_tz_msc_cfg_sec_resp(level);
169
+ s->cfg_sec_resp = level;
170
+}
171
+
172
+static void tz_msc_irq_clear(void *opaque, int n, int level)
173
+{
174
+ TZMSC *s = TZ_MSC(opaque);
175
+
176
+ trace_tz_msc_irq_clear(level);
177
+
178
+ s->irq_clear = level;
179
+ if (level) {
180
+ s->irq_status = false;
181
+ tz_msc_update_irq(s);
182
+ }
183
+}
184
+
185
+/* The MSC may either block a transaction by aborting it, block a
186
+ * transaction by making it RAZ/WI, allow it through with
187
+ * MemTxAttrs indicating a secure transaction, or allow it with
188
+ * MemTxAttrs indicating a non-secure transaction.
189
+ */
190
+typedef enum MSCAction {
191
+ MSCBlockAbort,
192
+ MSCBlockRAZWI,
193
+ MSCAllowSecure,
194
+ MSCAllowNonSecure,
195
+} MSCAction;
196
+
197
+static MSCAction tz_msc_check(TZMSC *s, hwaddr addr)
198
+{
199
+ /*
200
+ * Check whether to allow an access from the bus master, returning
201
+ * an MSCAction indicating the required behaviour. If the transaction
202
+ * is blocked, the caller must check cfg_sec_resp to determine
203
+ * whether to abort or RAZ/WI the transaction.
204
+ */
205
+ IDAUInterfaceClass *iic = IDAU_INTERFACE_GET_CLASS(s->idau);
206
+ IDAUInterface *ii = IDAU_INTERFACE(s->idau);
207
+ bool idau_exempt = false, idau_ns = true, idau_nsc = true;
208
+ int idau_region = IREGION_NOTVALID;
209
+
210
+ iic->check(ii, addr, &idau_region, &idau_exempt, &idau_ns, &idau_nsc);
211
+
212
+ if (idau_exempt) {
213
+ /*
214
+ * Uncheck region -- OK, transaction type depends on
215
+ * whether bus master is configured as Secure or NonSecure
216
+ */
217
+ return s->cfg_nonsec ? MSCAllowNonSecure : MSCAllowSecure;
218
+ }
54
+ }
219
+
55
+
220
+ if (idau_ns) {
56
+ ptr_tag = allocation_tag_from_addr(ptr);
221
+ /* NonSecure region -- always forward as NS transaction */
222
+ return MSCAllowNonSecure;
223
+ }
224
+
57
+
225
+ if (!s->cfg_nonsec) {
58
+ if (tcma_check(desc, bit55, ptr_tag)) {
226
+ /* Access to Secure region by Secure bus master: OK */
59
+ goto done;
227
+ return MSCAllowSecure;
228
+ }
229
+
230
+ /* Attempted access to Secure region by NS bus master: block */
231
+ trace_tz_msc_access_blocked(addr);
232
+ if (!s->cfg_sec_resp) {
233
+ return MSCBlockRAZWI;
234
+ }
60
+ }
235
+
61
+
236
+ /*
62
+ /*
237
+ * The TRM isn't clear on behaviour if irq_clear is high when a
63
+ * In arm_cpu_realizefn, we asserted that dcz > LOG2_TAG_GRANULE+1,
238
+ * transaction is blocked. We assume that the MSC behaves like the
64
+ * i.e. 32 bytes, which is an unreasonably small dcz anyway, to make
239
+ * PPC, where holding irq_clear high suppresses the interrupt.
65
+ * sure that we can access one complete tag byte here.
240
+ */
66
+ */
241
+ if (!s->irq_clear) {
67
+ log2_dcz_bytes = env_archcpu(env)->dcz_blocksize + 2;
242
+ s->irq_status = true;
68
+ log2_tag_bytes = log2_dcz_bytes - (LOG2_TAG_GRANULE + 1);
243
+ tz_msc_update_irq(s);
69
+ dcz_bytes = (intptr_t)1 << log2_dcz_bytes;
244
+ }
70
+ tag_bytes = (intptr_t)1 << log2_tag_bytes;
245
+ return MSCBlockAbort;
71
+ align_ptr = ptr & -dcz_bytes;
246
+}
247
+
72
+
248
+static MemTxResult tz_msc_read(void *opaque, hwaddr addr, uint64_t *pdata,
73
+ /*
249
+ unsigned size, MemTxAttrs attrs)
74
+ * Trap if accessing an invalid page. DC_ZVA requires that we supply
250
+{
75
+ * the original pointer for an invalid page. But watchpoints require
251
+ TZMSC *s = opaque;
76
+ * that we probe the actual space. So do both.
252
+ AddressSpace *as = &s->downstream_as;
77
+ */
253
+ uint64_t data;
78
+ mmu_idx = FIELD_EX32(desc, MTEDESC, MIDX);
254
+ MemTxResult res;
79
+ (void) probe_write(env, ptr, 1, mmu_idx, ra);
255
+
80
+ mem = allocation_tag_mem(env, mmu_idx, align_ptr, MMU_DATA_STORE,
256
+ switch (tz_msc_check(s, addr)) {
81
+ dcz_bytes, MMU_DATA_LOAD, tag_bytes, ra);
257
+ case MSCBlockAbort:
82
+ if (!mem) {
258
+ return MEMTX_ERROR;
83
+ goto done;
259
+ case MSCBlockRAZWI:
260
+ *pdata = 0;
261
+ return MEMTX_OK;
262
+ case MSCAllowSecure:
263
+ attrs.secure = 1;
264
+ attrs.unspecified = 0;
265
+ break;
266
+ case MSCAllowNonSecure:
267
+ attrs.secure = 0;
268
+ attrs.unspecified = 0;
269
+ break;
270
+ }
84
+ }
271
+
85
+
272
+ switch (size) {
86
+ /*
273
+ case 1:
87
+ * Unlike the reasoning for checkN, DC_ZVA is always aligned, and thus
274
+ data = address_space_ldub(as, addr, attrs, &res);
88
+ * it is quite easy to perform all of the comparisons at once without
89
+ * any extra masking.
90
+ *
91
+ * The most common zva block size is 64; some of the thunderx cpus use
92
+ * a block size of 128. For user-only, aarch64_max_initfn will set the
93
+ * block size to 512. Fill out the other cases for future-proofing.
94
+ *
95
+ * In order to be able to find the first miscompare later, we want the
96
+ * tag bytes to be in little-endian order.
97
+ */
98
+ switch (log2_tag_bytes) {
99
+ case 0: /* zva_blocksize 32 */
100
+ mem_tag = *(uint8_t *)mem;
101
+ ptr_tag *= 0x11u;
275
+ break;
102
+ break;
276
+ case 2:
103
+ case 1: /* zva_blocksize 64 */
277
+ data = address_space_lduw_le(as, addr, attrs, &res);
104
+ mem_tag = cpu_to_le16(*(uint16_t *)mem);
105
+ ptr_tag *= 0x1111u;
278
+ break;
106
+ break;
279
+ case 4:
107
+ case 2: /* zva_blocksize 128 */
280
+ data = address_space_ldl_le(as, addr, attrs, &res);
108
+ mem_tag = cpu_to_le32(*(uint32_t *)mem);
109
+ ptr_tag *= 0x11111111u;
281
+ break;
110
+ break;
282
+ case 8:
111
+ case 3: /* zva_blocksize 256 */
283
+ data = address_space_ldq_le(as, addr, attrs, &res);
112
+ mem_tag = cpu_to_le64(*(uint64_t *)mem);
113
+ ptr_tag *= 0x1111111111111111ull;
284
+ break;
114
+ break;
285
+ default:
286
+ g_assert_not_reached();
287
+ }
288
+ *pdata = data;
289
+ return res;
290
+}
291
+
115
+
292
+static MemTxResult tz_msc_write(void *opaque, hwaddr addr, uint64_t val,
116
+ default: /* zva_blocksize 512, 1024, 2048 */
293
+ unsigned size, MemTxAttrs attrs)
117
+ ptr_tag *= 0x1111111111111111ull;
294
+{
118
+ i = 0;
295
+ TZMSC *s = opaque;
119
+ do {
296
+ AddressSpace *as = &s->downstream_as;
120
+ mem_tag = cpu_to_le64(*(uint64_t *)(mem + i));
297
+ MemTxResult res;
121
+ if (unlikely(mem_tag != ptr_tag)) {
298
+
122
+ goto fail;
299
+ switch (tz_msc_check(s, addr)) {
123
+ }
300
+ case MSCBlockAbort:
124
+ i += 8;
301
+ return MEMTX_ERROR;
125
+ align_ptr += 16 * TAG_GRANULE;
302
+ case MSCBlockRAZWI:
126
+ } while (i < tag_bytes);
303
+ return MEMTX_OK;
127
+ goto done;
304
+ case MSCAllowSecure:
305
+ attrs.secure = 1;
306
+ attrs.unspecified = 0;
307
+ break;
308
+ case MSCAllowNonSecure:
309
+ attrs.secure = 0;
310
+ attrs.unspecified = 0;
311
+ break;
312
+ }
128
+ }
313
+
129
+
314
+ switch (size) {
130
+ if (likely(mem_tag == ptr_tag)) {
315
+ case 1:
131
+ goto done;
316
+ address_space_stb(as, addr, val, attrs, &res);
317
+ break;
318
+ case 2:
319
+ address_space_stw_le(as, addr, val, attrs, &res);
320
+ break;
321
+ case 4:
322
+ address_space_stl_le(as, addr, val, attrs, &res);
323
+ break;
324
+ case 8:
325
+ address_space_stq_le(as, addr, val, attrs, &res);
326
+ break;
327
+ default:
328
+ g_assert_not_reached();
329
+ }
330
+ return res;
331
+}
332
+
333
+static const MemoryRegionOps tz_msc_ops = {
334
+ .read_with_attrs = tz_msc_read,
335
+ .write_with_attrs = tz_msc_write,
336
+ .endianness = DEVICE_LITTLE_ENDIAN,
337
+};
338
+
339
+static void tz_msc_reset(DeviceState *dev)
340
+{
341
+ TZMSC *s = TZ_MSC(dev);
342
+
343
+ trace_tz_msc_reset();
344
+ s->cfg_sec_resp = false;
345
+ s->cfg_nonsec = false;
346
+ s->irq_clear = 0;
347
+ s->irq_status = 0;
348
+}
349
+
350
+static void tz_msc_init(Object *obj)
351
+{
352
+ DeviceState *dev = DEVICE(obj);
353
+ TZMSC *s = TZ_MSC(obj);
354
+
355
+ qdev_init_gpio_in_named(dev, tz_msc_cfg_nonsec, "cfg_nonsec", 1);
356
+ qdev_init_gpio_in_named(dev, tz_msc_cfg_sec_resp, "cfg_sec_resp", 1);
357
+ qdev_init_gpio_in_named(dev, tz_msc_irq_clear, "irq_clear", 1);
358
+ qdev_init_gpio_out_named(dev, &s->irq, "irq", 1);
359
+}
360
+
361
+static void tz_msc_realize(DeviceState *dev, Error **errp)
362
+{
363
+ Object *obj = OBJECT(dev);
364
+ SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
365
+ TZMSC *s = TZ_MSC(dev);
366
+ const char *name = "tz-msc-downstream";
367
+ uint64_t size;
368
+
369
+ /*
370
+ * We can't create the upstream end of the port until realize,
371
+ * as we don't know the size of the MR used as the downstream until then.
372
+ * We insist on having a downstream, to avoid complicating the
373
+ * code with handling the "don't know how big this is" case. It's easy
374
+ * enough for the user to create an unimplemented_device as downstream
375
+ * if they have nothing else to plug into this.
376
+ */
377
+ if (!s->downstream) {
378
+ error_setg(errp, "MSC 'downstream' link not set");
379
+ return;
380
+ }
381
+ if (!s->idau) {
382
+ error_setg(errp, "MSC 'idau' link not set");
383
+ return;
384
+ }
132
+ }
385
+
133
+
386
+ size = memory_region_size(s->downstream);
134
+ fail:
387
+ address_space_init(&s->downstream_as, s->downstream, name);
135
+ /* Locate the first nibble that differs. */
388
+ memory_region_init_io(&s->upstream, obj, &tz_msc_ops, s, name, size);
136
+ i = ctz64(mem_tag ^ ptr_tag) >> 4;
389
+ sysbus_init_mmio(sbd, &s->upstream);
137
+ mte_check_fail(env, mmu_idx, align_ptr + i * TAG_GRANULE, ra);
138
+
139
+ done:
140
+ return useronly_clean_ptr(ptr);
390
+}
141
+}
142
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
143
index XXXXXXX..XXXXXXX 100644
144
--- a/target/arm/translate-a64.c
145
+++ b/target/arm/translate-a64.c
146
@@ -XXX,XX +XXX,XX @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
147
return;
148
case ARM_CP_DC_ZVA:
149
/* Writes clear the aligned block of memory which rt points into. */
150
- tcg_rt = clean_data_tbi(s, cpu_reg(s, rt));
151
+ if (s->mte_active[0]) {
152
+ TCGv_i32 t_desc;
153
+ int desc = 0;
391
+
154
+
392
+static const VMStateDescription tz_msc_vmstate = {
155
+ desc = FIELD_DP32(desc, MTEDESC, MIDX, get_mem_index(s));
393
+ .name = "tz-msc",
156
+ desc = FIELD_DP32(desc, MTEDESC, TBI, s->tbid);
394
+ .version_id = 1,
157
+ desc = FIELD_DP32(desc, MTEDESC, TCMA, s->tcma);
395
+ .minimum_version_id = 1,
158
+ t_desc = tcg_const_i32(desc);
396
+ .fields = (VMStateField[]) {
397
+ VMSTATE_BOOL(cfg_nonsec, TZMSC),
398
+ VMSTATE_BOOL(cfg_sec_resp, TZMSC),
399
+ VMSTATE_BOOL(irq_clear, TZMSC),
400
+ VMSTATE_BOOL(irq_status, TZMSC),
401
+ VMSTATE_END_OF_LIST()
402
+ }
403
+};
404
+
159
+
405
+static Property tz_msc_properties[] = {
160
+ tcg_rt = new_tmp_a64(s);
406
+ DEFINE_PROP_LINK("downstream", TZMSC, downstream,
161
+ gen_helper_mte_check_zva(tcg_rt, cpu_env, t_desc, cpu_reg(s, rt));
407
+ TYPE_MEMORY_REGION, MemoryRegion *),
162
+ tcg_temp_free_i32(t_desc);
408
+ DEFINE_PROP_LINK("idau", TZMSC, idau,
163
+ } else {
409
+ TYPE_IDAU_INTERFACE, IDAUInterface *),
164
+ tcg_rt = clean_data_tbi(s, cpu_reg(s, rt));
410
+ DEFINE_PROP_END_OF_LIST(),
165
+ }
411
+};
166
gen_helper_dc_zva(cpu_env, tcg_rt);
412
+
167
return;
413
+static void tz_msc_class_init(ObjectClass *klass, void *data)
168
default:
414
+{
415
+ DeviceClass *dc = DEVICE_CLASS(klass);
416
+
417
+ dc->realize = tz_msc_realize;
418
+ dc->vmsd = &tz_msc_vmstate;
419
+ dc->reset = tz_msc_reset;
420
+ dc->props = tz_msc_properties;
421
+}
422
+
423
+static const TypeInfo tz_msc_info = {
424
+ .name = TYPE_TZ_MSC,
425
+ .parent = TYPE_SYS_BUS_DEVICE,
426
+ .instance_size = sizeof(TZMSC),
427
+ .instance_init = tz_msc_init,
428
+ .class_init = tz_msc_class_init,
429
+};
430
+
431
+static void tz_msc_register_types(void)
432
+{
433
+ type_register_static(&tz_msc_info);
434
+}
435
+
436
+type_init(tz_msc_register_types);
437
diff --git a/MAINTAINERS b/MAINTAINERS
438
index XXXXXXX..XXXXXXX 100644
439
--- a/MAINTAINERS
440
+++ b/MAINTAINERS
441
@@ -XXX,XX +XXX,XX @@ F: hw/misc/tz-ppc.c
442
F: include/hw/misc/tz-ppc.h
443
F: hw/misc/tz-mpc.c
444
F: include/hw/misc/tz-mpc.h
445
+F: hw/misc/tz-msc.c
446
+F: include/hw/misc/tz-msc.h
447
448
ARM cores
449
M: Peter Maydell <peter.maydell@linaro.org>
450
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
451
index XXXXXXX..XXXXXXX 100644
452
--- a/default-configs/arm-softmmu.mak
453
+++ b/default-configs/arm-softmmu.mak
454
@@ -XXX,XX +XXX,XX @@ CONFIG_MPS2_FPGAIO=y
455
CONFIG_MPS2_SCC=y
456
457
CONFIG_TZ_MPC=y
458
+CONFIG_TZ_MSC=y
459
CONFIG_TZ_PPC=y
460
CONFIG_IOTKIT=y
461
CONFIG_IOTKIT_SECCTL=y
462
diff --git a/hw/misc/trace-events b/hw/misc/trace-events
463
index XXXXXXX..XXXXXXX 100644
464
--- a/hw/misc/trace-events
465
+++ b/hw/misc/trace-events
466
@@ -XXX,XX +XXX,XX @@ tz_mpc_mem_blocked_write(uint64_t addr, uint64_t data, unsigned size, bool secur
467
tz_mpc_translate(uint64_t addr, int flags, const char *idx, const char *res) "TZ MPC translate: addr 0x%" PRIx64 " flags 0x%x iommu_idx %s: %s"
468
tz_mpc_iommu_notify(uint64_t addr) "TZ MPC iommu: notifying UNMAP/MAP for 0x%" PRIx64
469
470
+# hw/misc/tz-msc.c
471
+tz_msc_reset(void) "TZ MSC: reset"
472
+tz_msc_cfg_nonsec(int level) "TZ MSC: cfg_nonsec = %d"
473
+tz_msc_cfg_sec_resp(int level) "TZ MSC: cfg_sec_resp = %d"
474
+tz_msc_irq_enable(int level) "TZ MSC: int_enable = %d"
475
+tz_msc_irq_clear(int level) "TZ MSC: int_clear = %d"
476
+tz_msc_update_irq(int level) "TZ MSC: setting irq line to %d"
477
+tz_msc_access_blocked(uint64_t offset) "TZ MSC: offset 0x%" PRIx64 " access blocked"
478
+
479
# hw/misc/tz-ppc.c
480
tz_ppc_reset(void) "TZ PPC: reset"
481
tz_ppc_cfg_nonsec(int n, int level) "TZ PPC: cfg_nonsec[%d] = %d"
482
--
169
--
483
2.18.0
170
2.20.1
484
171
485
172
diff view generated by jsdifflib
1
Connect the VIRQ and VFIQ lines from the GIC to the CPU;
1
From: Richard Henderson <richard.henderson@linaro.org>
2
these exist always for both CPU and GIC whether the
3
virtualization extensions are enabled or not, so we
4
can just unconditionally connect them.
5
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-29-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180821132811.17675-5-peter.maydell@linaro.org
9
---
7
---
10
hw/arm/highbank.c | 6 ++++++
8
target/arm/translate-sve.c | 61 +++++++++++++++++++++-----------------
11
1 file changed, 6 insertions(+)
9
1 file changed, 33 insertions(+), 28 deletions(-)
12
10
13
diff --git a/hw/arm/highbank.c b/hw/arm/highbank.c
11
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
14
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
15
--- a/hw/arm/highbank.c
13
--- a/target/arm/translate-sve.c
16
+++ b/hw/arm/highbank.c
14
+++ b/target/arm/translate-sve.c
17
@@ -XXX,XX +XXX,XX @@ static void calxeda_init(MachineState *machine, enum cxmachines machine_id)
15
@@ -XXX,XX +XXX,XX @@ static void do_ldr(DisasContext *s, uint32_t vofs, int len, int rn, int imm)
18
int n;
16
int len_remain = len % 8;
19
qemu_irq cpu_irq[4];
17
int nparts = len / 8 + ctpop8(len_remain);
20
qemu_irq cpu_fiq[4];
18
int midx = get_mem_index(s);
21
+ qemu_irq cpu_virq[4];
19
- TCGv_i64 addr, t0, t1;
22
+ qemu_irq cpu_vfiq[4];
20
+ TCGv_i64 dirty_addr, clean_addr, t0, t1;
23
MemoryRegion *sysram;
21
24
MemoryRegion *dram;
22
- addr = tcg_temp_new_i64();
25
MemoryRegion *sysmem;
23
- t0 = tcg_temp_new_i64();
26
@@ -XXX,XX +XXX,XX @@ static void calxeda_init(MachineState *machine, enum cxmachines machine_id)
24
+ dirty_addr = tcg_temp_new_i64();
27
object_property_set_bool(cpuobj, true, "realized", &error_fatal);
25
+ tcg_gen_addi_i64(dirty_addr, cpu_reg_sp(s, rn), imm);
28
cpu_irq[n] = qdev_get_gpio_in(DEVICE(cpu), ARM_CPU_IRQ);
26
+ clean_addr = gen_mte_checkN(s, dirty_addr, false, rn != 31, len, MO_8);
29
cpu_fiq[n] = qdev_get_gpio_in(DEVICE(cpu), ARM_CPU_FIQ);
27
+ tcg_temp_free_i64(dirty_addr);
30
+ cpu_virq[n] = qdev_get_gpio_in(DEVICE(cpu), ARM_CPU_VIRQ);
28
31
+ cpu_vfiq[n] = qdev_get_gpio_in(DEVICE(cpu), ARM_CPU_VFIQ);
29
- /* Note that unpredicated load/store of vector/predicate registers
30
+ /*
31
+ * Note that unpredicated load/store of vector/predicate registers
32
* are defined as a stream of bytes, which equates to little-endian
33
- * operations on larger quantities. There is no nice way to force
34
- * a little-endian load for aarch64_be-linux-user out of line.
35
- *
36
+ * operations on larger quantities.
37
* Attempt to keep code expansion to a minimum by limiting the
38
* amount of unrolling done.
39
*/
40
if (nparts <= 4) {
41
int i;
42
43
+ t0 = tcg_temp_new_i64();
44
for (i = 0; i < len_align; i += 8) {
45
- tcg_gen_addi_i64(addr, cpu_reg_sp(s, rn), imm + i);
46
- tcg_gen_qemu_ld_i64(t0, addr, midx, MO_LEQ);
47
+ tcg_gen_qemu_ld_i64(t0, clean_addr, midx, MO_LEQ);
48
tcg_gen_st_i64(t0, cpu_env, vofs + i);
49
+ tcg_gen_addi_i64(clean_addr, cpu_reg_sp(s, rn), 8);
50
}
51
+ tcg_temp_free_i64(t0);
52
} else {
53
TCGLabel *loop = gen_new_label();
54
TCGv_ptr tp, i = tcg_const_local_ptr(0);
55
56
+ /* Copy the clean address into a local temp, live across the loop. */
57
+ t0 = clean_addr;
58
+ clean_addr = tcg_temp_local_new_i64();
59
+ tcg_gen_mov_i64(clean_addr, t0);
60
+ tcg_temp_free_i64(t0);
61
+
62
gen_set_label(loop);
63
64
- /* Minimize the number of local temps that must be re-read from
65
- * the stack each iteration. Instead, re-compute values other
66
- * than the loop counter.
67
- */
68
+ t0 = tcg_temp_new_i64();
69
+ tcg_gen_qemu_ld_i64(t0, clean_addr, midx, MO_LEQ);
70
+ tcg_gen_addi_i64(clean_addr, clean_addr, 8);
71
+
72
tp = tcg_temp_new_ptr();
73
- tcg_gen_addi_ptr(tp, i, imm);
74
- tcg_gen_extu_ptr_i64(addr, tp);
75
- tcg_gen_add_i64(addr, addr, cpu_reg_sp(s, rn));
76
-
77
- tcg_gen_qemu_ld_i64(t0, addr, midx, MO_LEQ);
78
-
79
tcg_gen_add_ptr(tp, cpu_env, i);
80
tcg_gen_addi_ptr(i, i, 8);
81
tcg_gen_st_i64(t0, tp, vofs);
82
tcg_temp_free_ptr(tp);
83
+ tcg_temp_free_i64(t0);
84
85
tcg_gen_brcondi_ptr(TCG_COND_LTU, i, len_align, loop);
86
tcg_temp_free_ptr(i);
32
}
87
}
33
88
34
sysmem = get_system_memory();
89
- /* Predicate register loads can be any multiple of 2.
35
@@ -XXX,XX +XXX,XX @@ static void calxeda_init(MachineState *machine, enum cxmachines machine_id)
90
+ /*
36
for (n = 0; n < smp_cpus; n++) {
91
+ * Predicate register loads can be any multiple of 2.
37
sysbus_connect_irq(busdev, n, cpu_irq[n]);
92
* Note that we still store the entire 64-bit unit into cpu_env.
38
sysbus_connect_irq(busdev, n + smp_cpus, cpu_fiq[n]);
93
*/
39
+ sysbus_connect_irq(busdev, n + 2 * smp_cpus, cpu_virq[n]);
94
if (len_remain) {
40
+ sysbus_connect_irq(busdev, n + 3 * smp_cpus, cpu_vfiq[n]);
95
- tcg_gen_addi_i64(addr, cpu_reg_sp(s, rn), imm + len_align);
96
-
97
+ t0 = tcg_temp_new_i64();
98
switch (len_remain) {
99
case 2:
100
case 4:
101
case 8:
102
- tcg_gen_qemu_ld_i64(t0, addr, midx, MO_LE | ctz32(len_remain));
103
+ tcg_gen_qemu_ld_i64(t0, clean_addr, midx,
104
+ MO_LE | ctz32(len_remain));
105
break;
106
107
case 6:
108
t1 = tcg_temp_new_i64();
109
- tcg_gen_qemu_ld_i64(t0, addr, midx, MO_LEUL);
110
- tcg_gen_addi_i64(addr, addr, 4);
111
- tcg_gen_qemu_ld_i64(t1, addr, midx, MO_LEUW);
112
+ tcg_gen_qemu_ld_i64(t0, clean_addr, midx, MO_LEUL);
113
+ tcg_gen_addi_i64(clean_addr, clean_addr, 4);
114
+ tcg_gen_qemu_ld_i64(t1, clean_addr, midx, MO_LEUW);
115
tcg_gen_deposit_i64(t0, t0, t1, 32, 32);
116
tcg_temp_free_i64(t1);
117
break;
118
@@ -XXX,XX +XXX,XX @@ static void do_ldr(DisasContext *s, uint32_t vofs, int len, int rn, int imm)
119
g_assert_not_reached();
120
}
121
tcg_gen_st_i64(t0, cpu_env, vofs + len_align);
122
+ tcg_temp_free_i64(t0);
41
}
123
}
42
124
- tcg_temp_free_i64(addr);
43
for (n = 0; n < 128; n++) {
125
- tcg_temp_free_i64(t0);
126
+ tcg_temp_free_i64(clean_addr);
127
}
128
129
/* Similarly for stores. */
44
--
130
--
45
2.18.0
131
2.20.1
46
132
47
133
diff view generated by jsdifflib
1
The IoTKit has a CMSDK timer device that runs on the S32KCLK.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
Create this and wire it up.
3
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-30-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180820141116.9118-8-peter.maydell@linaro.org
8
---
7
---
9
include/hw/arm/iotkit.h | 2 +-
8
target/arm/translate-sve.c | 61 +++++++++++++++++++++-----------------
10
hw/arm/iotkit.c | 9 +++++----
9
1 file changed, 33 insertions(+), 28 deletions(-)
11
2 files changed, 6 insertions(+), 5 deletions(-)
12
10
13
diff --git a/include/hw/arm/iotkit.h b/include/hw/arm/iotkit.h
11
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
14
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
15
--- a/include/hw/arm/iotkit.h
13
--- a/target/arm/translate-sve.c
16
+++ b/include/hw/arm/iotkit.h
14
+++ b/target/arm/translate-sve.c
17
@@ -XXX,XX +XXX,XX @@ typedef struct IoTKit {
15
@@ -XXX,XX +XXX,XX @@ static void do_str(DisasContext *s, uint32_t vofs, int len, int rn, int imm)
18
TZMPC mpc;
16
int len_remain = len % 8;
19
CMSDKAPBTIMER timer0;
17
int nparts = len / 8 + ctpop8(len_remain);
20
CMSDKAPBTIMER timer1;
18
int midx = get_mem_index(s);
21
+ CMSDKAPBTIMER s32ktimer;
19
- TCGv_i64 addr, t0;
22
qemu_or_irq ppc_irq_orgate;
20
+ TCGv_i64 dirty_addr, clean_addr, t0;
23
SplitIRQ sec_resp_splitter;
21
24
SplitIRQ ppc_irq_splitter[NUM_PPCS];
22
- addr = tcg_temp_new_i64();
25
@@ -XXX,XX +XXX,XX @@ typedef struct IoTKit {
23
- t0 = tcg_temp_new_i64();
26
qemu_or_irq nmi_orgate;
24
+ dirty_addr = tcg_temp_new_i64();
27
25
+ tcg_gen_addi_i64(dirty_addr, cpu_reg_sp(s, rn), imm);
28
CMSDKAPBDualTimer dualtimer;
26
+ clean_addr = gen_mte_checkN(s, dirty_addr, false, rn != 31, len, MO_8);
29
- UnimplementedDeviceState s32ktimer;
27
+ tcg_temp_free_i64(dirty_addr);
30
28
31
CMSDKAPBWatchdog s32kwatchdog;
29
/* Note that unpredicated load/store of vector/predicate registers
32
CMSDKAPBWatchdog nswatchdog;
30
* are defined as a stream of bytes, which equates to little-endian
33
diff --git a/hw/arm/iotkit.c b/hw/arm/iotkit.c
31
@@ -XXX,XX +XXX,XX @@ static void do_str(DisasContext *s, uint32_t vofs, int len, int rn, int imm)
34
index XXXXXXX..XXXXXXX 100644
32
if (nparts <= 4) {
35
--- a/hw/arm/iotkit.c
33
int i;
36
+++ b/hw/arm/iotkit.c
34
37
@@ -XXX,XX +XXX,XX @@ static void iotkit_init(Object *obj)
35
+ t0 = tcg_temp_new_i64();
38
TYPE_CMSDK_APB_TIMER);
36
for (i = 0; i < len_align; i += 8) {
39
sysbus_init_child_obj(obj, "timer1", &s->timer1, sizeof(s->timer1),
37
tcg_gen_ld_i64(t0, cpu_env, vofs + i);
40
TYPE_CMSDK_APB_TIMER);
38
- tcg_gen_addi_i64(addr, cpu_reg_sp(s, rn), imm + i);
41
+ sysbus_init_child_obj(obj, "s32ktimer", &s->s32ktimer, sizeof(s->s32ktimer),
39
- tcg_gen_qemu_st_i64(t0, addr, midx, MO_LEQ);
42
+ TYPE_CMSDK_APB_TIMER);
40
+ tcg_gen_qemu_st_i64(t0, clean_addr, midx, MO_LEQ);
43
sysbus_init_child_obj(obj, "dualtimer", &s->dualtimer, sizeof(s->dualtimer),
41
+ tcg_gen_addi_i64(clean_addr, cpu_reg_sp(s, rn), 8);
44
TYPE_CMSDK_APB_DUALTIMER);
42
}
45
sysbus_init_child_obj(obj, "s32kwatchdog", &s->s32kwatchdog,
43
+ tcg_temp_free_i64(t0);
46
@@ -XXX,XX +XXX,XX @@ static void iotkit_init(Object *obj)
44
} else {
47
TYPE_SPLIT_IRQ, &error_abort, NULL);
45
TCGLabel *loop = gen_new_label();
48
g_free(name);
46
- TCGv_ptr t2, i = tcg_const_local_ptr(0);
47
+ TCGv_ptr tp, i = tcg_const_local_ptr(0);
48
+
49
+ /* Copy the clean address into a local temp, live across the loop. */
50
+ t0 = clean_addr;
51
+ clean_addr = tcg_temp_local_new_i64();
52
+ tcg_gen_mov_i64(clean_addr, t0);
53
+ tcg_temp_free_i64(t0);
54
55
gen_set_label(loop);
56
57
- t2 = tcg_temp_new_ptr();
58
- tcg_gen_add_ptr(t2, cpu_env, i);
59
- tcg_gen_ld_i64(t0, t2, vofs);
60
-
61
- /* Minimize the number of local temps that must be re-read from
62
- * the stack each iteration. Instead, re-compute values other
63
- * than the loop counter.
64
- */
65
- tcg_gen_addi_ptr(t2, i, imm);
66
- tcg_gen_extu_ptr_i64(addr, t2);
67
- tcg_gen_add_i64(addr, addr, cpu_reg_sp(s, rn));
68
- tcg_temp_free_ptr(t2);
69
-
70
- tcg_gen_qemu_st_i64(t0, addr, midx, MO_LEQ);
71
-
72
+ t0 = tcg_temp_new_i64();
73
+ tp = tcg_temp_new_ptr();
74
+ tcg_gen_add_ptr(tp, cpu_env, i);
75
+ tcg_gen_ld_i64(t0, tp, vofs);
76
tcg_gen_addi_ptr(i, i, 8);
77
+ tcg_temp_free_ptr(tp);
78
+
79
+ tcg_gen_qemu_st_i64(t0, clean_addr, midx, MO_LEQ);
80
+ tcg_gen_addi_i64(clean_addr, clean_addr, 8);
81
+ tcg_temp_free_i64(t0);
82
83
tcg_gen_brcondi_ptr(TCG_COND_LTU, i, len_align, loop);
84
tcg_temp_free_ptr(i);
85
@@ -XXX,XX +XXX,XX @@ static void do_str(DisasContext *s, uint32_t vofs, int len, int rn, int imm)
86
87
/* Predicate register stores can be any multiple of 2. */
88
if (len_remain) {
89
+ t0 = tcg_temp_new_i64();
90
tcg_gen_ld_i64(t0, cpu_env, vofs + len_align);
91
- tcg_gen_addi_i64(addr, cpu_reg_sp(s, rn), imm + len_align);
92
93
switch (len_remain) {
94
case 2:
95
case 4:
96
case 8:
97
- tcg_gen_qemu_st_i64(t0, addr, midx, MO_LE | ctz32(len_remain));
98
+ tcg_gen_qemu_st_i64(t0, clean_addr, midx,
99
+ MO_LE | ctz32(len_remain));
100
break;
101
102
case 6:
103
- tcg_gen_qemu_st_i64(t0, addr, midx, MO_LEUL);
104
- tcg_gen_addi_i64(addr, addr, 4);
105
+ tcg_gen_qemu_st_i64(t0, clean_addr, midx, MO_LEUL);
106
+ tcg_gen_addi_i64(clean_addr, clean_addr, 4);
107
tcg_gen_shri_i64(t0, t0, 32);
108
- tcg_gen_qemu_st_i64(t0, addr, midx, MO_LEUW);
109
+ tcg_gen_qemu_st_i64(t0, clean_addr, midx, MO_LEUW);
110
break;
111
112
default:
113
g_assert_not_reached();
114
}
115
+ tcg_temp_free_i64(t0);
49
}
116
}
50
- sysbus_init_child_obj(obj, "s32ktimer", &s->s32ktimer, sizeof(s->s32ktimer),
117
- tcg_temp_free_i64(addr);
51
- TYPE_UNIMPLEMENTED_DEVICE);
118
- tcg_temp_free_i64(t0);
119
+ tcg_temp_free_i64(clean_addr);
52
}
120
}
53
121
54
static void iotkit_exp_irq(void *opaque, int n, int level)
122
static bool trans_LDR_zri(DisasContext *s, arg_rri *a)
55
@@ -XXX,XX +XXX,XX @@ static void iotkit_realize(DeviceState *dev, Error **errp)
56
/* Devices behind APB PPC1:
57
* 0x4002f000: S32K timer
58
*/
59
- qdev_prop_set_string(DEVICE(&s->s32ktimer), "name", "S32KTIMER");
60
- qdev_prop_set_uint64(DEVICE(&s->s32ktimer), "size", 0x1000);
61
+ qdev_prop_set_uint32(DEVICE(&s->s32ktimer), "pclk-frq", S32KCLK);
62
object_property_set_bool(OBJECT(&s->s32ktimer), true, "realized", &err);
63
if (err) {
64
error_propagate(errp, err);
65
return;
66
}
67
+ sysbus_connect_irq(SYS_BUS_DEVICE(&s->s32ktimer), 0,
68
+ qdev_get_gpio_in(DEVICE(&s->armv7m), 2));
69
mr = sysbus_mmio_get_region(SYS_BUS_DEVICE(&s->s32ktimer), 0);
70
object_property_set_link(OBJECT(&s->apb_ppc1), OBJECT(mr), "port[0]", &err);
71
if (err) {
72
--
123
--
73
2.18.0
124
2.20.1
74
125
75
126
diff view generated by jsdifflib
1
In the PL022, register offset 0x20 is the ICR, a write-only
1
From: Richard Henderson <richard.henderson@linaro.org>
2
interrupt-clear register. Register offset 0x24 is DMACR, the DMA
3
control register. We were incorrectly implementing (a stub version
4
of) DMACR at 0x20, and not implementing anything at 0x24. Fix this
5
bug.
6
2
3
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
5
Message-id: 20200626033144.790098-31-richard.henderson@linaro.org
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Message-id: 20180820141116.9118-21-peter.maydell@linaro.org
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
---
7
---
11
hw/ssi/pl022.c | 12 ++++++++++--
8
target/arm/translate-sve.c | 6 ++++--
12
1 file changed, 10 insertions(+), 2 deletions(-)
9
1 file changed, 4 insertions(+), 2 deletions(-)
13
10
14
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
11
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
15
index XXXXXXX..XXXXXXX 100644
12
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/ssi/pl022.c
13
--- a/target/arm/translate-sve.c
17
+++ b/hw/ssi/pl022.c
14
+++ b/target/arm/translate-sve.c
18
@@ -XXX,XX +XXX,XX @@ static uint64_t pl022_read(void *opaque, hwaddr offset,
15
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
19
return s->is;
16
unsigned esz = dtype_esz[a->dtype];
20
case 0x1c: /* MIS */
17
unsigned msz = dtype_msz(a->dtype);
21
return s->im & s->is;
18
TCGLabel *over = gen_new_label();
22
- case 0x20: /* DMACR */
19
- TCGv_i64 temp;
23
+ case 0x24: /* DMACR */
20
+ TCGv_i64 temp, clean_addr;
24
/* Not implemented. */
21
25
return 0;
22
/* If the guarding predicate has no bits set, no load occurs. */
26
default:
23
if (psz <= 8) {
27
@@ -XXX,XX +XXX,XX @@ static void pl022_write(void *opaque, hwaddr offset,
24
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
28
s->im = value;
25
/* Load the data. */
29
pl022_update(s);
26
temp = tcg_temp_new_i64();
30
break;
27
tcg_gen_addi_i64(temp, cpu_reg_sp(s, a->rn), a->imm << msz);
31
- case 0x20: /* DMACR */
28
- tcg_gen_qemu_ld_i64(temp, temp, get_mem_index(s),
32
+ case 0x20: /* ICR */
29
+ clean_addr = gen_mte_check1(s, temp, false, true, msz);
33
+ /*
30
+
34
+ * write-1-to-clear: bit 0 clears ROR, bit 1 clears RT;
31
+ tcg_gen_qemu_ld_i64(temp, clean_addr, get_mem_index(s),
35
+ * RX and TX interrupts cannot be cleared this way.
32
s->be_data | dtype_mop[a->dtype]);
36
+ */
33
37
+ value &= PL022_INT_ROR | PL022_INT_RT;
34
/* Broadcast to *all* elements. */
38
+ s->is &= ~value;
39
+ break;
40
+ case 0x24: /* DMACR */
41
if (value) {
42
qemu_log_mask(LOG_UNIMP, "pl022: DMA not implemented\n");
43
}
44
--
35
--
45
2.18.0
36
2.20.1
46
37
47
38
diff view generated by jsdifflib
1
Create a new include file for the pl022's device struct,
1
From: Richard Henderson <richard.henderson@linaro.org>
2
type macros, etc, so that it can be instantiated using
3
the "embedded struct" coding style.
4
2
5
While we're adding the new file to MAINTAINERS, add
3
Move the variable declarations to the top of the function,
6
also the .c file, which was missing an entry.
4
but do not create a new label before sve_access_check.
7
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-32-richard.henderson@linaro.org
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
10
Message-id: 20180820141116.9118-16-peter.maydell@linaro.org
11
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
12
---
10
---
13
include/hw/ssi/pl022.h | 51 ++++++++++++++++++++++++++++++++++++++++++
11
target/arm/translate-sve.c | 12 +++++++-----
14
hw/ssi/pl022.c | 26 +--------------------
12
1 file changed, 7 insertions(+), 5 deletions(-)
15
MAINTAINERS | 2 ++
16
3 files changed, 54 insertions(+), 25 deletions(-)
17
create mode 100644 include/hw/ssi/pl022.h
18
13
19
diff --git a/include/hw/ssi/pl022.h b/include/hw/ssi/pl022.h
14
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
20
new file mode 100644
15
index XXXXXXX..XXXXXXX 100644
21
index XXXXXXX..XXXXXXX
16
--- a/target/arm/translate-sve.c
22
--- /dev/null
17
+++ b/target/arm/translate-sve.c
23
+++ b/include/hw/ssi/pl022.h
18
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1RQ_zpri(DisasContext *s, arg_rpri_load *a)
24
@@ -XXX,XX +XXX,XX @@
19
/* Load and broadcast element. */
25
+/*
20
static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
26
+ * ARM PrimeCell PL022 Synchronous Serial Port
21
{
27
+ *
22
- if (!sve_access_check(s)) {
28
+ * Copyright (c) 2007 CodeSourcery.
23
- return true;
29
+ * Written by Paul Brook
24
- }
30
+ *
25
-
31
+ * This program is free software; you can redistribute it and/or modify
26
unsigned vsz = vec_full_reg_size(s);
32
+ * it under the terms of the GNU General Public License version 2 or
27
unsigned psz = pred_full_reg_size(s);
33
+ * (at your option) any later version.
28
unsigned esz = dtype_esz[a->dtype];
34
+ */
29
unsigned msz = dtype_msz(a->dtype);
30
- TCGLabel *over = gen_new_label();
31
+ TCGLabel *over;
32
TCGv_i64 temp, clean_addr;
33
34
+ if (!sve_access_check(s)) {
35
+ return true;
36
+ }
35
+
37
+
36
+/* This is a model of the Arm PrimeCell PL022 synchronous serial port.
38
+ over = gen_new_label();
37
+ * The PL022 TRM is:
38
+ * http://infocenter.arm.com/help/topic/com.arm.doc.ddi0194h/DDI0194H_ssp_pl022_trm.pdf
39
+ *
40
+ * QEMU interface:
41
+ * + sysbus IRQ: SSPINTR combined interrupt line
42
+ * + sysbus MMIO region 0: MemoryRegion for the device's registers
43
+ */
44
+
39
+
45
+#ifndef HW_SSI_PL022_H
40
/* If the guarding predicate has no bits set, no load occurs. */
46
+#define HW_SSI_PL022_H
41
if (psz <= 8) {
47
+
42
/* Reduce the pred_esz_masks value simply to reduce the
48
+#include "hw/sysbus.h"
49
+
50
+#define TYPE_PL022 "pl022"
51
+#define PL022(obj) OBJECT_CHECK(PL022State, (obj), TYPE_PL022)
52
+
53
+typedef struct PL022State {
54
+ SysBusDevice parent_obj;
55
+
56
+ MemoryRegion iomem;
57
+ uint32_t cr0;
58
+ uint32_t cr1;
59
+ uint32_t bitmask;
60
+ uint32_t sr;
61
+ uint32_t cpsr;
62
+ uint32_t is;
63
+ uint32_t im;
64
+ /* The FIFO head points to the next empty entry. */
65
+ int tx_fifo_head;
66
+ int rx_fifo_head;
67
+ int tx_fifo_len;
68
+ int rx_fifo_len;
69
+ uint16_t tx_fifo[8];
70
+ uint16_t rx_fifo[8];
71
+ qemu_irq irq;
72
+ SSIBus *ssi;
73
+} PL022State;
74
+
75
+#endif
76
diff --git a/hw/ssi/pl022.c b/hw/ssi/pl022.c
77
index XXXXXXX..XXXXXXX 100644
78
--- a/hw/ssi/pl022.c
79
+++ b/hw/ssi/pl022.c
80
@@ -XXX,XX +XXX,XX @@
81
82
#include "qemu/osdep.h"
83
#include "hw/sysbus.h"
84
+#include "hw/ssi/pl022.h"
85
#include "hw/ssi/ssi.h"
86
#include "qemu/log.h"
87
88
@@ -XXX,XX +XXX,XX @@ do { fprintf(stderr, "pl022: error: " fmt , ## __VA_ARGS__);} while (0)
89
#define PL022_INT_RX 0x04
90
#define PL022_INT_TX 0x08
91
92
-#define TYPE_PL022 "pl022"
93
-#define PL022(obj) OBJECT_CHECK(PL022State, (obj), TYPE_PL022)
94
-
95
-typedef struct PL022State {
96
- SysBusDevice parent_obj;
97
-
98
- MemoryRegion iomem;
99
- uint32_t cr0;
100
- uint32_t cr1;
101
- uint32_t bitmask;
102
- uint32_t sr;
103
- uint32_t cpsr;
104
- uint32_t is;
105
- uint32_t im;
106
- /* The FIFO head points to the next empty entry. */
107
- int tx_fifo_head;
108
- int rx_fifo_head;
109
- int tx_fifo_len;
110
- int rx_fifo_len;
111
- uint16_t tx_fifo[8];
112
- uint16_t rx_fifo[8];
113
- qemu_irq irq;
114
- SSIBus *ssi;
115
-} PL022State;
116
-
117
static const unsigned char pl022_id[8] =
118
{ 0x22, 0x10, 0x04, 0x00, 0x0d, 0xf0, 0x05, 0xb1 };
119
120
diff --git a/MAINTAINERS b/MAINTAINERS
121
index XXXXXXX..XXXXXXX 100644
122
--- a/MAINTAINERS
123
+++ b/MAINTAINERS
124
@@ -XXX,XX +XXX,XX @@ F: hw/gpio/pl061.c
125
F: hw/input/pl050.c
126
F: hw/intc/pl190.c
127
F: hw/sd/pl181.c
128
+F: hw/ssi/pl022.c
129
+F: include/hw/ssi/pl022.h
130
F: hw/timer/pl031.c
131
F: include/hw/arm/primecell.h
132
F: hw/timer/cmsdk-apb-timer.c
133
--
43
--
134
2.18.0
44
2.20.1
135
45
136
46
diff view generated by jsdifflib
1
Don't request that the arm_load_kernel() code should boot in secure
1
From: Richard Henderson <richard.henderson@linaro.org>
2
state if the CPU doesn't have a secure state. Currently this
3
doesn't make a difference because the boot.c code only examines
4
the secure_boot flag in code guarded by an ARM_FEATURE_EL3 check,
5
but upcoming changes for supporting booting into Hyp mode will
6
change that.
7
2
3
Introduce an lvalue macro to wrap target_tlb_bit0.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-33-richard.henderson@linaro.org
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
10
Message-id: 20180821132811.17675-9-peter.maydell@linaro.org
11
---
9
---
12
hw/arm/vexpress.c | 4 ++--
10
target/arm/cpu.h | 13 +++++++++++++
13
1 file changed, 2 insertions(+), 2 deletions(-)
11
target/arm/helper.c | 2 +-
12
target/arm/translate-a64.c | 2 +-
13
3 files changed, 15 insertions(+), 2 deletions(-)
14
14
15
diff --git a/hw/arm/vexpress.c b/hw/arm/vexpress.c
15
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
16
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/vexpress.c
17
--- a/target/arm/cpu.h
18
+++ b/hw/arm/vexpress.c
18
+++ b/target/arm/cpu.h
19
@@ -XXX,XX +XXX,XX @@ static void vexpress_common_init(MachineState *machine)
19
@@ -XXX,XX +XXX,XX @@ static inline uint64_t *aa64_vfp_qreg(CPUARMState *env, unsigned regno)
20
daughterboard->bootinfo.smp_bootreg_addr = map[VE_SYSREGS] + 0x30;
20
/* Shared between translate-sve.c and sve_helper.c. */
21
daughterboard->bootinfo.gic_cpu_if_addr = daughterboard->gic_cpu_if_addr;
21
extern const uint64_t pred_esz_masks[4];
22
daughterboard->bootinfo.modify_dtb = vexpress_modify_dtb;
22
23
- /* Indicate that when booting Linux we should be in secure state */
23
+/* Helper for the macros below, validating the argument type. */
24
- daughterboard->bootinfo.secure_boot = true;
24
+static inline MemTxAttrs *typecheck_memtxattrs(MemTxAttrs *x)
25
+ /* When booting Linux we should be in secure state if the CPU has one. */
25
+{
26
+ daughterboard->bootinfo.secure_boot = vms->secure;
26
+ return x;
27
arm_load_kernel(ARM_CPU(first_cpu), &daughterboard->bootinfo);
27
+}
28
+
29
+/*
30
+ * Lvalue macros for ARM TLB bits that we must cache in the TCG TLB.
31
+ * Using these should be a bit more self-documenting than using the
32
+ * generic target bits directly.
33
+ */
34
+#define arm_tlb_bti_gp(x) (typecheck_memtxattrs(x)->target_tlb_bit0)
35
+
36
/*
37
* Naming convention for isar_feature functions:
38
* Functions which test 32-bit ID registers should have _aa32_ in
39
diff --git a/target/arm/helper.c b/target/arm/helper.c
40
index XXXXXXX..XXXXXXX 100644
41
--- a/target/arm/helper.c
42
+++ b/target/arm/helper.c
43
@@ -XXX,XX +XXX,XX @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
44
}
45
/* When in aarch64 mode, and BTI is enabled, remember GP in the IOTLB. */
46
if (aarch64 && guarded && cpu_isar_feature(aa64_bti, cpu)) {
47
- txattrs->target_tlb_bit0 = true;
48
+ arm_tlb_bti_gp(txattrs) = true;
49
}
50
51
if (cacheattrs != NULL) {
52
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
53
index XXXXXXX..XXXXXXX 100644
54
--- a/target/arm/translate-a64.c
55
+++ b/target/arm/translate-a64.c
56
@@ -XXX,XX +XXX,XX @@ static bool is_guarded_page(CPUARMState *env, DisasContext *s)
57
* table entry even for that case.
58
*/
59
return (tlb_hit(entry->addr_code, addr) &&
60
- env_tlb(env)->d[mmu_idx].iotlb[index].attrs.target_tlb_bit0);
61
+ arm_tlb_bti_gp(&env_tlb(env)->d[mmu_idx].iotlb[index].attrs));
62
#endif
28
}
63
}
29
64
30
--
65
--
31
2.18.0
66
2.20.1
32
67
33
68
diff view generated by jsdifflib
1
The IoTKit does not have any Master Security Contollers itself,
1
From: Richard Henderson <richard.henderson@linaro.org>
2
but it does provide registers in the secure privilege control
3
block which allow control of MSCs in the external system.
4
Add support for these registers.
5
2
3
Because the elements are sequential, we can eliminate many tests all
4
at once when the tag hits TCMA, or if the page(s) are not Tagged.
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-34-richard.henderson@linaro.org
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Message-id: 20180820141116.9118-13-peter.maydell@linaro.org
9
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
10
---
10
---
11
include/hw/misc/iotkit-secctl.h | 14 +++++++
11
target/arm/cpu.h | 1 +
12
hw/misc/iotkit-secctl.c | 73 +++++++++++++++++++++++++++++----
12
target/arm/helper-sve.h | 58 ++++++++++
13
2 files changed, 79 insertions(+), 8 deletions(-)
13
target/arm/internals.h | 6 +
14
target/arm/sve_helper.c | 218 ++++++++++++++++++++++++++++++-------
15
target/arm/translate-sve.c | 186 ++++++++++++++++++++++---------
16
5 files changed, 378 insertions(+), 91 deletions(-)
14
17
15
diff --git a/include/hw/misc/iotkit-secctl.h b/include/hw/misc/iotkit-secctl.h
18
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
16
index XXXXXXX..XXXXXXX 100644
19
index XXXXXXX..XXXXXXX 100644
17
--- a/include/hw/misc/iotkit-secctl.h
20
--- a/target/arm/cpu.h
18
+++ b/include/hw/misc/iotkit-secctl.h
21
+++ b/target/arm/cpu.h
19
@@ -XXX,XX +XXX,XX @@
22
@@ -XXX,XX +XXX,XX @@ static inline MemTxAttrs *typecheck_memtxattrs(MemTxAttrs *x)
20
* + named GPIO output "sec_resp_cfg" indicating whether blocked accesses
23
* generic target bits directly.
21
* should RAZ/WI or bus error
22
* + named GPIO output "nsc_cfg" whose value tracks the NSCCFG register value
23
+ * + named GPIO output "msc_irq" for the combined IRQ line from the MSCs
24
* Controlling the 2 APB PPCs in the IoTKit:
25
* + named GPIO outputs apb_ppc0_nonsec[0..2] and apb_ppc1_nonsec
26
* + named GPIO outputs apb_ppc0_ap[0..2] and apb_ppc1_ap
27
@@ -XXX,XX +XXX,XX @@
28
* Controlling each of the 16 expansion MPCs which a system using the IoTKit
29
* might provide:
30
* + named GPIO inputs mpcexp_status[0..15]
31
+ * Controlling each of the 16 expansion MSCs which a system using the IoTKit
32
+ * might provide:
33
+ * + named GPIO inputs mscexp_status[0..15]
34
+ * + named GPIO outputs mscexp_clear[0..15]
35
+ * + named GPIO outputs mscexp_ns[0..15]
36
*/
24
*/
37
25
#define arm_tlb_bti_gp(x) (typecheck_memtxattrs(x)->target_tlb_bit0)
38
#ifndef IOTKIT_SECCTL_H
26
+#define arm_tlb_mte_tagged(x) (typecheck_memtxattrs(x)->target_tlb_bit1)
39
@@ -XXX,XX +XXX,XX @@
27
40
#define IOTS_NUM_AHB_EXP_PPC 4
28
/*
41
#define IOTS_NUM_EXP_MPC 16
29
* Naming convention for isar_feature functions:
42
#define IOTS_NUM_MPC 1
30
diff --git a/target/arm/helper-sve.h b/target/arm/helper-sve.h
43
+#define IOTS_NUM_EXP_MSC 16
44
45
typedef struct IoTKitSecCtl IoTKitSecCtl;
46
47
@@ -XXX,XX +XXX,XX @@ struct IoTKitSecCtl {
48
uint32_t brginten;
49
uint32_t mpcintstatus;
50
51
+ uint32_t secmscintstat;
52
+ uint32_t secmscinten;
53
+ uint32_t nsmscexp;
54
+ qemu_irq mscexp_clear[IOTS_NUM_EXP_MSC];
55
+ qemu_irq mscexp_ns[IOTS_NUM_EXP_MSC];
56
+ qemu_irq msc_irq;
57
+
58
IoTKitSecCtlPPC apb[IOTS_NUM_APB_PPC];
59
IoTKitSecCtlPPC apbexp[IOTS_NUM_APB_EXP_PPC];
60
IoTKitSecCtlPPC ahbexp[IOTS_NUM_APB_EXP_PPC];
61
diff --git a/hw/misc/iotkit-secctl.c b/hw/misc/iotkit-secctl.c
62
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
63
--- a/hw/misc/iotkit-secctl.c
32
--- a/target/arm/helper-sve.h
64
+++ b/hw/misc/iotkit-secctl.c
33
+++ b/target/arm/helper-sve.h
65
@@ -XXX,XX +XXX,XX @@ static MemTxResult iotkit_secctl_s_read(void *opaque, hwaddr addr,
34
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_4(sve_ld1sds_le_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
66
r = s->apbexp[offset_to_ppc_idx(offset)].sp;
35
DEF_HELPER_FLAGS_4(sve_ld1sdu_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
67
break;
36
DEF_HELPER_FLAGS_4(sve_ld1sds_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
68
case A_SECMSCINTSTAT:
37
69
+ r = s->secmscintstat;
38
+DEF_HELPER_FLAGS_4(sve_ld1bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
70
+ break;
39
+DEF_HELPER_FLAGS_4(sve_ld2bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
71
case A_SECMSCINTEN:
40
+DEF_HELPER_FLAGS_4(sve_ld3bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
72
+ r = s->secmscinten;
41
+DEF_HELPER_FLAGS_4(sve_ld4bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
73
+ break;
42
+
74
case A_NSMSCEXP:
43
+DEF_HELPER_FLAGS_4(sve_ld1hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
75
- qemu_log_mask(LOG_UNIMP,
44
+DEF_HELPER_FLAGS_4(sve_ld2hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
76
- "IoTKit SecCtl S block read: "
45
+DEF_HELPER_FLAGS_4(sve_ld3hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
77
- "unimplemented offset 0x%x\n", offset);
46
+DEF_HELPER_FLAGS_4(sve_ld4hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
78
- r = 0;
47
+
79
+ r = s->nsmscexp;
48
+DEF_HELPER_FLAGS_4(sve_ld1hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
80
break;
49
+DEF_HELPER_FLAGS_4(sve_ld2hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
81
case A_PID4:
50
+DEF_HELPER_FLAGS_4(sve_ld3hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
82
case A_PID5:
51
+DEF_HELPER_FLAGS_4(sve_ld4hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
83
@@ -XXX,XX +XXX,XX @@ static void iotkit_secctl_ppc_update_irq_enable(IoTKitSecCtlPPC *ppc)
52
+
84
qemu_set_irq(ppc->irq_enable, extract32(value, ppc->irq_bit_offset, 1));
53
+DEF_HELPER_FLAGS_4(sve_ld1ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
85
}
54
+DEF_HELPER_FLAGS_4(sve_ld2ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
86
55
+DEF_HELPER_FLAGS_4(sve_ld3ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
87
+static void iotkit_secctl_update_mscexp_irqs(qemu_irq *msc_irqs, uint32_t value)
56
+DEF_HELPER_FLAGS_4(sve_ld4ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
57
+
58
+DEF_HELPER_FLAGS_4(sve_ld1ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
59
+DEF_HELPER_FLAGS_4(sve_ld2ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
60
+DEF_HELPER_FLAGS_4(sve_ld3ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
61
+DEF_HELPER_FLAGS_4(sve_ld4ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
62
+
63
+DEF_HELPER_FLAGS_4(sve_ld1dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
64
+DEF_HELPER_FLAGS_4(sve_ld2dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
65
+DEF_HELPER_FLAGS_4(sve_ld3dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
66
+DEF_HELPER_FLAGS_4(sve_ld4dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
67
+
68
+DEF_HELPER_FLAGS_4(sve_ld1dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
69
+DEF_HELPER_FLAGS_4(sve_ld2dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
70
+DEF_HELPER_FLAGS_4(sve_ld3dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
71
+DEF_HELPER_FLAGS_4(sve_ld4dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
72
+
73
+DEF_HELPER_FLAGS_4(sve_ld1bhu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
74
+DEF_HELPER_FLAGS_4(sve_ld1bsu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
75
+DEF_HELPER_FLAGS_4(sve_ld1bdu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
76
+DEF_HELPER_FLAGS_4(sve_ld1bhs_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
77
+DEF_HELPER_FLAGS_4(sve_ld1bss_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
78
+DEF_HELPER_FLAGS_4(sve_ld1bds_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
79
+
80
+DEF_HELPER_FLAGS_4(sve_ld1hsu_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
81
+DEF_HELPER_FLAGS_4(sve_ld1hdu_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
82
+DEF_HELPER_FLAGS_4(sve_ld1hss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
83
+DEF_HELPER_FLAGS_4(sve_ld1hds_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
84
+
85
+DEF_HELPER_FLAGS_4(sve_ld1hsu_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
86
+DEF_HELPER_FLAGS_4(sve_ld1hdu_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
87
+DEF_HELPER_FLAGS_4(sve_ld1hss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
88
+DEF_HELPER_FLAGS_4(sve_ld1hds_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
89
+
90
+DEF_HELPER_FLAGS_4(sve_ld1sdu_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
91
+DEF_HELPER_FLAGS_4(sve_ld1sds_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
92
+
93
+DEF_HELPER_FLAGS_4(sve_ld1sdu_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
94
+DEF_HELPER_FLAGS_4(sve_ld1sds_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
95
+
96
DEF_HELPER_FLAGS_4(sve_ldff1bb_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
97
DEF_HELPER_FLAGS_4(sve_ldff1bhu_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
98
DEF_HELPER_FLAGS_4(sve_ldff1bsu_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
99
diff --git a/target/arm/internals.h b/target/arm/internals.h
100
index XXXXXXX..XXXXXXX 100644
101
--- a/target/arm/internals.h
102
+++ b/target/arm/internals.h
103
@@ -XXX,XX +XXX,XX @@ void arm_log_exception(int idx);
104
#define LOG2_TAG_GRANULE 4
105
#define TAG_GRANULE (1 << LOG2_TAG_GRANULE)
106
107
+/*
108
+ * The SVE simd_data field, for memory ops, contains either
109
+ * rd (5 bits) or a shift count (2 bits).
110
+ */
111
+#define SVE_MTEDESC_SHIFT 5
112
+
113
/* Bits within a descriptor passed to the helper_mte_check* functions. */
114
FIELD(MTEDESC, MIDX, 0, 4)
115
FIELD(MTEDESC, TBI, 4, 2)
116
diff --git a/target/arm/sve_helper.c b/target/arm/sve_helper.c
117
index XXXXXXX..XXXXXXX 100644
118
--- a/target/arm/sve_helper.c
119
+++ b/target/arm/sve_helper.c
120
@@ -XXX,XX +XXX,XX @@ static void sve_cont_ldst_watchpoints(SVEContLdSt *info, CPUARMState *env,
121
#endif
122
}
123
124
+typedef uint64_t mte_check_fn(CPUARMState *, uint32_t, uint64_t, uintptr_t);
125
+
126
+static inline QEMU_ALWAYS_INLINE
127
+void sve_cont_ldst_mte_check_int(SVEContLdSt *info, CPUARMState *env,
128
+ uint64_t *vg, target_ulong addr, int esize,
129
+ int msize, uint32_t mtedesc, uintptr_t ra,
130
+ mte_check_fn *check)
88
+{
131
+{
89
+ int i;
132
+ intptr_t mem_off, reg_off, reg_last;
90
+
133
+
91
+ for (i = 0; i < IOTS_NUM_EXP_MSC; i++) {
134
+ /* Process the page only if MemAttr == Tagged. */
92
+ qemu_set_irq(msc_irqs[i], extract32(value, i + 16, 1));
135
+ if (arm_tlb_mte_tagged(&info->page[0].attrs)) {
136
+ mem_off = info->mem_off_first[0];
137
+ reg_off = info->reg_off_first[0];
138
+ reg_last = info->reg_off_split;
139
+ if (reg_last < 0) {
140
+ reg_last = info->reg_off_last[0];
141
+ }
142
+
143
+ do {
144
+ uint64_t pg = vg[reg_off >> 6];
145
+ do {
146
+ if ((pg >> (reg_off & 63)) & 1) {
147
+ check(env, mtedesc, addr, ra);
148
+ }
149
+ reg_off += esize;
150
+ mem_off += msize;
151
+ } while (reg_off <= reg_last && (reg_off & 63));
152
+ } while (reg_off <= reg_last);
153
+ }
154
+
155
+ mem_off = info->mem_off_first[1];
156
+ if (mem_off >= 0 && arm_tlb_mte_tagged(&info->page[1].attrs)) {
157
+ reg_off = info->reg_off_first[1];
158
+ reg_last = info->reg_off_last[1];
159
+
160
+ do {
161
+ uint64_t pg = vg[reg_off >> 6];
162
+ do {
163
+ if ((pg >> (reg_off & 63)) & 1) {
164
+ check(env, mtedesc, addr, ra);
165
+ }
166
+ reg_off += esize;
167
+ mem_off += msize;
168
+ } while (reg_off & 63);
169
+ } while (reg_off <= reg_last);
93
+ }
170
+ }
94
+}
171
+}
95
+
172
+
96
+static void iotkit_secctl_update_msc_irq(IoTKitSecCtl *s)
173
+typedef void sve_cont_ldst_mte_check_fn(SVEContLdSt *info, CPUARMState *env,
174
+ uint64_t *vg, target_ulong addr,
175
+ int esize, int msize, uint32_t mtedesc,
176
+ uintptr_t ra);
177
+
178
+static void sve_cont_ldst_mte_check1(SVEContLdSt *info, CPUARMState *env,
179
+ uint64_t *vg, target_ulong addr,
180
+ int esize, int msize, uint32_t mtedesc,
181
+ uintptr_t ra)
97
+{
182
+{
98
+ /* Update the combined MSC IRQ, based on S_MSCEXP_STATUS and S_MSCEXP_EN */
183
+ sve_cont_ldst_mte_check_int(info, env, vg, addr, esize, msize,
99
+ bool level = s->secmscintstat & s->secmscinten;
184
+ mtedesc, ra, mte_check1);
100
+
101
+ qemu_set_irq(s->msc_irq, level);
102
+}
185
+}
103
+
186
+
104
static MemTxResult iotkit_secctl_s_write(void *opaque, hwaddr addr,
187
+static void sve_cont_ldst_mte_checkN(SVEContLdSt *info, CPUARMState *env,
105
uint64_t value,
188
+ uint64_t *vg, target_ulong addr,
106
unsigned size, MemTxAttrs attrs)
189
+ int esize, int msize, uint32_t mtedesc,
107
@@ -XXX,XX +XXX,XX @@ static MemTxResult iotkit_secctl_s_write(void *opaque, hwaddr addr,
190
+ uintptr_t ra)
108
iotkit_secctl_ppc_sp_write(ppc, value);
109
break;
110
case A_SECMSCINTCLR:
111
+ iotkit_secctl_update_mscexp_irqs(s->mscexp_clear, value);
112
+ break;
113
case A_SECMSCINTEN:
114
- qemu_log_mask(LOG_UNIMP,
115
- "IoTKit SecCtl S block write: "
116
- "unimplemented offset 0x%x\n", offset);
117
+ s->secmscinten = value;
118
+ iotkit_secctl_update_msc_irq(s);
119
+ break;
120
+ case A_NSMSCEXP:
121
+ s->nsmscexp = value;
122
+ iotkit_secctl_update_mscexp_irqs(s->mscexp_ns, value);
123
break;
124
case A_SECMPCINTSTATUS:
125
case A_SECPPCINTSTAT:
126
@@ -XXX,XX +XXX,XX @@ static MemTxResult iotkit_secctl_s_write(void *opaque, hwaddr addr,
127
case A_BRGINTSTAT:
128
case A_AHBNSPPC0:
129
case A_AHBSPPPC0:
130
- case A_NSMSCEXP:
131
case A_PID4:
132
case A_PID5:
133
case A_PID6:
134
@@ -XXX,XX +XXX,XX @@ static void iotkit_secctl_mpcexp_status(void *opaque, int n, int level)
135
s->mpcintstatus = deposit32(s->mpcintstatus, n + 16, 1, !!level);
136
}
137
138
+static void iotkit_secctl_mscexp_status(void *opaque, int n, int level)
139
+{
191
+{
140
+ IoTKitSecCtl *s = IOTKIT_SECCTL(opaque);
192
+ sve_cont_ldst_mte_check_int(info, env, vg, addr, esize, msize,
141
+
193
+ mtedesc, ra, mte_checkN);
142
+ s->secmscintstat = deposit32(s->secmscintstat, n + 16, 1, !!level);
143
+ iotkit_secctl_update_msc_irq(s);
144
+}
194
+}
145
+
195
+
146
static void iotkit_secctl_ppc_irqstatus(void *opaque, int n, int level)
196
+
197
/*
198
* Common helper for all contiguous 1,2,3,4-register predicated stores.
199
*/
200
static inline QEMU_ALWAYS_INLINE
201
void sve_ldN_r(CPUARMState *env, uint64_t *vg, const target_ulong addr,
202
uint32_t desc, const uintptr_t retaddr,
203
- const int esz, const int msz, const int N,
204
+ const int esz, const int msz, const int N, uint32_t mtedesc,
205
sve_ldst1_host_fn *host_fn,
206
- sve_ldst1_tlb_fn *tlb_fn)
207
+ sve_ldst1_tlb_fn *tlb_fn,
208
+ sve_cont_ldst_mte_check_fn *mte_check_fn)
147
{
209
{
148
IoTKitSecCtlPPC *ppc = opaque;
210
const unsigned rd = simd_data(desc);
149
@@ -XXX,XX +XXX,XX @@ static void iotkit_secctl_init(Object *obj)
211
const intptr_t reg_max = simd_oprsz(desc);
150
qdev_init_gpio_in_named(dev, iotkit_secctl_mpcexp_status,
212
@@ -XXX,XX +XXX,XX @@ void sve_ldN_r(CPUARMState *env, uint64_t *vg, const target_ulong addr,
151
"mpcexp_status", IOTS_NUM_EXP_MPC);
213
sve_cont_ldst_watchpoints(&info, env, vg, addr, 1 << esz, N << msz,
152
214
BP_MEM_READ, retaddr);
153
+ qdev_init_gpio_in_named(dev, iotkit_secctl_mscexp_status,
215
154
+ "mscexp_status", IOTS_NUM_EXP_MSC);
216
- /* TODO: MTE check. */
155
+ qdev_init_gpio_out_named(dev, s->mscexp_clear, "mscexp_clear",
217
+ /*
156
+ IOTS_NUM_EXP_MSC);
218
+ * Handle mte checks for all active elements.
157
+ qdev_init_gpio_out_named(dev, s->mscexp_ns, "mscexp_ns",
219
+ * Since TBI must be set for MTE, !mtedesc => !mte_active.
158
+ IOTS_NUM_EXP_MSC);
220
+ */
159
+ qdev_init_gpio_out_named(dev, &s->msc_irq, "msc_irq", 1);
221
+ if (mte_check_fn && mtedesc) {
160
+
222
+ mte_check_fn(&info, env, vg, addr, 1 << esz, N << msz,
161
memory_region_init_io(&s->s_regs, obj, &iotkit_secctl_s_ops,
223
+ mtedesc, retaddr);
162
s, "iotkit-secctl-s-regs", 0x1000);
224
+ }
163
memory_region_init_io(&s->ns_regs, obj, &iotkit_secctl_ns_ops,
225
164
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription iotkit_secctl_mpcintstatus_vmstate = {
226
flags = info.page[0].flags | info.page[1].flags;
227
if (unlikely(flags != 0)) {
228
@@ -XXX,XX +XXX,XX @@ void sve_ldN_r(CPUARMState *env, uint64_t *vg, const target_ulong addr,
165
}
229
}
230
}
231
232
-#define DO_LD1_1(NAME, ESZ) \
233
-void HELPER(sve_##NAME##_r)(CPUARMState *env, void *vg, \
234
- target_ulong addr, uint32_t desc) \
235
-{ \
236
- sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, 1, \
237
- sve_##NAME##_host, sve_##NAME##_tlb); \
238
+static inline QEMU_ALWAYS_INLINE
239
+void sve_ldN_r_mte(CPUARMState *env, uint64_t *vg, target_ulong addr,
240
+ uint32_t desc, const uintptr_t ra,
241
+ const int esz, const int msz, const int N,
242
+ sve_ldst1_host_fn *host_fn,
243
+ sve_ldst1_tlb_fn *tlb_fn)
244
+{
245
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
246
+ int bit55 = extract64(addr, 55, 1);
247
+
248
+ /* Remove mtedesc from the normal sve descriptor. */
249
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
250
+
251
+ /* Perform gross MTE suppression early. */
252
+ if (!tbi_check(desc, bit55) ||
253
+ tcma_check(desc, bit55, allocation_tag_from_addr(addr))) {
254
+ mtedesc = 0;
255
+ }
256
+
257
+ sve_ldN_r(env, vg, addr, desc, ra, esz, msz, N, mtedesc, host_fn, tlb_fn,
258
+ N == 1 ? sve_cont_ldst_mte_check1 : sve_cont_ldst_mte_checkN);
259
}
260
261
-#define DO_LD1_2(NAME, ESZ, MSZ) \
262
-void HELPER(sve_##NAME##_le_r)(CPUARMState *env, void *vg, \
263
- target_ulong addr, uint32_t desc) \
264
-{ \
265
- sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, \
266
- sve_##NAME##_le_host, sve_##NAME##_le_tlb); \
267
-} \
268
-void HELPER(sve_##NAME##_be_r)(CPUARMState *env, void *vg, \
269
- target_ulong addr, uint32_t desc) \
270
-{ \
271
- sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, \
272
- sve_##NAME##_be_host, sve_##NAME##_be_tlb); \
273
+#define DO_LD1_1(NAME, ESZ) \
274
+void HELPER(sve_##NAME##_r)(CPUARMState *env, void *vg, \
275
+ target_ulong addr, uint32_t desc) \
276
+{ \
277
+ sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, 1, 0, \
278
+ sve_##NAME##_host, sve_##NAME##_tlb, NULL); \
279
+} \
280
+void HELPER(sve_##NAME##_r_mte)(CPUARMState *env, void *vg, \
281
+ target_ulong addr, uint32_t desc) \
282
+{ \
283
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MO_8, 1, \
284
+ sve_##NAME##_host, sve_##NAME##_tlb); \
285
+}
286
+
287
+#define DO_LD1_2(NAME, ESZ, MSZ) \
288
+void HELPER(sve_##NAME##_le_r)(CPUARMState *env, void *vg, \
289
+ target_ulong addr, uint32_t desc) \
290
+{ \
291
+ sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, 0, \
292
+ sve_##NAME##_le_host, sve_##NAME##_le_tlb, NULL); \
293
+} \
294
+void HELPER(sve_##NAME##_be_r)(CPUARMState *env, void *vg, \
295
+ target_ulong addr, uint32_t desc) \
296
+{ \
297
+ sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, 0, \
298
+ sve_##NAME##_be_host, sve_##NAME##_be_tlb, NULL); \
299
+} \
300
+void HELPER(sve_##NAME##_le_r_mte)(CPUARMState *env, void *vg, \
301
+ target_ulong addr, uint32_t desc) \
302
+{ \
303
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, \
304
+ sve_##NAME##_le_host, sve_##NAME##_le_tlb); \
305
+} \
306
+void HELPER(sve_##NAME##_be_r_mte)(CPUARMState *env, void *vg, \
307
+ target_ulong addr, uint32_t desc) \
308
+{ \
309
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, 1, \
310
+ sve_##NAME##_be_host, sve_##NAME##_be_tlb); \
311
}
312
313
DO_LD1_1(ld1bb, MO_8)
314
@@ -XXX,XX +XXX,XX @@ DO_LD1_2(ld1dd, MO_64, MO_64)
315
#undef DO_LD1_1
316
#undef DO_LD1_2
317
318
-#define DO_LDN_1(N) \
319
-void HELPER(sve_ld##N##bb_r)(CPUARMState *env, void *vg, \
320
- target_ulong addr, uint32_t desc) \
321
-{ \
322
- sve_ldN_r(env, vg, addr, desc, GETPC(), MO_8, MO_8, N, \
323
- sve_ld1bb_host, sve_ld1bb_tlb); \
324
+#define DO_LDN_1(N) \
325
+void HELPER(sve_ld##N##bb_r)(CPUARMState *env, void *vg, \
326
+ target_ulong addr, uint32_t desc) \
327
+{ \
328
+ sve_ldN_r(env, vg, addr, desc, GETPC(), MO_8, MO_8, N, 0, \
329
+ sve_ld1bb_host, sve_ld1bb_tlb, NULL); \
330
+} \
331
+void HELPER(sve_ld##N##bb_r_mte)(CPUARMState *env, void *vg, \
332
+ target_ulong addr, uint32_t desc) \
333
+{ \
334
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), MO_8, MO_8, N, \
335
+ sve_ld1bb_host, sve_ld1bb_tlb); \
336
}
337
338
-#define DO_LDN_2(N, SUFF, ESZ) \
339
-void HELPER(sve_ld##N##SUFF##_le_r)(CPUARMState *env, void *vg, \
340
- target_ulong addr, uint32_t desc) \
341
-{ \
342
- sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, \
343
- sve_ld1##SUFF##_le_host, sve_ld1##SUFF##_le_tlb); \
344
-} \
345
-void HELPER(sve_ld##N##SUFF##_be_r)(CPUARMState *env, void *vg, \
346
- target_ulong addr, uint32_t desc) \
347
-{ \
348
- sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, \
349
- sve_ld1##SUFF##_be_host, sve_ld1##SUFF##_be_tlb); \
350
+#define DO_LDN_2(N, SUFF, ESZ) \
351
+void HELPER(sve_ld##N##SUFF##_le_r)(CPUARMState *env, void *vg, \
352
+ target_ulong addr, uint32_t desc) \
353
+{ \
354
+ sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, 0, \
355
+ sve_ld1##SUFF##_le_host, sve_ld1##SUFF##_le_tlb, NULL); \
356
+} \
357
+void HELPER(sve_ld##N##SUFF##_be_r)(CPUARMState *env, void *vg, \
358
+ target_ulong addr, uint32_t desc) \
359
+{ \
360
+ sve_ldN_r(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, 0, \
361
+ sve_ld1##SUFF##_be_host, sve_ld1##SUFF##_be_tlb, NULL); \
362
+} \
363
+void HELPER(sve_ld##N##SUFF##_le_r_mte)(CPUARMState *env, void *vg, \
364
+ target_ulong addr, uint32_t desc) \
365
+{ \
366
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, \
367
+ sve_ld1##SUFF##_le_host, sve_ld1##SUFF##_le_tlb); \
368
+} \
369
+void HELPER(sve_ld##N##SUFF##_be_r_mte)(CPUARMState *env, void *vg, \
370
+ target_ulong addr, uint32_t desc) \
371
+{ \
372
+ sve_ldN_r_mte(env, vg, addr, desc, GETPC(), ESZ, ESZ, N, \
373
+ sve_ld1##SUFF##_be_host, sve_ld1##SUFF##_be_tlb); \
374
}
375
376
DO_LDN_1(2)
377
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
378
index XXXXXXX..XXXXXXX 100644
379
--- a/target/arm/translate-sve.c
380
+++ b/target/arm/translate-sve.c
381
@@ -XXX,XX +XXX,XX @@ static const uint8_t dtype_esz[16] = {
166
};
382
};
167
383
168
+static bool needed_always(void *opaque)
384
static void do_mem_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
169
+{
385
- int dtype, gen_helper_gvec_mem *fn)
170
+ return true;
386
+ int dtype, uint32_t mte_n, bool is_write,
171
+}
387
+ gen_helper_gvec_mem *fn)
172
+
388
{
173
+static const VMStateDescription iotkit_secctl_msc_vmstate = {
389
unsigned vsz = vec_full_reg_size(s);
174
+ .name = "iotkit-secctl/msc",
390
TCGv_ptr t_pg;
175
+ .version_id = 1,
391
TCGv_i32 t_desc;
176
+ .minimum_version_id = 1,
392
- int desc;
177
+ .needed = needed_always,
393
+ int desc = 0;
178
+ .fields = (VMStateField[]) {
394
179
+ VMSTATE_UINT32(secmscintstat, IoTKitSecCtl),
395
- /* For e.g. LD4, there are not enough arguments to pass all 4
180
+ VMSTATE_UINT32(secmscinten, IoTKitSecCtl),
396
+ /*
181
+ VMSTATE_UINT32(nsmscexp, IoTKitSecCtl),
397
+ * For e.g. LD4, there are not enough arguments to pass all 4
182
+ VMSTATE_END_OF_LIST()
398
* registers as pointers, so encode the regno into the data field.
399
* For consistency, do this even for LD1.
400
+ * TODO: mte_n check here while callers are updated.
401
*/
402
- desc = simd_desc(vsz, vsz, zt);
403
+ if (mte_n && s->mte_active[0]) {
404
+ int msz = dtype_msz(dtype);
405
+
406
+ desc = FIELD_DP32(desc, MTEDESC, MIDX, get_mem_index(s));
407
+ desc = FIELD_DP32(desc, MTEDESC, TBI, s->tbid);
408
+ desc = FIELD_DP32(desc, MTEDESC, TCMA, s->tcma);
409
+ desc = FIELD_DP32(desc, MTEDESC, WRITE, is_write);
410
+ desc = FIELD_DP32(desc, MTEDESC, ESIZE, 1 << msz);
411
+ desc = FIELD_DP32(desc, MTEDESC, TSIZE, mte_n << msz);
412
+ desc <<= SVE_MTEDESC_SHIFT;
183
+ }
413
+ }
184
+};
414
+ desc = simd_desc(vsz, vsz, zt | desc);
185
+
415
t_desc = tcg_const_i32(desc);
186
static const VMStateDescription iotkit_secctl_vmstate = {
416
t_pg = tcg_temp_new_ptr();
187
.name = "iotkit-secctl",
417
188
.version_id = 1,
418
@@ -XXX,XX +XXX,XX @@ static void do_mem_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
189
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription iotkit_secctl_vmstate = {
419
static void do_ld_zpa(DisasContext *s, int zt, int pg,
190
},
420
TCGv_i64 addr, int dtype, int nreg)
191
.subsections = (const VMStateDescription*[]) {
421
{
192
&iotkit_secctl_mpcintstatus_vmstate,
422
- static gen_helper_gvec_mem * const fns[2][16][4] = {
193
+ &iotkit_secctl_msc_vmstate,
423
- /* Little-endian */
194
NULL
424
- { { gen_helper_sve_ld1bb_r, gen_helper_sve_ld2bb_r,
195
},
425
+ static gen_helper_gvec_mem * const fns[2][2][16][4] = {
196
};
426
+ { /* mte inactive, little-endian */
427
+ { { gen_helper_sve_ld1bb_r, gen_helper_sve_ld2bb_r,
428
gen_helper_sve_ld3bb_r, gen_helper_sve_ld4bb_r },
429
- { gen_helper_sve_ld1bhu_r, NULL, NULL, NULL },
430
- { gen_helper_sve_ld1bsu_r, NULL, NULL, NULL },
431
- { gen_helper_sve_ld1bdu_r, NULL, NULL, NULL },
432
+ { gen_helper_sve_ld1bhu_r, NULL, NULL, NULL },
433
+ { gen_helper_sve_ld1bsu_r, NULL, NULL, NULL },
434
+ { gen_helper_sve_ld1bdu_r, NULL, NULL, NULL },
435
436
- { gen_helper_sve_ld1sds_le_r, NULL, NULL, NULL },
437
- { gen_helper_sve_ld1hh_le_r, gen_helper_sve_ld2hh_le_r,
438
- gen_helper_sve_ld3hh_le_r, gen_helper_sve_ld4hh_le_r },
439
- { gen_helper_sve_ld1hsu_le_r, NULL, NULL, NULL },
440
- { gen_helper_sve_ld1hdu_le_r, NULL, NULL, NULL },
441
+ { gen_helper_sve_ld1sds_le_r, NULL, NULL, NULL },
442
+ { gen_helper_sve_ld1hh_le_r, gen_helper_sve_ld2hh_le_r,
443
+ gen_helper_sve_ld3hh_le_r, gen_helper_sve_ld4hh_le_r },
444
+ { gen_helper_sve_ld1hsu_le_r, NULL, NULL, NULL },
445
+ { gen_helper_sve_ld1hdu_le_r, NULL, NULL, NULL },
446
447
- { gen_helper_sve_ld1hds_le_r, NULL, NULL, NULL },
448
- { gen_helper_sve_ld1hss_le_r, NULL, NULL, NULL },
449
- { gen_helper_sve_ld1ss_le_r, gen_helper_sve_ld2ss_le_r,
450
- gen_helper_sve_ld3ss_le_r, gen_helper_sve_ld4ss_le_r },
451
- { gen_helper_sve_ld1sdu_le_r, NULL, NULL, NULL },
452
+ { gen_helper_sve_ld1hds_le_r, NULL, NULL, NULL },
453
+ { gen_helper_sve_ld1hss_le_r, NULL, NULL, NULL },
454
+ { gen_helper_sve_ld1ss_le_r, gen_helper_sve_ld2ss_le_r,
455
+ gen_helper_sve_ld3ss_le_r, gen_helper_sve_ld4ss_le_r },
456
+ { gen_helper_sve_ld1sdu_le_r, NULL, NULL, NULL },
457
458
- { gen_helper_sve_ld1bds_r, NULL, NULL, NULL },
459
- { gen_helper_sve_ld1bss_r, NULL, NULL, NULL },
460
- { gen_helper_sve_ld1bhs_r, NULL, NULL, NULL },
461
- { gen_helper_sve_ld1dd_le_r, gen_helper_sve_ld2dd_le_r,
462
- gen_helper_sve_ld3dd_le_r, gen_helper_sve_ld4dd_le_r } },
463
+ { gen_helper_sve_ld1bds_r, NULL, NULL, NULL },
464
+ { gen_helper_sve_ld1bss_r, NULL, NULL, NULL },
465
+ { gen_helper_sve_ld1bhs_r, NULL, NULL, NULL },
466
+ { gen_helper_sve_ld1dd_le_r, gen_helper_sve_ld2dd_le_r,
467
+ gen_helper_sve_ld3dd_le_r, gen_helper_sve_ld4dd_le_r } },
468
469
- /* Big-endian */
470
- { { gen_helper_sve_ld1bb_r, gen_helper_sve_ld2bb_r,
471
- gen_helper_sve_ld3bb_r, gen_helper_sve_ld4bb_r },
472
- { gen_helper_sve_ld1bhu_r, NULL, NULL, NULL },
473
- { gen_helper_sve_ld1bsu_r, NULL, NULL, NULL },
474
- { gen_helper_sve_ld1bdu_r, NULL, NULL, NULL },
475
+ /* mte inactive, big-endian */
476
+ { { gen_helper_sve_ld1bb_r, gen_helper_sve_ld2bb_r,
477
+ gen_helper_sve_ld3bb_r, gen_helper_sve_ld4bb_r },
478
+ { gen_helper_sve_ld1bhu_r, NULL, NULL, NULL },
479
+ { gen_helper_sve_ld1bsu_r, NULL, NULL, NULL },
480
+ { gen_helper_sve_ld1bdu_r, NULL, NULL, NULL },
481
482
- { gen_helper_sve_ld1sds_be_r, NULL, NULL, NULL },
483
- { gen_helper_sve_ld1hh_be_r, gen_helper_sve_ld2hh_be_r,
484
- gen_helper_sve_ld3hh_be_r, gen_helper_sve_ld4hh_be_r },
485
- { gen_helper_sve_ld1hsu_be_r, NULL, NULL, NULL },
486
- { gen_helper_sve_ld1hdu_be_r, NULL, NULL, NULL },
487
+ { gen_helper_sve_ld1sds_be_r, NULL, NULL, NULL },
488
+ { gen_helper_sve_ld1hh_be_r, gen_helper_sve_ld2hh_be_r,
489
+ gen_helper_sve_ld3hh_be_r, gen_helper_sve_ld4hh_be_r },
490
+ { gen_helper_sve_ld1hsu_be_r, NULL, NULL, NULL },
491
+ { gen_helper_sve_ld1hdu_be_r, NULL, NULL, NULL },
492
493
- { gen_helper_sve_ld1hds_be_r, NULL, NULL, NULL },
494
- { gen_helper_sve_ld1hss_be_r, NULL, NULL, NULL },
495
- { gen_helper_sve_ld1ss_be_r, gen_helper_sve_ld2ss_be_r,
496
- gen_helper_sve_ld3ss_be_r, gen_helper_sve_ld4ss_be_r },
497
- { gen_helper_sve_ld1sdu_be_r, NULL, NULL, NULL },
498
+ { gen_helper_sve_ld1hds_be_r, NULL, NULL, NULL },
499
+ { gen_helper_sve_ld1hss_be_r, NULL, NULL, NULL },
500
+ { gen_helper_sve_ld1ss_be_r, gen_helper_sve_ld2ss_be_r,
501
+ gen_helper_sve_ld3ss_be_r, gen_helper_sve_ld4ss_be_r },
502
+ { gen_helper_sve_ld1sdu_be_r, NULL, NULL, NULL },
503
504
- { gen_helper_sve_ld1bds_r, NULL, NULL, NULL },
505
- { gen_helper_sve_ld1bss_r, NULL, NULL, NULL },
506
- { gen_helper_sve_ld1bhs_r, NULL, NULL, NULL },
507
- { gen_helper_sve_ld1dd_be_r, gen_helper_sve_ld2dd_be_r,
508
- gen_helper_sve_ld3dd_be_r, gen_helper_sve_ld4dd_be_r } }
509
+ { gen_helper_sve_ld1bds_r, NULL, NULL, NULL },
510
+ { gen_helper_sve_ld1bss_r, NULL, NULL, NULL },
511
+ { gen_helper_sve_ld1bhs_r, NULL, NULL, NULL },
512
+ { gen_helper_sve_ld1dd_be_r, gen_helper_sve_ld2dd_be_r,
513
+ gen_helper_sve_ld3dd_be_r, gen_helper_sve_ld4dd_be_r } } },
514
+
515
+ { /* mte active, little-endian */
516
+ { { gen_helper_sve_ld1bb_r_mte,
517
+ gen_helper_sve_ld2bb_r_mte,
518
+ gen_helper_sve_ld3bb_r_mte,
519
+ gen_helper_sve_ld4bb_r_mte },
520
+ { gen_helper_sve_ld1bhu_r_mte, NULL, NULL, NULL },
521
+ { gen_helper_sve_ld1bsu_r_mte, NULL, NULL, NULL },
522
+ { gen_helper_sve_ld1bdu_r_mte, NULL, NULL, NULL },
523
+
524
+ { gen_helper_sve_ld1sds_le_r_mte, NULL, NULL, NULL },
525
+ { gen_helper_sve_ld1hh_le_r_mte,
526
+ gen_helper_sve_ld2hh_le_r_mte,
527
+ gen_helper_sve_ld3hh_le_r_mte,
528
+ gen_helper_sve_ld4hh_le_r_mte },
529
+ { gen_helper_sve_ld1hsu_le_r_mte, NULL, NULL, NULL },
530
+ { gen_helper_sve_ld1hdu_le_r_mte, NULL, NULL, NULL },
531
+
532
+ { gen_helper_sve_ld1hds_le_r_mte, NULL, NULL, NULL },
533
+ { gen_helper_sve_ld1hss_le_r_mte, NULL, NULL, NULL },
534
+ { gen_helper_sve_ld1ss_le_r_mte,
535
+ gen_helper_sve_ld2ss_le_r_mte,
536
+ gen_helper_sve_ld3ss_le_r_mte,
537
+ gen_helper_sve_ld4ss_le_r_mte },
538
+ { gen_helper_sve_ld1sdu_le_r_mte, NULL, NULL, NULL },
539
+
540
+ { gen_helper_sve_ld1bds_r_mte, NULL, NULL, NULL },
541
+ { gen_helper_sve_ld1bss_r_mte, NULL, NULL, NULL },
542
+ { gen_helper_sve_ld1bhs_r_mte, NULL, NULL, NULL },
543
+ { gen_helper_sve_ld1dd_le_r_mte,
544
+ gen_helper_sve_ld2dd_le_r_mte,
545
+ gen_helper_sve_ld3dd_le_r_mte,
546
+ gen_helper_sve_ld4dd_le_r_mte } },
547
+
548
+ /* mte active, big-endian */
549
+ { { gen_helper_sve_ld1bb_r_mte,
550
+ gen_helper_sve_ld2bb_r_mte,
551
+ gen_helper_sve_ld3bb_r_mte,
552
+ gen_helper_sve_ld4bb_r_mte },
553
+ { gen_helper_sve_ld1bhu_r_mte, NULL, NULL, NULL },
554
+ { gen_helper_sve_ld1bsu_r_mte, NULL, NULL, NULL },
555
+ { gen_helper_sve_ld1bdu_r_mte, NULL, NULL, NULL },
556
+
557
+ { gen_helper_sve_ld1sds_be_r_mte, NULL, NULL, NULL },
558
+ { gen_helper_sve_ld1hh_be_r_mte,
559
+ gen_helper_sve_ld2hh_be_r_mte,
560
+ gen_helper_sve_ld3hh_be_r_mte,
561
+ gen_helper_sve_ld4hh_be_r_mte },
562
+ { gen_helper_sve_ld1hsu_be_r_mte, NULL, NULL, NULL },
563
+ { gen_helper_sve_ld1hdu_be_r_mte, NULL, NULL, NULL },
564
+
565
+ { gen_helper_sve_ld1hds_be_r_mte, NULL, NULL, NULL },
566
+ { gen_helper_sve_ld1hss_be_r_mte, NULL, NULL, NULL },
567
+ { gen_helper_sve_ld1ss_be_r_mte,
568
+ gen_helper_sve_ld2ss_be_r_mte,
569
+ gen_helper_sve_ld3ss_be_r_mte,
570
+ gen_helper_sve_ld4ss_be_r_mte },
571
+ { gen_helper_sve_ld1sdu_be_r_mte, NULL, NULL, NULL },
572
+
573
+ { gen_helper_sve_ld1bds_r_mte, NULL, NULL, NULL },
574
+ { gen_helper_sve_ld1bss_r_mte, NULL, NULL, NULL },
575
+ { gen_helper_sve_ld1bhs_r_mte, NULL, NULL, NULL },
576
+ { gen_helper_sve_ld1dd_be_r_mte,
577
+ gen_helper_sve_ld2dd_be_r_mte,
578
+ gen_helper_sve_ld3dd_be_r_mte,
579
+ gen_helper_sve_ld4dd_be_r_mte } } },
580
};
581
- gen_helper_gvec_mem *fn = fns[s->be_data == MO_BE][dtype][nreg];
582
+ gen_helper_gvec_mem *fn
583
+ = fns[s->mte_active[0]][s->be_data == MO_BE][dtype][nreg];
584
585
- /* While there are holes in the table, they are not
586
+ /*
587
+ * While there are holes in the table, they are not
588
* accessible via the instruction encoding.
589
*/
590
assert(fn != NULL);
591
- do_mem_zpa(s, zt, pg, addr, dtype, fn);
592
+ do_mem_zpa(s, zt, pg, addr, dtype, nreg, false, fn);
593
}
594
595
static bool trans_LD_zprr(DisasContext *s, arg_rprr_load *a)
596
@@ -XXX,XX +XXX,XX @@ static bool trans_LDFF1_zprr(DisasContext *s, arg_rprr_load *a)
597
TCGv_i64 addr = new_tmp_a64(s);
598
tcg_gen_shli_i64(addr, cpu_reg(s, a->rm), dtype_msz(a->dtype));
599
tcg_gen_add_i64(addr, addr, cpu_reg_sp(s, a->rn));
600
- do_mem_zpa(s, a->rd, a->pg, addr, a->dtype,
601
+ do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 0, false,
602
fns[s->be_data == MO_BE][a->dtype]);
603
}
604
return true;
605
@@ -XXX,XX +XXX,XX @@ static bool trans_LDNF1_zpri(DisasContext *s, arg_rpri_load *a)
606
TCGv_i64 addr = new_tmp_a64(s);
607
608
tcg_gen_addi_i64(addr, cpu_reg_sp(s, a->rn), off);
609
- do_mem_zpa(s, a->rd, a->pg, addr, a->dtype,
610
+ do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 0, false,
611
fns[s->be_data == MO_BE][a->dtype]);
612
}
613
return true;
614
@@ -XXX,XX +XXX,XX @@ static void do_st_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
615
fn = fn_multiple[be][nreg - 1][msz];
616
}
617
assert(fn != NULL);
618
- do_mem_zpa(s, zt, pg, addr, msz_dtype(s, msz), fn);
619
+ do_mem_zpa(s, zt, pg, addr, msz_dtype(s, msz), 0, true, fn);
620
}
621
622
static bool trans_ST_zprr(DisasContext *s, arg_rprr_store *a)
197
--
623
--
198
2.18.0
624
2.20.1
199
625
200
626
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Richard Henderson <richard.henderson@linaro.org>
2
2
3
Because the elements are sequential, we can eliminate many tests all
4
at once when the tag hits TCMA, or if the page(s) are not Tagged.
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
4
Message-id: 20180814002653.12828-3-richard.henderson@linaro.org
8
Message-id: 20200626033144.790098-35-richard.henderson@linaro.org
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
10
---
8
include/fpu/softfloat.h | 85 ++++++---
11
target/arm/helper-sve.h | 47 +++++++++++
9
fpu/softfloat.c | 391 ++++++++++++++++++++++++++++++++--------
12
target/arm/sve_helper.c | 95 ++++++++++++++++------
10
2 files changed, 379 insertions(+), 97 deletions(-)
13
target/arm/translate-sve.c | 162 ++++++++++++++++++++++++-------------
11
14
3 files changed, 226 insertions(+), 78 deletions(-)
12
diff --git a/include/fpu/softfloat.h b/include/fpu/softfloat.h
15
16
diff --git a/target/arm/helper-sve.h b/target/arm/helper-sve.h
13
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
14
--- a/include/fpu/softfloat.h
18
--- a/target/arm/helper-sve.h
15
+++ b/include/fpu/softfloat.h
19
+++ b/target/arm/helper-sve.h
16
@@ -XXX,XX +XXX,XX @@ float128 uint64_to_float128(uint64_t, float_status *status);
20
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_4(sve_st1hd_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
17
/*----------------------------------------------------------------------------
21
DEF_HELPER_FLAGS_4(sve_st1sd_le_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
18
| Software half-precision conversion routines.
22
DEF_HELPER_FLAGS_4(sve_st1sd_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
19
*----------------------------------------------------------------------------*/
23
20
+
24
+DEF_HELPER_FLAGS_4(sve_st1bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
21
float16 float32_to_float16(float32, bool ieee, float_status *status);
25
+DEF_HELPER_FLAGS_4(sve_st2bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
22
float32 float16_to_float32(float16, bool ieee, float_status *status);
26
+DEF_HELPER_FLAGS_4(sve_st3bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
23
float16 float64_to_float16(float64 a, bool ieee, float_status *status);
27
+DEF_HELPER_FLAGS_4(sve_st4bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
24
float64 float16_to_float64(float16 a, bool ieee, float_status *status);
28
+
25
+
29
+DEF_HELPER_FLAGS_4(sve_st1hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
26
+int16_t float16_to_int16_scalbn(float16, int, int, float_status *status);
30
+DEF_HELPER_FLAGS_4(sve_st2hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
27
+int32_t float16_to_int32_scalbn(float16, int, int, float_status *status);
31
+DEF_HELPER_FLAGS_4(sve_st3hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
28
+int64_t float16_to_int64_scalbn(float16, int, int, float_status *status);
32
+DEF_HELPER_FLAGS_4(sve_st4hh_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
29
+
33
+
30
int16_t float16_to_int16(float16, float_status *status);
34
+DEF_HELPER_FLAGS_4(sve_st1hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
31
-uint16_t float16_to_uint16(float16 a, float_status *status);
35
+DEF_HELPER_FLAGS_4(sve_st2hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
32
-int16_t float16_to_int16_round_to_zero(float16, float_status *status);
36
+DEF_HELPER_FLAGS_4(sve_st3hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
33
-uint16_t float16_to_uint16_round_to_zero(float16 a, float_status *status);
37
+DEF_HELPER_FLAGS_4(sve_st4hh_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
34
int32_t float16_to_int32(float16, float_status *status);
38
+
35
-uint32_t float16_to_uint32(float16 a, float_status *status);
39
+DEF_HELPER_FLAGS_4(sve_st1ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
36
-int32_t float16_to_int32_round_to_zero(float16, float_status *status);
40
+DEF_HELPER_FLAGS_4(sve_st2ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
37
-uint32_t float16_to_uint32_round_to_zero(float16 a, float_status *status);
41
+DEF_HELPER_FLAGS_4(sve_st3ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
38
int64_t float16_to_int64(float16, float_status *status);
42
+DEF_HELPER_FLAGS_4(sve_st4ss_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
39
-uint64_t float16_to_uint64(float16 a, float_status *status);
43
+
40
+
44
+DEF_HELPER_FLAGS_4(sve_st1ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
41
+int16_t float16_to_int16_round_to_zero(float16, float_status *status);
45
+DEF_HELPER_FLAGS_4(sve_st2ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
42
+int32_t float16_to_int32_round_to_zero(float16, float_status *status);
46
+DEF_HELPER_FLAGS_4(sve_st3ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
43
int64_t float16_to_int64_round_to_zero(float16, float_status *status);
47
+DEF_HELPER_FLAGS_4(sve_st4ss_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
44
+
48
+
45
+uint16_t float16_to_uint16_scalbn(float16 a, int, int, float_status *status);
49
+DEF_HELPER_FLAGS_4(sve_st1dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
46
+uint32_t float16_to_uint32_scalbn(float16 a, int, int, float_status *status);
50
+DEF_HELPER_FLAGS_4(sve_st2dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
47
+uint64_t float16_to_uint64_scalbn(float16 a, int, int, float_status *status);
51
+DEF_HELPER_FLAGS_4(sve_st3dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
48
+
52
+DEF_HELPER_FLAGS_4(sve_st4dd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
49
+uint16_t float16_to_uint16(float16 a, float_status *status);
53
+
50
+uint32_t float16_to_uint32(float16 a, float_status *status);
54
+DEF_HELPER_FLAGS_4(sve_st1dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
51
+uint64_t float16_to_uint64(float16 a, float_status *status);
55
+DEF_HELPER_FLAGS_4(sve_st2dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
52
+
56
+DEF_HELPER_FLAGS_4(sve_st3dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
53
+uint16_t float16_to_uint16_round_to_zero(float16 a, float_status *status);
57
+DEF_HELPER_FLAGS_4(sve_st4dd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
54
+uint32_t float16_to_uint32_round_to_zero(float16 a, float_status *status);
58
+
55
uint64_t float16_to_uint64_round_to_zero(float16 a, float_status *status);
59
+DEF_HELPER_FLAGS_4(sve_st1bh_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
56
60
+DEF_HELPER_FLAGS_4(sve_st1bs_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
57
/*----------------------------------------------------------------------------
61
+DEF_HELPER_FLAGS_4(sve_st1bd_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
58
@@ -XXX,XX +XXX,XX @@ float16 float16_default_nan(float_status *status);
62
+
59
/*----------------------------------------------------------------------------
63
+DEF_HELPER_FLAGS_4(sve_st1hs_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
60
| Software IEC/IEEE single-precision conversion routines.
64
+DEF_HELPER_FLAGS_4(sve_st1hd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
61
*----------------------------------------------------------------------------*/
65
+DEF_HELPER_FLAGS_4(sve_st1hs_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
62
+
66
+DEF_HELPER_FLAGS_4(sve_st1hd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
63
+int16_t float32_to_int16_scalbn(float32, int, int, float_status *status);
67
+
64
+int32_t float32_to_int32_scalbn(float32, int, int, float_status *status);
68
+DEF_HELPER_FLAGS_4(sve_st1sd_le_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
65
+int64_t float32_to_int64_scalbn(float32, int, int, float_status *status);
69
+DEF_HELPER_FLAGS_4(sve_st1sd_be_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
66
+
70
+
67
int16_t float32_to_int16(float32, float_status *status);
71
DEF_HELPER_FLAGS_6(sve_ldbsu_zsu, TCG_CALL_NO_WG,
68
-uint16_t float32_to_uint16(float32, float_status *status);
72
void, env, ptr, ptr, ptr, tl, i32)
69
-int16_t float32_to_int16_round_to_zero(float32, float_status *status);
73
DEF_HELPER_FLAGS_6(sve_ldhsu_le_zsu, TCG_CALL_NO_WG,
70
-uint16_t float32_to_uint16_round_to_zero(float32, float_status *status);
74
diff --git a/target/arm/sve_helper.c b/target/arm/sve_helper.c
71
int32_t float32_to_int32(float32, float_status *status);
72
-int32_t float32_to_int32_round_to_zero(float32, float_status *status);
73
-uint32_t float32_to_uint32(float32, float_status *status);
74
-uint32_t float32_to_uint32_round_to_zero(float32, float_status *status);
75
int64_t float32_to_int64(float32, float_status *status);
76
-uint64_t float32_to_uint64(float32, float_status *status);
77
-uint64_t float32_to_uint64_round_to_zero(float32, float_status *status);
78
+
79
+int16_t float32_to_int16_round_to_zero(float32, float_status *status);
80
+int32_t float32_to_int32_round_to_zero(float32, float_status *status);
81
int64_t float32_to_int64_round_to_zero(float32, float_status *status);
82
+
83
+uint16_t float32_to_uint16_scalbn(float32, int, int, float_status *status);
84
+uint32_t float32_to_uint32_scalbn(float32, int, int, float_status *status);
85
+uint64_t float32_to_uint64_scalbn(float32, int, int, float_status *status);
86
+
87
+uint16_t float32_to_uint16(float32, float_status *status);
88
+uint32_t float32_to_uint32(float32, float_status *status);
89
+uint64_t float32_to_uint64(float32, float_status *status);
90
+
91
+uint16_t float32_to_uint16_round_to_zero(float32, float_status *status);
92
+uint32_t float32_to_uint32_round_to_zero(float32, float_status *status);
93
+uint64_t float32_to_uint64_round_to_zero(float32, float_status *status);
94
+
95
float64 float32_to_float64(float32, float_status *status);
96
floatx80 float32_to_floatx80(float32, float_status *status);
97
float128 float32_to_float128(float32, float_status *status);
98
@@ -XXX,XX +XXX,XX @@ float32 float32_default_nan(float_status *status);
99
/*----------------------------------------------------------------------------
100
| Software IEC/IEEE double-precision conversion routines.
101
*----------------------------------------------------------------------------*/
102
+
103
+int16_t float64_to_int16_scalbn(float64, int, int, float_status *status);
104
+int32_t float64_to_int32_scalbn(float64, int, int, float_status *status);
105
+int64_t float64_to_int64_scalbn(float64, int, int, float_status *status);
106
+
107
int16_t float64_to_int16(float64, float_status *status);
108
-uint16_t float64_to_uint16(float64, float_status *status);
109
-int16_t float64_to_int16_round_to_zero(float64, float_status *status);
110
-uint16_t float64_to_uint16_round_to_zero(float64, float_status *status);
111
int32_t float64_to_int32(float64, float_status *status);
112
-int32_t float64_to_int32_round_to_zero(float64, float_status *status);
113
-uint32_t float64_to_uint32(float64, float_status *status);
114
-uint32_t float64_to_uint32_round_to_zero(float64, float_status *status);
115
int64_t float64_to_int64(float64, float_status *status);
116
+
117
+int16_t float64_to_int16_round_to_zero(float64, float_status *status);
118
+int32_t float64_to_int32_round_to_zero(float64, float_status *status);
119
int64_t float64_to_int64_round_to_zero(float64, float_status *status);
120
-uint64_t float64_to_uint64(float64 a, float_status *status);
121
-uint64_t float64_to_uint64_round_to_zero(float64 a, float_status *status);
122
+
123
+uint16_t float64_to_uint16_scalbn(float64, int, int, float_status *status);
124
+uint32_t float64_to_uint32_scalbn(float64, int, int, float_status *status);
125
+uint64_t float64_to_uint64_scalbn(float64, int, int, float_status *status);
126
+
127
+uint16_t float64_to_uint16(float64, float_status *status);
128
+uint32_t float64_to_uint32(float64, float_status *status);
129
+uint64_t float64_to_uint64(float64, float_status *status);
130
+
131
+uint16_t float64_to_uint16_round_to_zero(float64, float_status *status);
132
+uint32_t float64_to_uint32_round_to_zero(float64, float_status *status);
133
+uint64_t float64_to_uint64_round_to_zero(float64, float_status *status);
134
+
135
float32 float64_to_float32(float64, float_status *status);
136
floatx80 float64_to_floatx80(float64, float_status *status);
137
float128 float64_to_float128(float64, float_status *status);
138
diff --git a/fpu/softfloat.c b/fpu/softfloat.c
139
index XXXXXXX..XXXXXXX 100644
75
index XXXXXXX..XXXXXXX 100644
140
--- a/fpu/softfloat.c
76
--- a/target/arm/sve_helper.c
141
+++ b/fpu/softfloat.c
77
+++ b/target/arm/sve_helper.c
142
@@ -XXX,XX +XXX,XX @@ float32 float64_to_float32(float64 a, float_status *s)
78
@@ -XXX,XX +XXX,XX @@ DO_LDFF1_LDNF1_2(dd, MO_64, MO_64)
143
* Arithmetic.
144
*/
79
*/
145
80
146
-static FloatParts round_to_int(FloatParts a, int rounding_mode, float_status *s)
81
static inline QEMU_ALWAYS_INLINE
147
+static FloatParts round_to_int(FloatParts a, int rmode,
82
-void sve_stN_r(CPUARMState *env, uint64_t *vg, target_ulong addr, uint32_t desc,
148
+ int scale, float_status *s)
83
- const uintptr_t retaddr, const int esz,
84
- const int msz, const int N,
85
+void sve_stN_r(CPUARMState *env, uint64_t *vg, target_ulong addr,
86
+ uint32_t desc, const uintptr_t retaddr,
87
+ const int esz, const int msz, const int N, uint32_t mtedesc,
88
sve_ldst1_host_fn *host_fn,
89
- sve_ldst1_tlb_fn *tlb_fn)
90
+ sve_ldst1_tlb_fn *tlb_fn,
91
+ sve_cont_ldst_mte_check_fn *mte_check_fn)
149
{
92
{
150
- if (is_nan(a.cls)) {
93
const unsigned rd = simd_data(desc);
151
- return return_nan(a, s);
94
const intptr_t reg_max = simd_oprsz(desc);
152
- }
95
@@ -XXX,XX +XXX,XX @@ void sve_stN_r(CPUARMState *env, uint64_t *vg, target_ulong addr, uint32_t desc,
153
-
96
sve_cont_ldst_watchpoints(&info, env, vg, addr, 1 << esz, N << msz,
154
switch (a.cls) {
97
BP_MEM_WRITE, retaddr);
155
+ case float_class_qnan:
98
156
+ case float_class_snan:
99
- /* TODO: MTE check. */
157
+ return return_nan(a, s);
100
+ /*
158
+
101
+ * Handle mte checks for all active elements.
159
case float_class_zero:
102
+ * Since TBI must be set for MTE, !mtedesc => !mte_active.
160
case float_class_inf:
103
+ */
161
- case float_class_qnan:
104
+ if (mte_check_fn && mtedesc) {
162
/* already "integral" */
105
+ mte_check_fn(&info, env, vg, addr, 1 << esz, N << msz,
163
break;
106
+ mtedesc, retaddr);
164
+
107
+ }
165
case float_class_normal:
108
166
+ scale = MIN(MAX(scale, -0x10000), 0x10000);
109
flags = info.page[0].flags | info.page[1].flags;
167
+ a.exp += scale;
110
if (unlikely(flags != 0)) {
168
+
111
@@ -XXX,XX +XXX,XX @@ void sve_stN_r(CPUARMState *env, uint64_t *vg, target_ulong addr, uint32_t desc,
169
if (a.exp >= DECOMPOSED_BINARY_POINT) {
170
/* already integral */
171
break;
172
@@ -XXX,XX +XXX,XX @@ static FloatParts round_to_int(FloatParts a, int rounding_mode, float_status *s)
173
bool one;
174
/* all fractional */
175
s->float_exception_flags |= float_flag_inexact;
176
- switch (rounding_mode) {
177
+ switch (rmode) {
178
case float_round_nearest_even:
179
one = a.exp == -1 && a.frac > DECOMPOSED_IMPLICIT_BIT;
180
break;
181
@@ -XXX,XX +XXX,XX @@ static FloatParts round_to_int(FloatParts a, int rounding_mode, float_status *s)
182
uint64_t rnd_mask = rnd_even_mask >> 1;
183
uint64_t inc;
184
185
- switch (rounding_mode) {
186
+ switch (rmode) {
187
case float_round_nearest_even:
188
inc = ((a.frac & rnd_even_mask) != frac_lsbm1 ? frac_lsbm1 : 0);
189
break;
190
@@ -XXX,XX +XXX,XX @@ static FloatParts round_to_int(FloatParts a, int rounding_mode, float_status *s)
191
float16 float16_round_to_int(float16 a, float_status *s)
192
{
193
FloatParts pa = float16_unpack_canonical(a, s);
194
- FloatParts pr = round_to_int(pa, s->float_rounding_mode, s);
195
+ FloatParts pr = round_to_int(pa, s->float_rounding_mode, 0, s);
196
return float16_round_pack_canonical(pr, s);
197
}
198
199
float32 float32_round_to_int(float32 a, float_status *s)
200
{
201
FloatParts pa = float32_unpack_canonical(a, s);
202
- FloatParts pr = round_to_int(pa, s->float_rounding_mode, s);
203
+ FloatParts pr = round_to_int(pa, s->float_rounding_mode, 0, s);
204
return float32_round_pack_canonical(pr, s);
205
}
206
207
float64 float64_round_to_int(float64 a, float_status *s)
208
{
209
FloatParts pa = float64_unpack_canonical(a, s);
210
- FloatParts pr = round_to_int(pa, s->float_rounding_mode, s);
211
+ FloatParts pr = round_to_int(pa, s->float_rounding_mode, 0, s);
212
return float64_round_pack_canonical(pr, s);
213
}
214
215
float64 float64_trunc_to_int(float64 a, float_status *s)
216
{
217
FloatParts pa = float64_unpack_canonical(a, s);
218
- FloatParts pr = round_to_int(pa, float_round_to_zero, s);
219
+ FloatParts pr = round_to_int(pa, float_round_to_zero, 0, s);
220
return float64_round_pack_canonical(pr, s);
221
}
222
223
@@ -XXX,XX +XXX,XX @@ float64 float64_trunc_to_int(float64 a, float_status *s)
224
* is returned.
225
*/
226
227
-static int64_t round_to_int_and_pack(FloatParts in, int rmode,
228
+static int64_t round_to_int_and_pack(FloatParts in, int rmode, int scale,
229
int64_t min, int64_t max,
230
float_status *s)
231
{
232
uint64_t r;
233
int orig_flags = get_float_exception_flags(s);
234
- FloatParts p = round_to_int(in, rmode, s);
235
+ FloatParts p = round_to_int(in, rmode, scale, s);
236
237
switch (p.cls) {
238
case float_class_snan:
239
@@ -XXX,XX +XXX,XX @@ static int64_t round_to_int_and_pack(FloatParts in, int rmode,
240
}
112
}
241
}
113
}
242
114
243
-#define FLOAT_TO_INT(fsz, isz) \
115
-#define DO_STN_1(N, NAME, ESZ) \
244
-int ## isz ## _t float ## fsz ## _to_int ## isz(float ## fsz a, \
116
-void HELPER(sve_st##N##NAME##_r)(CPUARMState *env, void *vg, \
245
- float_status *s) \
117
- target_ulong addr, uint32_t desc) \
246
-{ \
118
-{ \
247
- FloatParts p = float ## fsz ## _unpack_canonical(a, s); \
119
- sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, N, \
248
- return round_to_int_and_pack(p, s->float_rounding_mode, \
120
- sve_st1##NAME##_host, sve_st1##NAME##_tlb); \
249
- INT ## isz ## _MIN, INT ## isz ## _MAX,\
121
+static inline QEMU_ALWAYS_INLINE
250
- s); \
122
+void sve_stN_r_mte(CPUARMState *env, uint64_t *vg, target_ulong addr,
251
-} \
123
+ uint32_t desc, const uintptr_t ra,
252
- \
124
+ const int esz, const int msz, const int N,
253
-int ## isz ## _t float ## fsz ## _to_int ## isz ## _round_to_zero \
125
+ sve_ldst1_host_fn *host_fn,
254
- (float ## fsz a, float_status *s) \
126
+ sve_ldst1_tlb_fn *tlb_fn)
255
-{ \
256
- FloatParts p = float ## fsz ## _unpack_canonical(a, s); \
257
- return round_to_int_and_pack(p, float_round_to_zero, \
258
- INT ## isz ## _MIN, INT ## isz ## _MAX,\
259
- s); \
260
+int16_t float16_to_int16_scalbn(float16 a, int rmode, int scale,
261
+ float_status *s)
262
+{
127
+{
263
+ return round_to_int_and_pack(float16_unpack_canonical(a, s),
128
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
264
+ rmode, scale, INT16_MIN, INT16_MAX, s);
129
+ int bit55 = extract64(addr, 55, 1);
130
+
131
+ /* Remove mtedesc from the normal sve descriptor. */
132
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
133
+
134
+ /* Perform gross MTE suppression early. */
135
+ if (!tbi_check(desc, bit55) ||
136
+ tcma_check(desc, bit55, allocation_tag_from_addr(addr))) {
137
+ mtedesc = 0;
138
+ }
139
+
140
+ sve_stN_r(env, vg, addr, desc, ra, esz, msz, N, mtedesc, host_fn, tlb_fn,
141
+ N == 1 ? sve_cont_ldst_mte_check1 : sve_cont_ldst_mte_checkN);
265
}
142
}
266
143
267
-FLOAT_TO_INT(16, 16)
144
-#define DO_STN_2(N, NAME, ESZ, MSZ) \
268
-FLOAT_TO_INT(16, 32)
145
-void HELPER(sve_st##N##NAME##_le_r)(CPUARMState *env, void *vg, \
269
-FLOAT_TO_INT(16, 64)
146
- target_ulong addr, uint32_t desc) \
270
+int32_t float16_to_int32_scalbn(float16 a, int rmode, int scale,
147
-{ \
271
+ float_status *s)
148
- sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, \
272
+{
149
- sve_st1##NAME##_le_host, sve_st1##NAME##_le_tlb); \
273
+ return round_to_int_and_pack(float16_unpack_canonical(a, s),
150
-} \
274
+ rmode, scale, INT32_MIN, INT32_MAX, s);
151
-void HELPER(sve_st##N##NAME##_be_r)(CPUARMState *env, void *vg, \
152
- target_ulong addr, uint32_t desc) \
153
-{ \
154
- sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, \
155
- sve_st1##NAME##_be_host, sve_st1##NAME##_be_tlb); \
156
+#define DO_STN_1(N, NAME, ESZ) \
157
+void HELPER(sve_st##N##NAME##_r)(CPUARMState *env, void *vg, \
158
+ target_ulong addr, uint32_t desc) \
159
+{ \
160
+ sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, N, 0, \
161
+ sve_st1##NAME##_host, sve_st1##NAME##_tlb, NULL); \
162
+} \
163
+void HELPER(sve_st##N##NAME##_r_mte)(CPUARMState *env, void *vg, \
164
+ target_ulong addr, uint32_t desc) \
165
+{ \
166
+ sve_stN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MO_8, N, \
167
+ sve_st1##NAME##_host, sve_st1##NAME##_tlb); \
275
+}
168
+}
276
169
+
277
-FLOAT_TO_INT(32, 16)
170
+#define DO_STN_2(N, NAME, ESZ, MSZ) \
278
-FLOAT_TO_INT(32, 32)
171
+void HELPER(sve_st##N##NAME##_le_r)(CPUARMState *env, void *vg, \
279
-FLOAT_TO_INT(32, 64)
172
+ target_ulong addr, uint32_t desc) \
280
+int64_t float16_to_int64_scalbn(float16 a, int rmode, int scale,
173
+{ \
281
+ float_status *s)
174
+ sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, 0, \
282
+{
175
+ sve_st1##NAME##_le_host, sve_st1##NAME##_le_tlb, NULL); \
283
+ return round_to_int_and_pack(float16_unpack_canonical(a, s),
176
+} \
284
+ rmode, scale, INT64_MIN, INT64_MAX, s);
177
+void HELPER(sve_st##N##NAME##_be_r)(CPUARMState *env, void *vg, \
285
+}
178
+ target_ulong addr, uint32_t desc) \
286
179
+{ \
287
-FLOAT_TO_INT(64, 16)
180
+ sve_stN_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, 0, \
288
-FLOAT_TO_INT(64, 32)
181
+ sve_st1##NAME##_be_host, sve_st1##NAME##_be_tlb, NULL); \
289
-FLOAT_TO_INT(64, 64)
182
+} \
290
+int16_t float32_to_int16_scalbn(float32 a, int rmode, int scale,
183
+void HELPER(sve_st##N##NAME##_le_r_mte)(CPUARMState *env, void *vg, \
291
+ float_status *s)
184
+ target_ulong addr, uint32_t desc) \
292
+{
185
+{ \
293
+ return round_to_int_and_pack(float32_unpack_canonical(a, s),
186
+ sve_stN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, \
294
+ rmode, scale, INT16_MIN, INT16_MAX, s);
187
+ sve_st1##NAME##_le_host, sve_st1##NAME##_le_tlb); \
295
+}
188
+} \
296
189
+void HELPER(sve_st##N##NAME##_be_r_mte)(CPUARMState *env, void *vg, \
297
-#undef FLOAT_TO_INT
190
+ target_ulong addr, uint32_t desc) \
298
+int32_t float32_to_int32_scalbn(float32 a, int rmode, int scale,
191
+{ \
299
+ float_status *s)
192
+ sve_stN_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, N, \
300
+{
193
+ sve_st1##NAME##_be_host, sve_st1##NAME##_be_tlb); \
301
+ return round_to_int_and_pack(float32_unpack_canonical(a, s),
194
}
302
+ rmode, scale, INT32_MIN, INT32_MAX, s);
195
303
+}
196
DO_STN_1(1, bb, MO_8)
304
+
197
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
305
+int64_t float32_to_int64_scalbn(float32 a, int rmode, int scale,
198
index XXXXXXX..XXXXXXX 100644
306
+ float_status *s)
199
--- a/target/arm/translate-sve.c
307
+{
200
+++ b/target/arm/translate-sve.c
308
+ return round_to_int_and_pack(float32_unpack_canonical(a, s),
201
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1R_zpri(DisasContext *s, arg_rpri_load *a)
309
+ rmode, scale, INT64_MIN, INT64_MAX, s);
202
static void do_st_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
310
+}
203
int msz, int esz, int nreg)
311
+
312
+int16_t float64_to_int16_scalbn(float64 a, int rmode, int scale,
313
+ float_status *s)
314
+{
315
+ return round_to_int_and_pack(float64_unpack_canonical(a, s),
316
+ rmode, scale, INT16_MIN, INT16_MAX, s);
317
+}
318
+
319
+int32_t float64_to_int32_scalbn(float64 a, int rmode, int scale,
320
+ float_status *s)
321
+{
322
+ return round_to_int_and_pack(float64_unpack_canonical(a, s),
323
+ rmode, scale, INT32_MIN, INT32_MAX, s);
324
+}
325
+
326
+int64_t float64_to_int64_scalbn(float64 a, int rmode, int scale,
327
+ float_status *s)
328
+{
329
+ return round_to_int_and_pack(float64_unpack_canonical(a, s),
330
+ rmode, scale, INT64_MIN, INT64_MAX, s);
331
+}
332
+
333
+int16_t float16_to_int16(float16 a, float_status *s)
334
+{
335
+ return float16_to_int16_scalbn(a, s->float_rounding_mode, 0, s);
336
+}
337
+
338
+int32_t float16_to_int32(float16 a, float_status *s)
339
+{
340
+ return float16_to_int32_scalbn(a, s->float_rounding_mode, 0, s);
341
+}
342
+
343
+int64_t float16_to_int64(float16 a, float_status *s)
344
+{
345
+ return float16_to_int64_scalbn(a, s->float_rounding_mode, 0, s);
346
+}
347
+
348
+int16_t float32_to_int16(float32 a, float_status *s)
349
+{
350
+ return float32_to_int16_scalbn(a, s->float_rounding_mode, 0, s);
351
+}
352
+
353
+int32_t float32_to_int32(float32 a, float_status *s)
354
+{
355
+ return float32_to_int32_scalbn(a, s->float_rounding_mode, 0, s);
356
+}
357
+
358
+int64_t float32_to_int64(float32 a, float_status *s)
359
+{
360
+ return float32_to_int64_scalbn(a, s->float_rounding_mode, 0, s);
361
+}
362
+
363
+int16_t float64_to_int16(float64 a, float_status *s)
364
+{
365
+ return float64_to_int16_scalbn(a, s->float_rounding_mode, 0, s);
366
+}
367
+
368
+int32_t float64_to_int32(float64 a, float_status *s)
369
+{
370
+ return float64_to_int32_scalbn(a, s->float_rounding_mode, 0, s);
371
+}
372
+
373
+int64_t float64_to_int64(float64 a, float_status *s)
374
+{
375
+ return float64_to_int64_scalbn(a, s->float_rounding_mode, 0, s);
376
+}
377
+
378
+int16_t float16_to_int16_round_to_zero(float16 a, float_status *s)
379
+{
380
+ return float16_to_int16_scalbn(a, float_round_to_zero, 0, s);
381
+}
382
+
383
+int32_t float16_to_int32_round_to_zero(float16 a, float_status *s)
384
+{
385
+ return float16_to_int32_scalbn(a, float_round_to_zero, 0, s);
386
+}
387
+
388
+int64_t float16_to_int64_round_to_zero(float16 a, float_status *s)
389
+{
390
+ return float16_to_int64_scalbn(a, float_round_to_zero, 0, s);
391
+}
392
+
393
+int16_t float32_to_int16_round_to_zero(float32 a, float_status *s)
394
+{
395
+ return float32_to_int16_scalbn(a, float_round_to_zero, 0, s);
396
+}
397
+
398
+int32_t float32_to_int32_round_to_zero(float32 a, float_status *s)
399
+{
400
+ return float32_to_int32_scalbn(a, float_round_to_zero, 0, s);
401
+}
402
+
403
+int64_t float32_to_int64_round_to_zero(float32 a, float_status *s)
404
+{
405
+ return float32_to_int64_scalbn(a, float_round_to_zero, 0, s);
406
+}
407
+
408
+int16_t float64_to_int16_round_to_zero(float64 a, float_status *s)
409
+{
410
+ return float64_to_int16_scalbn(a, float_round_to_zero, 0, s);
411
+}
412
+
413
+int32_t float64_to_int32_round_to_zero(float64 a, float_status *s)
414
+{
415
+ return float64_to_int32_scalbn(a, float_round_to_zero, 0, s);
416
+}
417
+
418
+int64_t float64_to_int64_round_to_zero(float64 a, float_status *s)
419
+{
420
+ return float64_to_int64_scalbn(a, float_round_to_zero, 0, s);
421
+}
422
423
/*
424
* Returns the result of converting the floating-point value `a' to
425
@@ -XXX,XX +XXX,XX @@ FLOAT_TO_INT(64, 64)
426
* flag.
427
*/
428
429
-static uint64_t round_to_uint_and_pack(FloatParts in, int rmode, uint64_t max,
430
- float_status *s)
431
+static uint64_t round_to_uint_and_pack(FloatParts in, int rmode, int scale,
432
+ uint64_t max, float_status *s)
433
{
204
{
434
int orig_flags = get_float_exception_flags(s);
205
- static gen_helper_gvec_mem * const fn_single[2][4][4] = {
435
- FloatParts p = round_to_int(in, rmode, s);
206
- { { gen_helper_sve_st1bb_r,
436
+ FloatParts p = round_to_int(in, rmode, scale, s);
207
- gen_helper_sve_st1bh_r,
437
+ uint64_t r;
208
- gen_helper_sve_st1bs_r,
438
209
- gen_helper_sve_st1bd_r },
439
switch (p.cls) {
210
- { NULL,
440
case float_class_snan:
211
- gen_helper_sve_st1hh_le_r,
441
@@ -XXX,XX +XXX,XX @@ static uint64_t round_to_uint_and_pack(FloatParts in, int rmode, uint64_t max,
212
- gen_helper_sve_st1hs_le_r,
442
case float_class_zero:
213
- gen_helper_sve_st1hd_le_r },
443
return 0;
214
- { NULL, NULL,
444
case float_class_normal:
215
- gen_helper_sve_st1ss_le_r,
445
- {
216
- gen_helper_sve_st1sd_le_r },
446
- uint64_t r;
217
- { NULL, NULL, NULL,
447
if (p.sign) {
218
- gen_helper_sve_st1dd_le_r } },
448
s->float_exception_flags = orig_flags | float_flag_invalid;
219
- { { gen_helper_sve_st1bb_r,
449
return 0;
220
- gen_helper_sve_st1bh_r,
450
@@ -XXX,XX +XXX,XX @@ static uint64_t round_to_uint_and_pack(FloatParts in, int rmode, uint64_t max,
221
- gen_helper_sve_st1bs_r,
451
if (r > max) {
222
- gen_helper_sve_st1bd_r },
452
s->float_exception_flags = orig_flags | float_flag_invalid;
223
- { NULL,
453
return max;
224
- gen_helper_sve_st1hh_be_r,
454
- } else {
225
- gen_helper_sve_st1hs_be_r,
455
- return r;
226
- gen_helper_sve_st1hd_be_r },
456
}
227
- { NULL, NULL,
457
- }
228
- gen_helper_sve_st1ss_be_r,
458
+ return r;
229
- gen_helper_sve_st1sd_be_r },
459
default:
230
- { NULL, NULL, NULL,
460
g_assert_not_reached();
231
- gen_helper_sve_st1dd_be_r } },
232
+ static gen_helper_gvec_mem * const fn_single[2][2][4][4] = {
233
+ { { { gen_helper_sve_st1bb_r,
234
+ gen_helper_sve_st1bh_r,
235
+ gen_helper_sve_st1bs_r,
236
+ gen_helper_sve_st1bd_r },
237
+ { NULL,
238
+ gen_helper_sve_st1hh_le_r,
239
+ gen_helper_sve_st1hs_le_r,
240
+ gen_helper_sve_st1hd_le_r },
241
+ { NULL, NULL,
242
+ gen_helper_sve_st1ss_le_r,
243
+ gen_helper_sve_st1sd_le_r },
244
+ { NULL, NULL, NULL,
245
+ gen_helper_sve_st1dd_le_r } },
246
+ { { gen_helper_sve_st1bb_r,
247
+ gen_helper_sve_st1bh_r,
248
+ gen_helper_sve_st1bs_r,
249
+ gen_helper_sve_st1bd_r },
250
+ { NULL,
251
+ gen_helper_sve_st1hh_be_r,
252
+ gen_helper_sve_st1hs_be_r,
253
+ gen_helper_sve_st1hd_be_r },
254
+ { NULL, NULL,
255
+ gen_helper_sve_st1ss_be_r,
256
+ gen_helper_sve_st1sd_be_r },
257
+ { NULL, NULL, NULL,
258
+ gen_helper_sve_st1dd_be_r } } },
259
+
260
+ { { { gen_helper_sve_st1bb_r_mte,
261
+ gen_helper_sve_st1bh_r_mte,
262
+ gen_helper_sve_st1bs_r_mte,
263
+ gen_helper_sve_st1bd_r_mte },
264
+ { NULL,
265
+ gen_helper_sve_st1hh_le_r_mte,
266
+ gen_helper_sve_st1hs_le_r_mte,
267
+ gen_helper_sve_st1hd_le_r_mte },
268
+ { NULL, NULL,
269
+ gen_helper_sve_st1ss_le_r_mte,
270
+ gen_helper_sve_st1sd_le_r_mte },
271
+ { NULL, NULL, NULL,
272
+ gen_helper_sve_st1dd_le_r_mte } },
273
+ { { gen_helper_sve_st1bb_r_mte,
274
+ gen_helper_sve_st1bh_r_mte,
275
+ gen_helper_sve_st1bs_r_mte,
276
+ gen_helper_sve_st1bd_r_mte },
277
+ { NULL,
278
+ gen_helper_sve_st1hh_be_r_mte,
279
+ gen_helper_sve_st1hs_be_r_mte,
280
+ gen_helper_sve_st1hd_be_r_mte },
281
+ { NULL, NULL,
282
+ gen_helper_sve_st1ss_be_r_mte,
283
+ gen_helper_sve_st1sd_be_r_mte },
284
+ { NULL, NULL, NULL,
285
+ gen_helper_sve_st1dd_be_r_mte } } },
286
};
287
- static gen_helper_gvec_mem * const fn_multiple[2][3][4] = {
288
- { { gen_helper_sve_st2bb_r,
289
- gen_helper_sve_st2hh_le_r,
290
- gen_helper_sve_st2ss_le_r,
291
- gen_helper_sve_st2dd_le_r },
292
- { gen_helper_sve_st3bb_r,
293
- gen_helper_sve_st3hh_le_r,
294
- gen_helper_sve_st3ss_le_r,
295
- gen_helper_sve_st3dd_le_r },
296
- { gen_helper_sve_st4bb_r,
297
- gen_helper_sve_st4hh_le_r,
298
- gen_helper_sve_st4ss_le_r,
299
- gen_helper_sve_st4dd_le_r } },
300
- { { gen_helper_sve_st2bb_r,
301
- gen_helper_sve_st2hh_be_r,
302
- gen_helper_sve_st2ss_be_r,
303
- gen_helper_sve_st2dd_be_r },
304
- { gen_helper_sve_st3bb_r,
305
- gen_helper_sve_st3hh_be_r,
306
- gen_helper_sve_st3ss_be_r,
307
- gen_helper_sve_st3dd_be_r },
308
- { gen_helper_sve_st4bb_r,
309
- gen_helper_sve_st4hh_be_r,
310
- gen_helper_sve_st4ss_be_r,
311
- gen_helper_sve_st4dd_be_r } },
312
+ static gen_helper_gvec_mem * const fn_multiple[2][2][3][4] = {
313
+ { { { gen_helper_sve_st2bb_r,
314
+ gen_helper_sve_st2hh_le_r,
315
+ gen_helper_sve_st2ss_le_r,
316
+ gen_helper_sve_st2dd_le_r },
317
+ { gen_helper_sve_st3bb_r,
318
+ gen_helper_sve_st3hh_le_r,
319
+ gen_helper_sve_st3ss_le_r,
320
+ gen_helper_sve_st3dd_le_r },
321
+ { gen_helper_sve_st4bb_r,
322
+ gen_helper_sve_st4hh_le_r,
323
+ gen_helper_sve_st4ss_le_r,
324
+ gen_helper_sve_st4dd_le_r } },
325
+ { { gen_helper_sve_st2bb_r,
326
+ gen_helper_sve_st2hh_be_r,
327
+ gen_helper_sve_st2ss_be_r,
328
+ gen_helper_sve_st2dd_be_r },
329
+ { gen_helper_sve_st3bb_r,
330
+ gen_helper_sve_st3hh_be_r,
331
+ gen_helper_sve_st3ss_be_r,
332
+ gen_helper_sve_st3dd_be_r },
333
+ { gen_helper_sve_st4bb_r,
334
+ gen_helper_sve_st4hh_be_r,
335
+ gen_helper_sve_st4ss_be_r,
336
+ gen_helper_sve_st4dd_be_r } } },
337
+ { { { gen_helper_sve_st2bb_r_mte,
338
+ gen_helper_sve_st2hh_le_r_mte,
339
+ gen_helper_sve_st2ss_le_r_mte,
340
+ gen_helper_sve_st2dd_le_r_mte },
341
+ { gen_helper_sve_st3bb_r_mte,
342
+ gen_helper_sve_st3hh_le_r_mte,
343
+ gen_helper_sve_st3ss_le_r_mte,
344
+ gen_helper_sve_st3dd_le_r_mte },
345
+ { gen_helper_sve_st4bb_r_mte,
346
+ gen_helper_sve_st4hh_le_r_mte,
347
+ gen_helper_sve_st4ss_le_r_mte,
348
+ gen_helper_sve_st4dd_le_r_mte } },
349
+ { { gen_helper_sve_st2bb_r_mte,
350
+ gen_helper_sve_st2hh_be_r_mte,
351
+ gen_helper_sve_st2ss_be_r_mte,
352
+ gen_helper_sve_st2dd_be_r_mte },
353
+ { gen_helper_sve_st3bb_r_mte,
354
+ gen_helper_sve_st3hh_be_r_mte,
355
+ gen_helper_sve_st3ss_be_r_mte,
356
+ gen_helper_sve_st3dd_be_r_mte },
357
+ { gen_helper_sve_st4bb_r_mte,
358
+ gen_helper_sve_st4hh_be_r_mte,
359
+ gen_helper_sve_st4ss_be_r_mte,
360
+ gen_helper_sve_st4dd_be_r_mte } } },
361
};
362
gen_helper_gvec_mem *fn;
363
int be = s->be_data == MO_BE;
364
365
if (nreg == 0) {
366
/* ST1 */
367
- fn = fn_single[be][msz][esz];
368
+ fn = fn_single[s->mte_active[0]][be][msz][esz];
369
+ nreg = 1;
370
} else {
371
/* ST2, ST3, ST4 -- msz == esz, enforced by encoding */
372
assert(msz == esz);
373
- fn = fn_multiple[be][nreg - 1][msz];
374
+ fn = fn_multiple[s->mte_active[0]][be][nreg - 1][msz];
461
}
375
}
376
assert(fn != NULL);
377
- do_mem_zpa(s, zt, pg, addr, msz_dtype(s, msz), 0, true, fn);
378
+ do_mem_zpa(s, zt, pg, addr, msz_dtype(s, msz), nreg, true, fn);
462
}
379
}
463
380
464
-#define FLOAT_TO_UINT(fsz, isz) \
381
static bool trans_ST_zprr(DisasContext *s, arg_rprr_store *a)
465
-uint ## isz ## _t float ## fsz ## _to_uint ## isz(float ## fsz a, \
466
- float_status *s) \
467
-{ \
468
- FloatParts p = float ## fsz ## _unpack_canonical(a, s); \
469
- return round_to_uint_and_pack(p, s->float_rounding_mode, \
470
- UINT ## isz ## _MAX, s); \
471
-} \
472
- \
473
-uint ## isz ## _t float ## fsz ## _to_uint ## isz ## _round_to_zero \
474
- (float ## fsz a, float_status *s) \
475
-{ \
476
- FloatParts p = float ## fsz ## _unpack_canonical(a, s); \
477
- return round_to_uint_and_pack(p, float_round_to_zero, \
478
- UINT ## isz ## _MAX, s); \
479
+uint16_t float16_to_uint16_scalbn(float16 a, int rmode, int scale,
480
+ float_status *s)
481
+{
482
+ return round_to_uint_and_pack(float16_unpack_canonical(a, s),
483
+ rmode, scale, UINT16_MAX, s);
484
}
485
486
-FLOAT_TO_UINT(16, 16)
487
-FLOAT_TO_UINT(16, 32)
488
-FLOAT_TO_UINT(16, 64)
489
+uint32_t float16_to_uint32_scalbn(float16 a, int rmode, int scale,
490
+ float_status *s)
491
+{
492
+ return round_to_uint_and_pack(float16_unpack_canonical(a, s),
493
+ rmode, scale, UINT32_MAX, s);
494
+}
495
496
-FLOAT_TO_UINT(32, 16)
497
-FLOAT_TO_UINT(32, 32)
498
-FLOAT_TO_UINT(32, 64)
499
+uint64_t float16_to_uint64_scalbn(float16 a, int rmode, int scale,
500
+ float_status *s)
501
+{
502
+ return round_to_uint_and_pack(float16_unpack_canonical(a, s),
503
+ rmode, scale, UINT64_MAX, s);
504
+}
505
506
-FLOAT_TO_UINT(64, 16)
507
-FLOAT_TO_UINT(64, 32)
508
-FLOAT_TO_UINT(64, 64)
509
+uint16_t float32_to_uint16_scalbn(float32 a, int rmode, int scale,
510
+ float_status *s)
511
+{
512
+ return round_to_uint_and_pack(float32_unpack_canonical(a, s),
513
+ rmode, scale, UINT16_MAX, s);
514
+}
515
516
-#undef FLOAT_TO_UINT
517
+uint32_t float32_to_uint32_scalbn(float32 a, int rmode, int scale,
518
+ float_status *s)
519
+{
520
+ return round_to_uint_and_pack(float32_unpack_canonical(a, s),
521
+ rmode, scale, UINT32_MAX, s);
522
+}
523
+
524
+uint64_t float32_to_uint64_scalbn(float32 a, int rmode, int scale,
525
+ float_status *s)
526
+{
527
+ return round_to_uint_and_pack(float32_unpack_canonical(a, s),
528
+ rmode, scale, UINT64_MAX, s);
529
+}
530
+
531
+uint16_t float64_to_uint16_scalbn(float64 a, int rmode, int scale,
532
+ float_status *s)
533
+{
534
+ return round_to_uint_and_pack(float64_unpack_canonical(a, s),
535
+ rmode, scale, UINT16_MAX, s);
536
+}
537
+
538
+uint32_t float64_to_uint32_scalbn(float64 a, int rmode, int scale,
539
+ float_status *s)
540
+{
541
+ return round_to_uint_and_pack(float64_unpack_canonical(a, s),
542
+ rmode, scale, UINT32_MAX, s);
543
+}
544
+
545
+uint64_t float64_to_uint64_scalbn(float64 a, int rmode, int scale,
546
+ float_status *s)
547
+{
548
+ return round_to_uint_and_pack(float64_unpack_canonical(a, s),
549
+ rmode, scale, UINT64_MAX, s);
550
+}
551
+
552
+uint16_t float16_to_uint16(float16 a, float_status *s)
553
+{
554
+ return float16_to_uint16_scalbn(a, s->float_rounding_mode, 0, s);
555
+}
556
+
557
+uint32_t float16_to_uint32(float16 a, float_status *s)
558
+{
559
+ return float16_to_uint32_scalbn(a, s->float_rounding_mode, 0, s);
560
+}
561
+
562
+uint64_t float16_to_uint64(float16 a, float_status *s)
563
+{
564
+ return float16_to_uint64_scalbn(a, s->float_rounding_mode, 0, s);
565
+}
566
+
567
+uint16_t float32_to_uint16(float32 a, float_status *s)
568
+{
569
+ return float32_to_uint16_scalbn(a, s->float_rounding_mode, 0, s);
570
+}
571
+
572
+uint32_t float32_to_uint32(float32 a, float_status *s)
573
+{
574
+ return float32_to_uint32_scalbn(a, s->float_rounding_mode, 0, s);
575
+}
576
+
577
+uint64_t float32_to_uint64(float32 a, float_status *s)
578
+{
579
+ return float32_to_uint64_scalbn(a, s->float_rounding_mode, 0, s);
580
+}
581
+
582
+uint16_t float64_to_uint16(float64 a, float_status *s)
583
+{
584
+ return float64_to_uint16_scalbn(a, s->float_rounding_mode, 0, s);
585
+}
586
+
587
+uint32_t float64_to_uint32(float64 a, float_status *s)
588
+{
589
+ return float64_to_uint32_scalbn(a, s->float_rounding_mode, 0, s);
590
+}
591
+
592
+uint64_t float64_to_uint64(float64 a, float_status *s)
593
+{
594
+ return float64_to_uint64_scalbn(a, s->float_rounding_mode, 0, s);
595
+}
596
+
597
+uint16_t float16_to_uint16_round_to_zero(float16 a, float_status *s)
598
+{
599
+ return float16_to_uint16_scalbn(a, float_round_to_zero, 0, s);
600
+}
601
+
602
+uint32_t float16_to_uint32_round_to_zero(float16 a, float_status *s)
603
+{
604
+ return float16_to_uint32_scalbn(a, float_round_to_zero, 0, s);
605
+}
606
+
607
+uint64_t float16_to_uint64_round_to_zero(float16 a, float_status *s)
608
+{
609
+ return float16_to_uint64_scalbn(a, float_round_to_zero, 0, s);
610
+}
611
+
612
+uint16_t float32_to_uint16_round_to_zero(float32 a, float_status *s)
613
+{
614
+ return float32_to_uint16_scalbn(a, float_round_to_zero, 0, s);
615
+}
616
+
617
+uint32_t float32_to_uint32_round_to_zero(float32 a, float_status *s)
618
+{
619
+ return float32_to_uint32_scalbn(a, float_round_to_zero, 0, s);
620
+}
621
+
622
+uint64_t float32_to_uint64_round_to_zero(float32 a, float_status *s)
623
+{
624
+ return float32_to_uint64_scalbn(a, float_round_to_zero, 0, s);
625
+}
626
+
627
+uint16_t float64_to_uint16_round_to_zero(float64 a, float_status *s)
628
+{
629
+ return float64_to_uint16_scalbn(a, float_round_to_zero, 0, s);
630
+}
631
+
632
+uint32_t float64_to_uint32_round_to_zero(float64 a, float_status *s)
633
+{
634
+ return float64_to_uint32_scalbn(a, float_round_to_zero, 0, s);
635
+}
636
+
637
+uint64_t float64_to_uint64_round_to_zero(float64 a, float_status *s)
638
+{
639
+ return float64_to_uint64_scalbn(a, float_round_to_zero, 0, s);
640
+}
641
642
/*
643
* Integer to float conversions
644
--
382
--
645
2.18.0
383
2.20.1
646
384
647
385
diff view generated by jsdifflib
1
In the MPS2 FPGAIO, PSCNTR is a free-running downcounter with
1
From: Richard Henderson <richard.henderson@linaro.org>
2
a reload value configured via the PRESCALE register, and
3
COUNTER counts up by 1 every time PSCNTR reaches zero.
4
Implement these counters.
5
2
6
We can just increment the counters migration subsection's
3
Because the elements are sequential, we can eliminate many tests all
7
version ID because we only added it in the previous commit,
4
at once when the tag hits TCMA, or if the page(s) are not Tagged.
8
so no released QEMU versions will be using it.
9
5
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-36-richard.henderson@linaro.org
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
12
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
13
Message-id: 20180820141116.9118-3-peter.maydell@linaro.org
14
---
10
---
15
include/hw/misc/mps2-fpgaio.h | 6 +++
11
target/arm/helper-sve.h | 98 ++++++++++++++++
16
hw/misc/mps2-fpgaio.c | 97 +++++++++++++++++++++++++++++++++--
12
target/arm/sve_helper.c | 99 ++++++++++++++--
17
2 files changed, 99 insertions(+), 4 deletions(-)
13
target/arm/translate-sve.c | 232 +++++++++++++++++++++++++------------
14
3 files changed, 343 insertions(+), 86 deletions(-)
18
15
19
diff --git a/include/hw/misc/mps2-fpgaio.h b/include/hw/misc/mps2-fpgaio.h
16
diff --git a/target/arm/helper-sve.h b/target/arm/helper-sve.h
20
index XXXXXXX..XXXXXXX 100644
17
index XXXXXXX..XXXXXXX 100644
21
--- a/include/hw/misc/mps2-fpgaio.h
18
--- a/target/arm/helper-sve.h
22
+++ b/include/hw/misc/mps2-fpgaio.h
19
+++ b/target/arm/helper-sve.h
23
@@ -XXX,XX +XXX,XX @@ typedef struct {
20
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_4(sve_ldff1sds_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
24
uint32_t prescale;
21
DEF_HELPER_FLAGS_4(sve_ldff1dd_le_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
25
uint32_t misc;
22
DEF_HELPER_FLAGS_4(sve_ldff1dd_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
26
23
27
+ /* QEMU_CLOCK_VIRTUAL time at which counter and pscntr were last synced */
24
+DEF_HELPER_FLAGS_4(sve_ldff1bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
28
+ int64_t pscntr_sync_ticks;
25
+DEF_HELPER_FLAGS_4(sve_ldff1bhu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
29
+ /* Values of COUNTER and PSCNTR at time pscntr_sync_ticks */
26
+DEF_HELPER_FLAGS_4(sve_ldff1bsu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
30
+ uint32_t counter;
27
+DEF_HELPER_FLAGS_4(sve_ldff1bdu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
31
+ uint32_t pscntr;
28
+DEF_HELPER_FLAGS_4(sve_ldff1bhs_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
32
+
29
+DEF_HELPER_FLAGS_4(sve_ldff1bss_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
33
uint32_t prescale_clk;
30
+DEF_HELPER_FLAGS_4(sve_ldff1bds_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
34
31
+
35
/* These hold the CLOCK_VIRTUAL ns tick when the CLK1HZ/CLK100HZ was zero */
32
+DEF_HELPER_FLAGS_4(sve_ldff1hh_le_r_mte, TCG_CALL_NO_WG,
36
diff --git a/hw/misc/mps2-fpgaio.c b/hw/misc/mps2-fpgaio.c
33
+ void, env, ptr, tl, i32)
34
+DEF_HELPER_FLAGS_4(sve_ldff1hsu_le_r_mte, TCG_CALL_NO_WG,
35
+ void, env, ptr, tl, i32)
36
+DEF_HELPER_FLAGS_4(sve_ldff1hdu_le_r_mte, TCG_CALL_NO_WG,
37
+ void, env, ptr, tl, i32)
38
+DEF_HELPER_FLAGS_4(sve_ldff1hss_le_r_mte, TCG_CALL_NO_WG,
39
+ void, env, ptr, tl, i32)
40
+DEF_HELPER_FLAGS_4(sve_ldff1hds_le_r_mte, TCG_CALL_NO_WG,
41
+ void, env, ptr, tl, i32)
42
+
43
+DEF_HELPER_FLAGS_4(sve_ldff1hh_be_r_mte, TCG_CALL_NO_WG,
44
+ void, env, ptr, tl, i32)
45
+DEF_HELPER_FLAGS_4(sve_ldff1hsu_be_r_mte, TCG_CALL_NO_WG,
46
+ void, env, ptr, tl, i32)
47
+DEF_HELPER_FLAGS_4(sve_ldff1hdu_be_r_mte, TCG_CALL_NO_WG,
48
+ void, env, ptr, tl, i32)
49
+DEF_HELPER_FLAGS_4(sve_ldff1hss_be_r_mte, TCG_CALL_NO_WG,
50
+ void, env, ptr, tl, i32)
51
+DEF_HELPER_FLAGS_4(sve_ldff1hds_be_r_mte, TCG_CALL_NO_WG,
52
+ void, env, ptr, tl, i32)
53
+
54
+DEF_HELPER_FLAGS_4(sve_ldff1ss_le_r_mte, TCG_CALL_NO_WG,
55
+ void, env, ptr, tl, i32)
56
+DEF_HELPER_FLAGS_4(sve_ldff1sdu_le_r_mte, TCG_CALL_NO_WG,
57
+ void, env, ptr, tl, i32)
58
+DEF_HELPER_FLAGS_4(sve_ldff1sds_le_r_mte, TCG_CALL_NO_WG,
59
+ void, env, ptr, tl, i32)
60
+
61
+DEF_HELPER_FLAGS_4(sve_ldff1ss_be_r_mte, TCG_CALL_NO_WG,
62
+ void, env, ptr, tl, i32)
63
+DEF_HELPER_FLAGS_4(sve_ldff1sdu_be_r_mte, TCG_CALL_NO_WG,
64
+ void, env, ptr, tl, i32)
65
+DEF_HELPER_FLAGS_4(sve_ldff1sds_be_r_mte, TCG_CALL_NO_WG,
66
+ void, env, ptr, tl, i32)
67
+
68
+DEF_HELPER_FLAGS_4(sve_ldff1dd_le_r_mte, TCG_CALL_NO_WG,
69
+ void, env, ptr, tl, i32)
70
+DEF_HELPER_FLAGS_4(sve_ldff1dd_be_r_mte, TCG_CALL_NO_WG,
71
+ void, env, ptr, tl, i32)
72
+
73
DEF_HELPER_FLAGS_4(sve_ldnf1bb_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
74
DEF_HELPER_FLAGS_4(sve_ldnf1bhu_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
75
DEF_HELPER_FLAGS_4(sve_ldnf1bsu_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
76
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_4(sve_ldnf1sds_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
77
DEF_HELPER_FLAGS_4(sve_ldnf1dd_le_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
78
DEF_HELPER_FLAGS_4(sve_ldnf1dd_be_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
79
80
+DEF_HELPER_FLAGS_4(sve_ldnf1bb_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
81
+DEF_HELPER_FLAGS_4(sve_ldnf1bhu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
82
+DEF_HELPER_FLAGS_4(sve_ldnf1bsu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
83
+DEF_HELPER_FLAGS_4(sve_ldnf1bdu_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
84
+DEF_HELPER_FLAGS_4(sve_ldnf1bhs_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
85
+DEF_HELPER_FLAGS_4(sve_ldnf1bss_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
86
+DEF_HELPER_FLAGS_4(sve_ldnf1bds_r_mte, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
87
+
88
+DEF_HELPER_FLAGS_4(sve_ldnf1hh_le_r_mte, TCG_CALL_NO_WG,
89
+ void, env, ptr, tl, i32)
90
+DEF_HELPER_FLAGS_4(sve_ldnf1hsu_le_r_mte, TCG_CALL_NO_WG,
91
+ void, env, ptr, tl, i32)
92
+DEF_HELPER_FLAGS_4(sve_ldnf1hdu_le_r_mte, TCG_CALL_NO_WG,
93
+ void, env, ptr, tl, i32)
94
+DEF_HELPER_FLAGS_4(sve_ldnf1hss_le_r_mte, TCG_CALL_NO_WG,
95
+ void, env, ptr, tl, i32)
96
+DEF_HELPER_FLAGS_4(sve_ldnf1hds_le_r_mte, TCG_CALL_NO_WG,
97
+ void, env, ptr, tl, i32)
98
+
99
+DEF_HELPER_FLAGS_4(sve_ldnf1hh_be_r_mte, TCG_CALL_NO_WG,
100
+ void, env, ptr, tl, i32)
101
+DEF_HELPER_FLAGS_4(sve_ldnf1hsu_be_r_mte, TCG_CALL_NO_WG,
102
+ void, env, ptr, tl, i32)
103
+DEF_HELPER_FLAGS_4(sve_ldnf1hdu_be_r_mte, TCG_CALL_NO_WG,
104
+ void, env, ptr, tl, i32)
105
+DEF_HELPER_FLAGS_4(sve_ldnf1hss_be_r_mte, TCG_CALL_NO_WG,
106
+ void, env, ptr, tl, i32)
107
+DEF_HELPER_FLAGS_4(sve_ldnf1hds_be_r_mte, TCG_CALL_NO_WG,
108
+ void, env, ptr, tl, i32)
109
+
110
+DEF_HELPER_FLAGS_4(sve_ldnf1ss_le_r_mte, TCG_CALL_NO_WG,
111
+ void, env, ptr, tl, i32)
112
+DEF_HELPER_FLAGS_4(sve_ldnf1sdu_le_r_mte, TCG_CALL_NO_WG,
113
+ void, env, ptr, tl, i32)
114
+DEF_HELPER_FLAGS_4(sve_ldnf1sds_le_r_mte, TCG_CALL_NO_WG,
115
+ void, env, ptr, tl, i32)
116
+
117
+DEF_HELPER_FLAGS_4(sve_ldnf1ss_be_r_mte, TCG_CALL_NO_WG,
118
+ void, env, ptr, tl, i32)
119
+DEF_HELPER_FLAGS_4(sve_ldnf1sdu_be_r_mte, TCG_CALL_NO_WG,
120
+ void, env, ptr, tl, i32)
121
+DEF_HELPER_FLAGS_4(sve_ldnf1sds_be_r_mte, TCG_CALL_NO_WG,
122
+ void, env, ptr, tl, i32)
123
+
124
+DEF_HELPER_FLAGS_4(sve_ldnf1dd_le_r_mte, TCG_CALL_NO_WG,
125
+ void, env, ptr, tl, i32)
126
+DEF_HELPER_FLAGS_4(sve_ldnf1dd_be_r_mte, TCG_CALL_NO_WG,
127
+ void, env, ptr, tl, i32)
128
+
129
DEF_HELPER_FLAGS_4(sve_st1bb_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
130
DEF_HELPER_FLAGS_4(sve_st2bb_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
131
DEF_HELPER_FLAGS_4(sve_st3bb_r, TCG_CALL_NO_WG, void, env, ptr, tl, i32)
132
diff --git a/target/arm/sve_helper.c b/target/arm/sve_helper.c
37
index XXXXXXX..XXXXXXX 100644
133
index XXXXXXX..XXXXXXX 100644
38
--- a/hw/misc/mps2-fpgaio.c
134
--- a/target/arm/sve_helper.c
39
+++ b/hw/misc/mps2-fpgaio.c
135
+++ b/target/arm/sve_helper.c
40
@@ -XXX,XX +XXX,XX @@ static int64_t tickoff_from_counter(int64_t now, uint32_t count, int frq)
136
@@ -XXX,XX +XXX,XX @@ static void record_fault(CPUARMState *env, uintptr_t i, uintptr_t oprsz)
41
return now - muldiv64(count, NANOSECONDS_PER_SECOND, frq);
137
*/
138
static inline QEMU_ALWAYS_INLINE
139
void sve_ldnfff1_r(CPUARMState *env, void *vg, const target_ulong addr,
140
- uint32_t desc, const uintptr_t retaddr,
141
+ uint32_t desc, const uintptr_t retaddr, uint32_t mtedesc,
142
const int esz, const int msz, const SVEContFault fault,
143
sve_ldst1_host_fn *host_fn,
144
sve_ldst1_tlb_fn *tlb_fn)
145
@@ -XXX,XX +XXX,XX @@ void sve_ldnfff1_r(CPUARMState *env, void *vg, const target_ulong addr,
146
mem_off = info.mem_off_first[0];
147
flags = info.page[0].flags;
148
149
+ /*
150
+ * Disable MTE checking if the Tagged bit is not set. Since TBI must
151
+ * be set within MTEDESC for MTE, !mtedesc => !mte_active.
152
+ */
153
+ if (arm_tlb_mte_tagged(&info.page[0].attrs)) {
154
+ mtedesc = 0;
155
+ }
156
+
157
if (fault == FAULT_FIRST) {
158
+ /* Trapping mte check for the first-fault element. */
159
+ if (mtedesc) {
160
+ mte_check1(env, mtedesc, addr + mem_off, retaddr);
161
+ }
162
+
163
/*
164
* Special handling of the first active element,
165
* if it crosses a page boundary or is MMIO.
166
*/
167
bool is_split = mem_off == info.mem_off_split;
168
- /* TODO: MTE check. */
169
if (unlikely(flags != 0) || unlikely(is_split)) {
170
/*
171
* Use the slow path for cross-page handling.
172
@@ -XXX,XX +XXX,XX @@ void sve_ldnfff1_r(CPUARMState *env, void *vg, const target_ulong addr,
173
/* Watchpoint hit, see below. */
174
goto do_fault;
175
}
176
- /* TODO: MTE check. */
177
+ if (mtedesc && !mte_probe1(env, mtedesc, addr + mem_off)) {
178
+ goto do_fault;
179
+ }
180
/*
181
* Use the slow path for cross-page handling.
182
* This is RAM, without a watchpoint, and will not trap.
183
@@ -XXX,XX +XXX,XX @@ void sve_ldnfff1_r(CPUARMState *env, void *vg, const target_ulong addr,
184
& BP_MEM_READ)) {
185
goto do_fault;
186
}
187
- /* TODO: MTE check. */
188
+ if (mtedesc && !mte_probe1(env, mtedesc, addr + mem_off)) {
189
+ goto do_fault;
190
+ }
191
host_fn(vd, reg_off, host + mem_off);
192
}
193
reg_off += 1 << esz;
194
@@ -XXX,XX +XXX,XX @@ void sve_ldnfff1_r(CPUARMState *env, void *vg, const target_ulong addr,
195
record_fault(env, reg_off, reg_max);
42
}
196
}
43
197
44
+static void resync_counter(MPS2FPGAIO *s)
198
-#define DO_LDFF1_LDNF1_1(PART, ESZ) \
199
+static inline QEMU_ALWAYS_INLINE
200
+void sve_ldnfff1_r_mte(CPUARMState *env, void *vg, target_ulong addr,
201
+ uint32_t desc, const uintptr_t retaddr,
202
+ const int esz, const int msz, const SVEContFault fault,
203
+ sve_ldst1_host_fn *host_fn,
204
+ sve_ldst1_tlb_fn *tlb_fn)
45
+{
205
+{
46
+ /*
206
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
47
+ * Update s->counter and s->pscntr to their true current values
207
+ int bit55 = extract64(addr, 55, 1);
48
+ * by calculating how many times PSCNTR has ticked since the
208
+
49
+ * last time we did a resync.
209
+ /* Remove mtedesc from the normal sve descriptor. */
50
+ */
210
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
51
+ int64_t now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
211
+
52
+ int64_t elapsed = now - s->pscntr_sync_ticks;
212
+ /* Perform gross MTE suppression early. */
53
+
213
+ if (!tbi_check(desc, bit55) ||
54
+ /*
214
+ tcma_check(desc, bit55, allocation_tag_from_addr(addr))) {
55
+ * Round elapsed down to a whole number of PSCNTR ticks, so we don't
215
+ mtedesc = 0;
56
+ * lose time if we do multiple resyncs in a single tick.
57
+ */
58
+ uint64_t ticks = muldiv64(elapsed, s->prescale_clk, NANOSECONDS_PER_SECOND);
59
+
60
+ /*
61
+ * Work out what PSCNTR and COUNTER have moved to. We assume that
62
+ * PSCNTR reloads from PRESCALE one tick-period after it hits zero,
63
+ * and that COUNTER increments at the same moment.
64
+ */
65
+ if (ticks == 0) {
66
+ /* We haven't ticked since the last time we were asked */
67
+ return;
68
+ } else if (ticks < s->pscntr) {
69
+ /* We haven't yet reached zero, just reduce the PSCNTR */
70
+ s->pscntr -= ticks;
71
+ } else {
72
+ if (s->prescale == 0) {
73
+ /*
74
+ * If the reload value is zero then the PSCNTR will stick
75
+ * at zero once it reaches it, and so we will increment
76
+ * COUNTER every tick after that.
77
+ */
78
+ s->counter += ticks - s->pscntr;
79
+ s->pscntr = 0;
80
+ } else {
81
+ /*
82
+ * This is the complicated bit. This ASCII art diagram gives an
83
+ * example with PRESCALE==5 PSCNTR==7:
84
+ *
85
+ * ticks 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14
86
+ * PSCNTR 7 6 5 4 3 2 1 0 5 4 3 2 1 0 5
87
+ * cinc 1 2
88
+ * y 0 1 2 3 4 5 6 7 8 9 10 11 12
89
+ * x 0 1 2 3 4 5 0 1 2 3 4 5 0
90
+ *
91
+ * where x = y % (s->prescale + 1)
92
+ * and so PSCNTR = s->prescale - x
93
+ * and COUNTER is incremented by y / (s->prescale + 1)
94
+ *
95
+ * The case where PSCNTR < PRESCALE works out the same,
96
+ * though we must be careful to calculate y as 64-bit unsigned
97
+ * for all parts of the expression.
98
+ * y < 0 is not possible because that implies ticks < s->pscntr.
99
+ */
100
+ uint64_t y = ticks - s->pscntr + s->prescale;
101
+ s->pscntr = s->prescale - (y % (s->prescale + 1));
102
+ s->counter += y / (s->prescale + 1);
103
+ }
104
+ }
216
+ }
105
+
217
+
106
+ /*
218
+ sve_ldnfff1_r(env, vg, addr, desc, retaddr, mtedesc,
107
+ * Only advance the sync time to the timestamp of the last PSCNTR tick,
219
+ esz, msz, fault, host_fn, tlb_fn);
108
+ * not all the way to 'now', so we don't lose time if we do multiple
109
+ * resyncs in a single tick.
110
+ */
111
+ s->pscntr_sync_ticks += muldiv64(ticks, NANOSECONDS_PER_SECOND,
112
+ s->prescale_clk);
113
+}
220
+}
114
+
221
+
115
static uint64_t mps2_fpgaio_read(void *opaque, hwaddr offset, unsigned size)
222
+#define DO_LDFF1_LDNF1_1(PART, ESZ) \
223
void HELPER(sve_ldff1##PART##_r)(CPUARMState *env, void *vg, \
224
target_ulong addr, uint32_t desc) \
225
{ \
226
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, FAULT_FIRST, \
227
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MO_8, FAULT_FIRST, \
228
sve_ld1##PART##_host, sve_ld1##PART##_tlb); \
229
} \
230
void HELPER(sve_ldnf1##PART##_r)(CPUARMState *env, void *vg, \
231
target_ulong addr, uint32_t desc) \
232
{ \
233
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MO_8, FAULT_NO, \
234
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MO_8, FAULT_NO, \
235
+ sve_ld1##PART##_host, sve_ld1##PART##_tlb); \
236
+} \
237
+void HELPER(sve_ldff1##PART##_r_mte)(CPUARMState *env, void *vg, \
238
+ target_ulong addr, uint32_t desc) \
239
+{ \
240
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MO_8, FAULT_FIRST, \
241
+ sve_ld1##PART##_host, sve_ld1##PART##_tlb); \
242
+} \
243
+void HELPER(sve_ldnf1##PART##_r_mte)(CPUARMState *env, void *vg, \
244
+ target_ulong addr, uint32_t desc) \
245
+{ \
246
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MO_8, FAULT_NO, \
247
sve_ld1##PART##_host, sve_ld1##PART##_tlb); \
248
}
249
250
-#define DO_LDFF1_LDNF1_2(PART, ESZ, MSZ) \
251
+#define DO_LDFF1_LDNF1_2(PART, ESZ, MSZ) \
252
void HELPER(sve_ldff1##PART##_le_r)(CPUARMState *env, void *vg, \
253
target_ulong addr, uint32_t desc) \
254
{ \
255
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_FIRST, \
256
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MSZ, FAULT_FIRST, \
257
sve_ld1##PART##_le_host, sve_ld1##PART##_le_tlb); \
258
} \
259
void HELPER(sve_ldnf1##PART##_le_r)(CPUARMState *env, void *vg, \
260
target_ulong addr, uint32_t desc) \
261
{ \
262
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_NO, \
263
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MSZ, FAULT_NO, \
264
sve_ld1##PART##_le_host, sve_ld1##PART##_le_tlb); \
265
} \
266
void HELPER(sve_ldff1##PART##_be_r)(CPUARMState *env, void *vg, \
267
target_ulong addr, uint32_t desc) \
268
{ \
269
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_FIRST, \
270
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MSZ, FAULT_FIRST, \
271
sve_ld1##PART##_be_host, sve_ld1##PART##_be_tlb); \
272
} \
273
void HELPER(sve_ldnf1##PART##_be_r)(CPUARMState *env, void *vg, \
274
target_ulong addr, uint32_t desc) \
275
{ \
276
- sve_ldnfff1_r(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_NO, \
277
+ sve_ldnfff1_r(env, vg, addr, desc, GETPC(), 0, ESZ, MSZ, FAULT_NO, \
278
sve_ld1##PART##_be_host, sve_ld1##PART##_be_tlb); \
279
+} \
280
+void HELPER(sve_ldff1##PART##_le_r_mte)(CPUARMState *env, void *vg, \
281
+ target_ulong addr, uint32_t desc) \
282
+{ \
283
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_FIRST, \
284
+ sve_ld1##PART##_le_host, sve_ld1##PART##_le_tlb); \
285
+} \
286
+void HELPER(sve_ldnf1##PART##_le_r_mte)(CPUARMState *env, void *vg, \
287
+ target_ulong addr, uint32_t desc) \
288
+{ \
289
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_NO, \
290
+ sve_ld1##PART##_le_host, sve_ld1##PART##_le_tlb); \
291
+} \
292
+void HELPER(sve_ldff1##PART##_be_r_mte)(CPUARMState *env, void *vg, \
293
+ target_ulong addr, uint32_t desc) \
294
+{ \
295
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_FIRST, \
296
+ sve_ld1##PART##_be_host, sve_ld1##PART##_be_tlb); \
297
+} \
298
+void HELPER(sve_ldnf1##PART##_be_r_mte)(CPUARMState *env, void *vg, \
299
+ target_ulong addr, uint32_t desc) \
300
+{ \
301
+ sve_ldnfff1_r_mte(env, vg, addr, desc, GETPC(), ESZ, MSZ, FAULT_NO, \
302
+ sve_ld1##PART##_be_host, sve_ld1##PART##_be_tlb); \
303
}
304
305
DO_LDFF1_LDNF1_1(bb, MO_8)
306
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
307
index XXXXXXX..XXXXXXX 100644
308
--- a/target/arm/translate-sve.c
309
+++ b/target/arm/translate-sve.c
310
@@ -XXX,XX +XXX,XX @@ static bool trans_LD_zpri(DisasContext *s, arg_rpri_load *a)
311
312
static bool trans_LDFF1_zprr(DisasContext *s, arg_rprr_load *a)
116
{
313
{
117
MPS2FPGAIO *s = MPS2_FPGAIO(opaque);
314
- static gen_helper_gvec_mem * const fns[2][16] = {
118
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_fpgaio_read(void *opaque, hwaddr offset, unsigned size)
315
- /* Little-endian */
119
r = counter_from_tickoff(now, s->clk100hz_tick_offset, 100);
316
- { gen_helper_sve_ldff1bb_r,
120
break;
317
- gen_helper_sve_ldff1bhu_r,
121
case A_COUNTER:
318
- gen_helper_sve_ldff1bsu_r,
122
+ resync_counter(s);
319
- gen_helper_sve_ldff1bdu_r,
123
+ r = s->counter;
320
+ static gen_helper_gvec_mem * const fns[2][2][16] = {
124
+ break;
321
+ { /* mte inactive, little-endian */
125
case A_PSCNTR:
322
+ { gen_helper_sve_ldff1bb_r,
126
- qemu_log_mask(LOG_UNIMP, "MPS2 FPGAIO: counters unimplemented\n");
323
+ gen_helper_sve_ldff1bhu_r,
127
- r = 0;
324
+ gen_helper_sve_ldff1bsu_r,
128
+ resync_counter(s);
325
+ gen_helper_sve_ldff1bdu_r,
129
+ r = s->pscntr;
326
130
break;
327
- gen_helper_sve_ldff1sds_le_r,
131
default:
328
- gen_helper_sve_ldff1hh_le_r,
132
qemu_log_mask(LOG_GUEST_ERROR,
329
- gen_helper_sve_ldff1hsu_le_r,
133
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_write(void *opaque, hwaddr offset, uint64_t value,
330
- gen_helper_sve_ldff1hdu_le_r,
134
s->led0 = value & 0x3;
331
+ gen_helper_sve_ldff1sds_le_r,
135
break;
332
+ gen_helper_sve_ldff1hh_le_r,
136
case A_PRESCALE:
333
+ gen_helper_sve_ldff1hsu_le_r,
137
+ resync_counter(s);
334
+ gen_helper_sve_ldff1hdu_le_r,
138
s->prescale = value;
335
139
break;
336
- gen_helper_sve_ldff1hds_le_r,
140
case A_MISC:
337
- gen_helper_sve_ldff1hss_le_r,
141
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_write(void *opaque, hwaddr offset, uint64_t value,
338
- gen_helper_sve_ldff1ss_le_r,
142
now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
339
- gen_helper_sve_ldff1sdu_le_r,
143
s->clk100hz_tick_offset = tickoff_from_counter(now, value, 100);
340
+ gen_helper_sve_ldff1hds_le_r,
144
break;
341
+ gen_helper_sve_ldff1hss_le_r,
145
+ case A_COUNTER:
342
+ gen_helper_sve_ldff1ss_le_r,
146
+ resync_counter(s);
343
+ gen_helper_sve_ldff1sdu_le_r,
147
+ s->counter = value;
344
148
+ break;
345
- gen_helper_sve_ldff1bds_r,
149
+ case A_PSCNTR:
346
- gen_helper_sve_ldff1bss_r,
150
+ resync_counter(s);
347
- gen_helper_sve_ldff1bhs_r,
151
+ s->pscntr = value;
348
- gen_helper_sve_ldff1dd_le_r },
152
+ break;
349
+ gen_helper_sve_ldff1bds_r,
153
default:
350
+ gen_helper_sve_ldff1bss_r,
154
qemu_log_mask(LOG_GUEST_ERROR,
351
+ gen_helper_sve_ldff1bhs_r,
155
"MPS2 FPGAIO write: bad offset 0x%x\n", (int) offset);
352
+ gen_helper_sve_ldff1dd_le_r },
156
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_reset(DeviceState *dev)
353
157
s->misc = 0;
354
- /* Big-endian */
158
s->clk1hz_tick_offset = tickoff_from_counter(now, 0, 1);
355
- { gen_helper_sve_ldff1bb_r,
159
s->clk100hz_tick_offset = tickoff_from_counter(now, 0, 100);
356
- gen_helper_sve_ldff1bhu_r,
160
+ s->counter = 0;
357
- gen_helper_sve_ldff1bsu_r,
161
+ s->pscntr = 0;
358
- gen_helper_sve_ldff1bdu_r,
162
+ s->pscntr_sync_ticks = now;
359
+ /* mte inactive, big-endian */
360
+ { gen_helper_sve_ldff1bb_r,
361
+ gen_helper_sve_ldff1bhu_r,
362
+ gen_helper_sve_ldff1bsu_r,
363
+ gen_helper_sve_ldff1bdu_r,
364
365
- gen_helper_sve_ldff1sds_be_r,
366
- gen_helper_sve_ldff1hh_be_r,
367
- gen_helper_sve_ldff1hsu_be_r,
368
- gen_helper_sve_ldff1hdu_be_r,
369
+ gen_helper_sve_ldff1sds_be_r,
370
+ gen_helper_sve_ldff1hh_be_r,
371
+ gen_helper_sve_ldff1hsu_be_r,
372
+ gen_helper_sve_ldff1hdu_be_r,
373
374
- gen_helper_sve_ldff1hds_be_r,
375
- gen_helper_sve_ldff1hss_be_r,
376
- gen_helper_sve_ldff1ss_be_r,
377
- gen_helper_sve_ldff1sdu_be_r,
378
+ gen_helper_sve_ldff1hds_be_r,
379
+ gen_helper_sve_ldff1hss_be_r,
380
+ gen_helper_sve_ldff1ss_be_r,
381
+ gen_helper_sve_ldff1sdu_be_r,
382
383
- gen_helper_sve_ldff1bds_r,
384
- gen_helper_sve_ldff1bss_r,
385
- gen_helper_sve_ldff1bhs_r,
386
- gen_helper_sve_ldff1dd_be_r },
387
+ gen_helper_sve_ldff1bds_r,
388
+ gen_helper_sve_ldff1bss_r,
389
+ gen_helper_sve_ldff1bhs_r,
390
+ gen_helper_sve_ldff1dd_be_r } },
391
+
392
+ { /* mte active, little-endian */
393
+ { gen_helper_sve_ldff1bb_r_mte,
394
+ gen_helper_sve_ldff1bhu_r_mte,
395
+ gen_helper_sve_ldff1bsu_r_mte,
396
+ gen_helper_sve_ldff1bdu_r_mte,
397
+
398
+ gen_helper_sve_ldff1sds_le_r_mte,
399
+ gen_helper_sve_ldff1hh_le_r_mte,
400
+ gen_helper_sve_ldff1hsu_le_r_mte,
401
+ gen_helper_sve_ldff1hdu_le_r_mte,
402
+
403
+ gen_helper_sve_ldff1hds_le_r_mte,
404
+ gen_helper_sve_ldff1hss_le_r_mte,
405
+ gen_helper_sve_ldff1ss_le_r_mte,
406
+ gen_helper_sve_ldff1sdu_le_r_mte,
407
+
408
+ gen_helper_sve_ldff1bds_r_mte,
409
+ gen_helper_sve_ldff1bss_r_mte,
410
+ gen_helper_sve_ldff1bhs_r_mte,
411
+ gen_helper_sve_ldff1dd_le_r_mte },
412
+
413
+ /* mte active, big-endian */
414
+ { gen_helper_sve_ldff1bb_r_mte,
415
+ gen_helper_sve_ldff1bhu_r_mte,
416
+ gen_helper_sve_ldff1bsu_r_mte,
417
+ gen_helper_sve_ldff1bdu_r_mte,
418
+
419
+ gen_helper_sve_ldff1sds_be_r_mte,
420
+ gen_helper_sve_ldff1hh_be_r_mte,
421
+ gen_helper_sve_ldff1hsu_be_r_mte,
422
+ gen_helper_sve_ldff1hdu_be_r_mte,
423
+
424
+ gen_helper_sve_ldff1hds_be_r_mte,
425
+ gen_helper_sve_ldff1hss_be_r_mte,
426
+ gen_helper_sve_ldff1ss_be_r_mte,
427
+ gen_helper_sve_ldff1sdu_be_r_mte,
428
+
429
+ gen_helper_sve_ldff1bds_r_mte,
430
+ gen_helper_sve_ldff1bss_r_mte,
431
+ gen_helper_sve_ldff1bhs_r_mte,
432
+ gen_helper_sve_ldff1dd_be_r_mte } },
433
};
434
435
if (sve_access_check(s)) {
436
TCGv_i64 addr = new_tmp_a64(s);
437
tcg_gen_shli_i64(addr, cpu_reg(s, a->rm), dtype_msz(a->dtype));
438
tcg_gen_add_i64(addr, addr, cpu_reg_sp(s, a->rn));
439
- do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 0, false,
440
- fns[s->be_data == MO_BE][a->dtype]);
441
+ do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 1, false,
442
+ fns[s->mte_active[0]][s->be_data == MO_BE][a->dtype]);
443
}
444
return true;
163
}
445
}
164
446
165
static void mps2_fpgaio_init(Object *obj)
447
static bool trans_LDNF1_zpri(DisasContext *s, arg_rpri_load *a)
166
@@ -XXX,XX +XXX,XX @@ static bool mps2_fpgaio_counters_needed(void *opaque)
448
{
167
449
- static gen_helper_gvec_mem * const fns[2][16] = {
168
static const VMStateDescription mps2_fpgaio_counters_vmstate = {
450
- /* Little-endian */
169
.name = "mps2-fpgaio/counters",
451
- { gen_helper_sve_ldnf1bb_r,
170
- .version_id = 1,
452
- gen_helper_sve_ldnf1bhu_r,
171
- .minimum_version_id = 1,
453
- gen_helper_sve_ldnf1bsu_r,
172
+ .version_id = 2,
454
- gen_helper_sve_ldnf1bdu_r,
173
+ .minimum_version_id = 2,
455
+ static gen_helper_gvec_mem * const fns[2][2][16] = {
174
.needed = mps2_fpgaio_counters_needed,
456
+ { /* mte inactive, little-endian */
175
.fields = (VMStateField[]) {
457
+ { gen_helper_sve_ldnf1bb_r,
176
VMSTATE_INT64(clk1hz_tick_offset, MPS2FPGAIO),
458
+ gen_helper_sve_ldnf1bhu_r,
177
VMSTATE_INT64(clk100hz_tick_offset, MPS2FPGAIO),
459
+ gen_helper_sve_ldnf1bsu_r,
178
+ VMSTATE_UINT32(counter, MPS2FPGAIO),
460
+ gen_helper_sve_ldnf1bdu_r,
179
+ VMSTATE_UINT32(pscntr, MPS2FPGAIO),
461
180
+ VMSTATE_INT64(pscntr_sync_ticks, MPS2FPGAIO),
462
- gen_helper_sve_ldnf1sds_le_r,
181
VMSTATE_END_OF_LIST()
463
- gen_helper_sve_ldnf1hh_le_r,
464
- gen_helper_sve_ldnf1hsu_le_r,
465
- gen_helper_sve_ldnf1hdu_le_r,
466
+ gen_helper_sve_ldnf1sds_le_r,
467
+ gen_helper_sve_ldnf1hh_le_r,
468
+ gen_helper_sve_ldnf1hsu_le_r,
469
+ gen_helper_sve_ldnf1hdu_le_r,
470
471
- gen_helper_sve_ldnf1hds_le_r,
472
- gen_helper_sve_ldnf1hss_le_r,
473
- gen_helper_sve_ldnf1ss_le_r,
474
- gen_helper_sve_ldnf1sdu_le_r,
475
+ gen_helper_sve_ldnf1hds_le_r,
476
+ gen_helper_sve_ldnf1hss_le_r,
477
+ gen_helper_sve_ldnf1ss_le_r,
478
+ gen_helper_sve_ldnf1sdu_le_r,
479
480
- gen_helper_sve_ldnf1bds_r,
481
- gen_helper_sve_ldnf1bss_r,
482
- gen_helper_sve_ldnf1bhs_r,
483
- gen_helper_sve_ldnf1dd_le_r },
484
+ gen_helper_sve_ldnf1bds_r,
485
+ gen_helper_sve_ldnf1bss_r,
486
+ gen_helper_sve_ldnf1bhs_r,
487
+ gen_helper_sve_ldnf1dd_le_r },
488
489
- /* Big-endian */
490
- { gen_helper_sve_ldnf1bb_r,
491
- gen_helper_sve_ldnf1bhu_r,
492
- gen_helper_sve_ldnf1bsu_r,
493
- gen_helper_sve_ldnf1bdu_r,
494
+ /* mte inactive, big-endian */
495
+ { gen_helper_sve_ldnf1bb_r,
496
+ gen_helper_sve_ldnf1bhu_r,
497
+ gen_helper_sve_ldnf1bsu_r,
498
+ gen_helper_sve_ldnf1bdu_r,
499
500
- gen_helper_sve_ldnf1sds_be_r,
501
- gen_helper_sve_ldnf1hh_be_r,
502
- gen_helper_sve_ldnf1hsu_be_r,
503
- gen_helper_sve_ldnf1hdu_be_r,
504
+ gen_helper_sve_ldnf1sds_be_r,
505
+ gen_helper_sve_ldnf1hh_be_r,
506
+ gen_helper_sve_ldnf1hsu_be_r,
507
+ gen_helper_sve_ldnf1hdu_be_r,
508
509
- gen_helper_sve_ldnf1hds_be_r,
510
- gen_helper_sve_ldnf1hss_be_r,
511
- gen_helper_sve_ldnf1ss_be_r,
512
- gen_helper_sve_ldnf1sdu_be_r,
513
+ gen_helper_sve_ldnf1hds_be_r,
514
+ gen_helper_sve_ldnf1hss_be_r,
515
+ gen_helper_sve_ldnf1ss_be_r,
516
+ gen_helper_sve_ldnf1sdu_be_r,
517
518
- gen_helper_sve_ldnf1bds_r,
519
- gen_helper_sve_ldnf1bss_r,
520
- gen_helper_sve_ldnf1bhs_r,
521
- gen_helper_sve_ldnf1dd_be_r },
522
+ gen_helper_sve_ldnf1bds_r,
523
+ gen_helper_sve_ldnf1bss_r,
524
+ gen_helper_sve_ldnf1bhs_r,
525
+ gen_helper_sve_ldnf1dd_be_r } },
526
+
527
+ { /* mte inactive, little-endian */
528
+ { gen_helper_sve_ldnf1bb_r_mte,
529
+ gen_helper_sve_ldnf1bhu_r_mte,
530
+ gen_helper_sve_ldnf1bsu_r_mte,
531
+ gen_helper_sve_ldnf1bdu_r_mte,
532
+
533
+ gen_helper_sve_ldnf1sds_le_r_mte,
534
+ gen_helper_sve_ldnf1hh_le_r_mte,
535
+ gen_helper_sve_ldnf1hsu_le_r_mte,
536
+ gen_helper_sve_ldnf1hdu_le_r_mte,
537
+
538
+ gen_helper_sve_ldnf1hds_le_r_mte,
539
+ gen_helper_sve_ldnf1hss_le_r_mte,
540
+ gen_helper_sve_ldnf1ss_le_r_mte,
541
+ gen_helper_sve_ldnf1sdu_le_r_mte,
542
+
543
+ gen_helper_sve_ldnf1bds_r_mte,
544
+ gen_helper_sve_ldnf1bss_r_mte,
545
+ gen_helper_sve_ldnf1bhs_r_mte,
546
+ gen_helper_sve_ldnf1dd_le_r_mte },
547
+
548
+ /* mte inactive, big-endian */
549
+ { gen_helper_sve_ldnf1bb_r_mte,
550
+ gen_helper_sve_ldnf1bhu_r_mte,
551
+ gen_helper_sve_ldnf1bsu_r_mte,
552
+ gen_helper_sve_ldnf1bdu_r_mte,
553
+
554
+ gen_helper_sve_ldnf1sds_be_r_mte,
555
+ gen_helper_sve_ldnf1hh_be_r_mte,
556
+ gen_helper_sve_ldnf1hsu_be_r_mte,
557
+ gen_helper_sve_ldnf1hdu_be_r_mte,
558
+
559
+ gen_helper_sve_ldnf1hds_be_r_mte,
560
+ gen_helper_sve_ldnf1hss_be_r_mte,
561
+ gen_helper_sve_ldnf1ss_be_r_mte,
562
+ gen_helper_sve_ldnf1sdu_be_r_mte,
563
+
564
+ gen_helper_sve_ldnf1bds_r_mte,
565
+ gen_helper_sve_ldnf1bss_r_mte,
566
+ gen_helper_sve_ldnf1bhs_r_mte,
567
+ gen_helper_sve_ldnf1dd_be_r_mte } },
568
};
569
570
if (sve_access_check(s)) {
571
@@ -XXX,XX +XXX,XX @@ static bool trans_LDNF1_zpri(DisasContext *s, arg_rpri_load *a)
572
TCGv_i64 addr = new_tmp_a64(s);
573
574
tcg_gen_addi_i64(addr, cpu_reg_sp(s, a->rn), off);
575
- do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 0, false,
576
- fns[s->be_data == MO_BE][a->dtype]);
577
+ do_mem_zpa(s, a->rd, a->pg, addr, a->dtype, 1, false,
578
+ fns[s->mte_active[0]][s->be_data == MO_BE][a->dtype]);
182
}
579
}
183
};
580
return true;
581
}
184
--
582
--
185
2.18.0
583
2.20.1
186
584
187
585
diff view generated by jsdifflib
1
The IoTKit doesn't have any MSCs itself but it does need
1
From: Richard Henderson <richard.henderson@linaro.org>
2
some wiring to connect the external signals from MSCs
3
in the outer board model up to the registers and the
4
NVIC IRQ line.
5
2
6
We also need to expose a MemoryRegion corresponding to
3
We still need to handle tbi for user-only when mte is inactive.
7
the AHB bus, so that MSCs in the outer board model can
8
use that as their downstream port. (In the FPGA this is
9
the "AHB Slave Expansion" ports shown in the block
10
diagram in the AN505 documentation.)
11
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-37-richard.henderson@linaro.org
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
14
Message-id: 20180820141116.9118-14-peter.maydell@linaro.org
15
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
16
---
9
---
17
include/hw/arm/iotkit.h | 8 ++++++++
10
target/arm/translate-a64.h | 1 +
18
hw/arm/iotkit.c | 15 +++++++++++++++
11
target/arm/translate-a64.c | 2 +-
19
2 files changed, 23 insertions(+)
12
target/arm/translate-sve.c | 6 ++++--
13
3 files changed, 6 insertions(+), 3 deletions(-)
20
14
21
diff --git a/include/hw/arm/iotkit.h b/include/hw/arm/iotkit.h
15
diff --git a/target/arm/translate-a64.h b/target/arm/translate-a64.h
22
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
23
--- a/include/hw/arm/iotkit.h
17
--- a/target/arm/translate-a64.h
24
+++ b/include/hw/arm/iotkit.h
18
+++ b/target/arm/translate-a64.h
25
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ TCGv_ptr get_fpstatus_ptr(bool);
26
* + QOM property "EXP_NUMIRQ" sets the number of expansion interrupts
20
bool logic_imm_decode_wmask(uint64_t *result, unsigned int immn,
27
* + Named GPIO inputs "EXP_IRQ" 0..n are the expansion interrupts, which
21
unsigned int imms, unsigned int immr);
28
* are wired to the NVIC lines 32 .. n+32
22
bool sve_access_check(DisasContext *s);
29
+ * + sysbus MMIO region 0 is the "AHB Slave Expansion" which allows
23
+TCGv_i64 clean_data_tbi(DisasContext *s, TCGv_i64 addr);
30
+ * bus master devices in the board model to make transactions into
24
TCGv_i64 gen_mte_check1(DisasContext *s, TCGv_i64 addr, bool is_write,
31
+ * all the devices and memory areas in the IoTKit
25
bool tag_checked, int log2_size);
32
* Controlling up to 4 AHB expansion PPBs which a system using the IoTKit
26
TCGv_i64 gen_mte_checkN(DisasContext *s, TCGv_i64 addr, bool is_write,
33
* might provide:
27
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
34
* + named GPIO outputs apb_ppcexp{0,1,2,3}_nonsec[0..15]
28
index XXXXXXX..XXXXXXX 100644
35
@@ -XXX,XX +XXX,XX @@
29
--- a/target/arm/translate-a64.c
36
* Controlling each of the 16 expansion MPCs which a system using the IoTKit
30
+++ b/target/arm/translate-a64.c
37
* might provide:
31
@@ -XXX,XX +XXX,XX @@ static void gen_a64_set_pc(DisasContext *s, TCGv_i64 src)
38
* + named GPIO inputs mpcexp_status[0..15]
32
* of the write-back address.
39
+ * Controlling each of the 16 expansion MSCs which a system using the IoTKit
40
+ * might provide:
41
+ * + named GPIO inputs mscexp_status[0..15]
42
+ * + named GPIO outputs mscexp_clear[0..15]
43
+ * + named GPIO outputs mscexp_ns[0..15]
44
*/
33
*/
45
34
46
#ifndef IOTKIT_H
35
-static TCGv_i64 clean_data_tbi(DisasContext *s, TCGv_i64 addr)
47
diff --git a/hw/arm/iotkit.c b/hw/arm/iotkit.c
36
+TCGv_i64 clean_data_tbi(DisasContext *s, TCGv_i64 addr)
37
{
38
TCGv_i64 clean = new_tmp_a64(s);
39
#ifdef CONFIG_USER_ONLY
40
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
48
index XXXXXXX..XXXXXXX 100644
41
index XXXXXXX..XXXXXXX 100644
49
--- a/hw/arm/iotkit.c
42
--- a/target/arm/translate-sve.c
50
+++ b/hw/arm/iotkit.c
43
+++ b/target/arm/translate-sve.c
51
@@ -XXX,XX +XXX,XX @@ static void iotkit_realize(DeviceState *dev, Error **errp)
44
@@ -XXX,XX +XXX,XX @@ static void do_mem_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
52
45
* For e.g. LD4, there are not enough arguments to pass all 4
53
iotkit_forward_sec_resp_cfg(s);
46
* registers as pointers, so encode the regno into the data field.
54
47
* For consistency, do this even for LD1.
55
+ /* Forward the MSC related signals */
48
- * TODO: mte_n check here while callers are updated.
56
+ qdev_pass_gpios(dev_secctl, dev, "mscexp_status");
49
*/
57
+ qdev_pass_gpios(dev_secctl, dev, "mscexp_clear");
50
- if (mte_n && s->mte_active[0]) {
58
+ qdev_pass_gpios(dev_secctl, dev, "mscexp_ns");
51
+ if (s->mte_active[0]) {
59
+ qdev_connect_gpio_out_named(dev_secctl, "msc_irq", 0,
52
int msz = dtype_msz(dtype);
60
+ qdev_get_gpio_in(DEVICE(&s->armv7m), 11));
53
54
desc = FIELD_DP32(desc, MTEDESC, MIDX, get_mem_index(s));
55
@@ -XXX,XX +XXX,XX @@ static void do_mem_zpa(DisasContext *s, int zt, int pg, TCGv_i64 addr,
56
desc = FIELD_DP32(desc, MTEDESC, ESIZE, 1 << msz);
57
desc = FIELD_DP32(desc, MTEDESC, TSIZE, mte_n << msz);
58
desc <<= SVE_MTEDESC_SHIFT;
59
+ } else {
60
+ addr = clean_data_tbi(s, addr);
61
}
61
+
62
+
62
+ /*
63
desc = simd_desc(vsz, vsz, zt | desc);
63
+ * Expose our container region to the board model; this corresponds
64
t_desc = tcg_const_i32(desc);
64
+ * to the AHB Slave Expansion ports which allow bus master devices
65
t_pg = tcg_temp_new_ptr();
65
+ * (eg DMA controllers) in the board model to make transactions into
66
+ * devices in the IoTKit.
67
+ */
68
+ sysbus_init_mmio(SYS_BUS_DEVICE(s), &s->container);
69
+
70
system_clock_scale = NANOSECONDS_PER_SECOND / s->mainclk_frq;
71
}
72
73
--
66
--
74
2.18.0
67
2.20.1
75
68
76
69
diff view generated by jsdifflib
1
The AN505 FPGA image includes four PL081 DMA controllers, each
1
From: Richard Henderson <richard.henderson@linaro.org>
2
of which is gated by a Master Security Controller that allows
3
the guest to prevent a non-secure DMA controller from accessing
4
memory that is used by secure guest code. Create and wire
5
up these devices.
6
2
3
Because the elements are non-sequential, we cannot eliminate many
4
tests straight away like we can for sequential operations. But
5
we often have the PTE details handy, so we can test for Tagged.
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-38-richard.henderson@linaro.org
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Message-id: 20180820141116.9118-15-peter.maydell@linaro.org
10
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
11
---
11
---
12
hw/arm/mps2-tz.c | 100 +++++++++++++++++++++++++++++++++++++++++++----
12
target/arm/helper-sve.h | 285 ++++++++++++++++
13
1 file changed, 93 insertions(+), 7 deletions(-)
13
target/arm/sve_helper.c | 185 +++++++++--
14
target/arm/translate-sve.c | 650 +++++++++++++++++++++++++------------
15
3 files changed, 872 insertions(+), 248 deletions(-)
14
16
15
diff --git a/hw/arm/mps2-tz.c b/hw/arm/mps2-tz.c
17
diff --git a/target/arm/helper-sve.h b/target/arm/helper-sve.h
16
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/arm/mps2-tz.c
19
--- a/target/arm/helper-sve.h
18
+++ b/hw/arm/mps2-tz.c
20
+++ b/target/arm/helper-sve.h
19
@@ -XXX,XX +XXX,XX @@
21
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_6(sve_ldsds_le_zd, TCG_CALL_NO_WG,
20
#include "hw/misc/mps2-scc.h"
22
DEF_HELPER_FLAGS_6(sve_ldsds_be_zd, TCG_CALL_NO_WG,
21
#include "hw/misc/mps2-fpgaio.h"
23
void, env, ptr, ptr, ptr, tl, i32)
22
#include "hw/misc/tz-mpc.h"
24
23
+#include "hw/misc/tz-msc.h"
25
+DEF_HELPER_FLAGS_6(sve_ldbsu_zsu_mte, TCG_CALL_NO_WG,
24
#include "hw/arm/iotkit.h"
26
+ void, env, ptr, ptr, ptr, tl, i32)
25
+#include "hw/dma/pl080.h"
27
+DEF_HELPER_FLAGS_6(sve_ldhsu_le_zsu_mte, TCG_CALL_NO_WG,
26
#include "hw/devices.h"
28
+ void, env, ptr, ptr, ptr, tl, i32)
27
#include "net/net.h"
29
+DEF_HELPER_FLAGS_6(sve_ldhsu_be_zsu_mte, TCG_CALL_NO_WG,
28
#include "hw/core/split-irq.h"
30
+ void, env, ptr, ptr, ptr, tl, i32)
29
@@ -XXX,XX +XXX,XX @@ typedef struct {
31
+DEF_HELPER_FLAGS_6(sve_ldss_le_zsu_mte, TCG_CALL_NO_WG,
30
UnimplementedDeviceState i2c[4];
32
+ void, env, ptr, ptr, ptr, tl, i32)
31
UnimplementedDeviceState i2s_audio;
33
+DEF_HELPER_FLAGS_6(sve_ldss_be_zsu_mte, TCG_CALL_NO_WG,
32
UnimplementedDeviceState gpio[4];
34
+ void, env, ptr, ptr, ptr, tl, i32)
33
- UnimplementedDeviceState dma[4];
35
+DEF_HELPER_FLAGS_6(sve_ldbss_zsu_mte, TCG_CALL_NO_WG,
34
UnimplementedDeviceState gfx;
36
+ void, env, ptr, ptr, ptr, tl, i32)
35
+ PL080State dma[4];
37
+DEF_HELPER_FLAGS_6(sve_ldhss_le_zsu_mte, TCG_CALL_NO_WG,
36
+ TZMSC msc[4];
38
+ void, env, ptr, ptr, ptr, tl, i32)
37
CMSDKAPBUART uart[5];
39
+DEF_HELPER_FLAGS_6(sve_ldhss_be_zsu_mte, TCG_CALL_NO_WG,
38
SplitIRQ sec_resp_splitter;
40
+ void, env, ptr, ptr, ptr, tl, i32)
39
qemu_or_irq uart_irq_orgate;
41
+
40
@@ -XXX,XX +XXX,XX @@ static MemoryRegion *make_mpc(MPS2TZMachineState *mms, void *opaque,
42
+DEF_HELPER_FLAGS_6(sve_ldbsu_zss_mte, TCG_CALL_NO_WG,
41
return sysbus_mmio_get_region(SYS_BUS_DEVICE(mpc), 0);
43
+ void, env, ptr, ptr, ptr, tl, i32)
44
+DEF_HELPER_FLAGS_6(sve_ldhsu_le_zss_mte, TCG_CALL_NO_WG,
45
+ void, env, ptr, ptr, ptr, tl, i32)
46
+DEF_HELPER_FLAGS_6(sve_ldhsu_be_zss_mte, TCG_CALL_NO_WG,
47
+ void, env, ptr, ptr, ptr, tl, i32)
48
+DEF_HELPER_FLAGS_6(sve_ldss_le_zss_mte, TCG_CALL_NO_WG,
49
+ void, env, ptr, ptr, ptr, tl, i32)
50
+DEF_HELPER_FLAGS_6(sve_ldss_be_zss_mte, TCG_CALL_NO_WG,
51
+ void, env, ptr, ptr, ptr, tl, i32)
52
+DEF_HELPER_FLAGS_6(sve_ldbss_zss_mte, TCG_CALL_NO_WG,
53
+ void, env, ptr, ptr, ptr, tl, i32)
54
+DEF_HELPER_FLAGS_6(sve_ldhss_le_zss_mte, TCG_CALL_NO_WG,
55
+ void, env, ptr, ptr, ptr, tl, i32)
56
+DEF_HELPER_FLAGS_6(sve_ldhss_be_zss_mte, TCG_CALL_NO_WG,
57
+ void, env, ptr, ptr, ptr, tl, i32)
58
+
59
+DEF_HELPER_FLAGS_6(sve_ldbdu_zsu_mte, TCG_CALL_NO_WG,
60
+ void, env, ptr, ptr, ptr, tl, i32)
61
+DEF_HELPER_FLAGS_6(sve_ldhdu_le_zsu_mte, TCG_CALL_NO_WG,
62
+ void, env, ptr, ptr, ptr, tl, i32)
63
+DEF_HELPER_FLAGS_6(sve_ldhdu_be_zsu_mte, TCG_CALL_NO_WG,
64
+ void, env, ptr, ptr, ptr, tl, i32)
65
+DEF_HELPER_FLAGS_6(sve_ldsdu_le_zsu_mte, TCG_CALL_NO_WG,
66
+ void, env, ptr, ptr, ptr, tl, i32)
67
+DEF_HELPER_FLAGS_6(sve_ldsdu_be_zsu_mte, TCG_CALL_NO_WG,
68
+ void, env, ptr, ptr, ptr, tl, i32)
69
+DEF_HELPER_FLAGS_6(sve_lddd_le_zsu_mte, TCG_CALL_NO_WG,
70
+ void, env, ptr, ptr, ptr, tl, i32)
71
+DEF_HELPER_FLAGS_6(sve_lddd_be_zsu_mte, TCG_CALL_NO_WG,
72
+ void, env, ptr, ptr, ptr, tl, i32)
73
+DEF_HELPER_FLAGS_6(sve_ldbds_zsu_mte, TCG_CALL_NO_WG,
74
+ void, env, ptr, ptr, ptr, tl, i32)
75
+DEF_HELPER_FLAGS_6(sve_ldhds_le_zsu_mte, TCG_CALL_NO_WG,
76
+ void, env, ptr, ptr, ptr, tl, i32)
77
+DEF_HELPER_FLAGS_6(sve_ldhds_be_zsu_mte, TCG_CALL_NO_WG,
78
+ void, env, ptr, ptr, ptr, tl, i32)
79
+DEF_HELPER_FLAGS_6(sve_ldsds_le_zsu_mte, TCG_CALL_NO_WG,
80
+ void, env, ptr, ptr, ptr, tl, i32)
81
+DEF_HELPER_FLAGS_6(sve_ldsds_be_zsu_mte, TCG_CALL_NO_WG,
82
+ void, env, ptr, ptr, ptr, tl, i32)
83
+
84
+DEF_HELPER_FLAGS_6(sve_ldbdu_zss_mte, TCG_CALL_NO_WG,
85
+ void, env, ptr, ptr, ptr, tl, i32)
86
+DEF_HELPER_FLAGS_6(sve_ldhdu_le_zss_mte, TCG_CALL_NO_WG,
87
+ void, env, ptr, ptr, ptr, tl, i32)
88
+DEF_HELPER_FLAGS_6(sve_ldhdu_be_zss_mte, TCG_CALL_NO_WG,
89
+ void, env, ptr, ptr, ptr, tl, i32)
90
+DEF_HELPER_FLAGS_6(sve_ldsdu_le_zss_mte, TCG_CALL_NO_WG,
91
+ void, env, ptr, ptr, ptr, tl, i32)
92
+DEF_HELPER_FLAGS_6(sve_ldsdu_be_zss_mte, TCG_CALL_NO_WG,
93
+ void, env, ptr, ptr, ptr, tl, i32)
94
+DEF_HELPER_FLAGS_6(sve_lddd_le_zss_mte, TCG_CALL_NO_WG,
95
+ void, env, ptr, ptr, ptr, tl, i32)
96
+DEF_HELPER_FLAGS_6(sve_lddd_be_zss_mte, TCG_CALL_NO_WG,
97
+ void, env, ptr, ptr, ptr, tl, i32)
98
+DEF_HELPER_FLAGS_6(sve_ldbds_zss_mte, TCG_CALL_NO_WG,
99
+ void, env, ptr, ptr, ptr, tl, i32)
100
+DEF_HELPER_FLAGS_6(sve_ldhds_le_zss_mte, TCG_CALL_NO_WG,
101
+ void, env, ptr, ptr, ptr, tl, i32)
102
+DEF_HELPER_FLAGS_6(sve_ldhds_be_zss_mte, TCG_CALL_NO_WG,
103
+ void, env, ptr, ptr, ptr, tl, i32)
104
+DEF_HELPER_FLAGS_6(sve_ldsds_le_zss_mte, TCG_CALL_NO_WG,
105
+ void, env, ptr, ptr, ptr, tl, i32)
106
+DEF_HELPER_FLAGS_6(sve_ldsds_be_zss_mte, TCG_CALL_NO_WG,
107
+ void, env, ptr, ptr, ptr, tl, i32)
108
+
109
+DEF_HELPER_FLAGS_6(sve_ldbdu_zd_mte, TCG_CALL_NO_WG,
110
+ void, env, ptr, ptr, ptr, tl, i32)
111
+DEF_HELPER_FLAGS_6(sve_ldhdu_le_zd_mte, TCG_CALL_NO_WG,
112
+ void, env, ptr, ptr, ptr, tl, i32)
113
+DEF_HELPER_FLAGS_6(sve_ldhdu_be_zd_mte, TCG_CALL_NO_WG,
114
+ void, env, ptr, ptr, ptr, tl, i32)
115
+DEF_HELPER_FLAGS_6(sve_ldsdu_le_zd_mte, TCG_CALL_NO_WG,
116
+ void, env, ptr, ptr, ptr, tl, i32)
117
+DEF_HELPER_FLAGS_6(sve_ldsdu_be_zd_mte, TCG_CALL_NO_WG,
118
+ void, env, ptr, ptr, ptr, tl, i32)
119
+DEF_HELPER_FLAGS_6(sve_lddd_le_zd_mte, TCG_CALL_NO_WG,
120
+ void, env, ptr, ptr, ptr, tl, i32)
121
+DEF_HELPER_FLAGS_6(sve_lddd_be_zd_mte, TCG_CALL_NO_WG,
122
+ void, env, ptr, ptr, ptr, tl, i32)
123
+DEF_HELPER_FLAGS_6(sve_ldbds_zd_mte, TCG_CALL_NO_WG,
124
+ void, env, ptr, ptr, ptr, tl, i32)
125
+DEF_HELPER_FLAGS_6(sve_ldhds_le_zd_mte, TCG_CALL_NO_WG,
126
+ void, env, ptr, ptr, ptr, tl, i32)
127
+DEF_HELPER_FLAGS_6(sve_ldhds_be_zd_mte, TCG_CALL_NO_WG,
128
+ void, env, ptr, ptr, ptr, tl, i32)
129
+DEF_HELPER_FLAGS_6(sve_ldsds_le_zd_mte, TCG_CALL_NO_WG,
130
+ void, env, ptr, ptr, ptr, tl, i32)
131
+DEF_HELPER_FLAGS_6(sve_ldsds_be_zd_mte, TCG_CALL_NO_WG,
132
+ void, env, ptr, ptr, ptr, tl, i32)
133
+
134
DEF_HELPER_FLAGS_6(sve_ldffbsu_zsu, TCG_CALL_NO_WG,
135
void, env, ptr, ptr, ptr, tl, i32)
136
DEF_HELPER_FLAGS_6(sve_ldffhsu_le_zsu, TCG_CALL_NO_WG,
137
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_6(sve_ldffsds_le_zd, TCG_CALL_NO_WG,
138
DEF_HELPER_FLAGS_6(sve_ldffsds_be_zd, TCG_CALL_NO_WG,
139
void, env, ptr, ptr, ptr, tl, i32)
140
141
+DEF_HELPER_FLAGS_6(sve_ldffbsu_zsu_mte, TCG_CALL_NO_WG,
142
+ void, env, ptr, ptr, ptr, tl, i32)
143
+DEF_HELPER_FLAGS_6(sve_ldffhsu_le_zsu_mte, TCG_CALL_NO_WG,
144
+ void, env, ptr, ptr, ptr, tl, i32)
145
+DEF_HELPER_FLAGS_6(sve_ldffhsu_be_zsu_mte, TCG_CALL_NO_WG,
146
+ void, env, ptr, ptr, ptr, tl, i32)
147
+DEF_HELPER_FLAGS_6(sve_ldffss_le_zsu_mte, TCG_CALL_NO_WG,
148
+ void, env, ptr, ptr, ptr, tl, i32)
149
+DEF_HELPER_FLAGS_6(sve_ldffss_be_zsu_mte, TCG_CALL_NO_WG,
150
+ void, env, ptr, ptr, ptr, tl, i32)
151
+DEF_HELPER_FLAGS_6(sve_ldffbss_zsu_mte, TCG_CALL_NO_WG,
152
+ void, env, ptr, ptr, ptr, tl, i32)
153
+DEF_HELPER_FLAGS_6(sve_ldffhss_le_zsu_mte, TCG_CALL_NO_WG,
154
+ void, env, ptr, ptr, ptr, tl, i32)
155
+DEF_HELPER_FLAGS_6(sve_ldffhss_be_zsu_mte, TCG_CALL_NO_WG,
156
+ void, env, ptr, ptr, ptr, tl, i32)
157
+
158
+DEF_HELPER_FLAGS_6(sve_ldffbsu_zss_mte, TCG_CALL_NO_WG,
159
+ void, env, ptr, ptr, ptr, tl, i32)
160
+DEF_HELPER_FLAGS_6(sve_ldffhsu_le_zss_mte, TCG_CALL_NO_WG,
161
+ void, env, ptr, ptr, ptr, tl, i32)
162
+DEF_HELPER_FLAGS_6(sve_ldffhsu_be_zss_mte, TCG_CALL_NO_WG,
163
+ void, env, ptr, ptr, ptr, tl, i32)
164
+DEF_HELPER_FLAGS_6(sve_ldffss_le_zss_mte, TCG_CALL_NO_WG,
165
+ void, env, ptr, ptr, ptr, tl, i32)
166
+DEF_HELPER_FLAGS_6(sve_ldffss_be_zss_mte, TCG_CALL_NO_WG,
167
+ void, env, ptr, ptr, ptr, tl, i32)
168
+DEF_HELPER_FLAGS_6(sve_ldffbss_zss_mte, TCG_CALL_NO_WG,
169
+ void, env, ptr, ptr, ptr, tl, i32)
170
+DEF_HELPER_FLAGS_6(sve_ldffhss_le_zss_mte, TCG_CALL_NO_WG,
171
+ void, env, ptr, ptr, ptr, tl, i32)
172
+DEF_HELPER_FLAGS_6(sve_ldffhss_be_zss_mte, TCG_CALL_NO_WG,
173
+ void, env, ptr, ptr, ptr, tl, i32)
174
+
175
+DEF_HELPER_FLAGS_6(sve_ldffbdu_zsu_mte, TCG_CALL_NO_WG,
176
+ void, env, ptr, ptr, ptr, tl, i32)
177
+DEF_HELPER_FLAGS_6(sve_ldffhdu_le_zsu_mte, TCG_CALL_NO_WG,
178
+ void, env, ptr, ptr, ptr, tl, i32)
179
+DEF_HELPER_FLAGS_6(sve_ldffhdu_be_zsu_mte, TCG_CALL_NO_WG,
180
+ void, env, ptr, ptr, ptr, tl, i32)
181
+DEF_HELPER_FLAGS_6(sve_ldffsdu_le_zsu_mte, TCG_CALL_NO_WG,
182
+ void, env, ptr, ptr, ptr, tl, i32)
183
+DEF_HELPER_FLAGS_6(sve_ldffsdu_be_zsu_mte, TCG_CALL_NO_WG,
184
+ void, env, ptr, ptr, ptr, tl, i32)
185
+DEF_HELPER_FLAGS_6(sve_ldffdd_le_zsu_mte, TCG_CALL_NO_WG,
186
+ void, env, ptr, ptr, ptr, tl, i32)
187
+DEF_HELPER_FLAGS_6(sve_ldffdd_be_zsu_mte, TCG_CALL_NO_WG,
188
+ void, env, ptr, ptr, ptr, tl, i32)
189
+DEF_HELPER_FLAGS_6(sve_ldffbds_zsu_mte, TCG_CALL_NO_WG,
190
+ void, env, ptr, ptr, ptr, tl, i32)
191
+DEF_HELPER_FLAGS_6(sve_ldffhds_le_zsu_mte, TCG_CALL_NO_WG,
192
+ void, env, ptr, ptr, ptr, tl, i32)
193
+DEF_HELPER_FLAGS_6(sve_ldffhds_be_zsu_mte, TCG_CALL_NO_WG,
194
+ void, env, ptr, ptr, ptr, tl, i32)
195
+DEF_HELPER_FLAGS_6(sve_ldffsds_le_zsu_mte, TCG_CALL_NO_WG,
196
+ void, env, ptr, ptr, ptr, tl, i32)
197
+DEF_HELPER_FLAGS_6(sve_ldffsds_be_zsu_mte, TCG_CALL_NO_WG,
198
+ void, env, ptr, ptr, ptr, tl, i32)
199
+
200
+DEF_HELPER_FLAGS_6(sve_ldffbdu_zss_mte, TCG_CALL_NO_WG,
201
+ void, env, ptr, ptr, ptr, tl, i32)
202
+DEF_HELPER_FLAGS_6(sve_ldffhdu_le_zss_mte, TCG_CALL_NO_WG,
203
+ void, env, ptr, ptr, ptr, tl, i32)
204
+DEF_HELPER_FLAGS_6(sve_ldffhdu_be_zss_mte, TCG_CALL_NO_WG,
205
+ void, env, ptr, ptr, ptr, tl, i32)
206
+DEF_HELPER_FLAGS_6(sve_ldffsdu_le_zss_mte, TCG_CALL_NO_WG,
207
+ void, env, ptr, ptr, ptr, tl, i32)
208
+DEF_HELPER_FLAGS_6(sve_ldffsdu_be_zss_mte, TCG_CALL_NO_WG,
209
+ void, env, ptr, ptr, ptr, tl, i32)
210
+DEF_HELPER_FLAGS_6(sve_ldffdd_le_zss_mte, TCG_CALL_NO_WG,
211
+ void, env, ptr, ptr, ptr, tl, i32)
212
+DEF_HELPER_FLAGS_6(sve_ldffdd_be_zss_mte, TCG_CALL_NO_WG,
213
+ void, env, ptr, ptr, ptr, tl, i32)
214
+DEF_HELPER_FLAGS_6(sve_ldffbds_zss_mte, TCG_CALL_NO_WG,
215
+ void, env, ptr, ptr, ptr, tl, i32)
216
+DEF_HELPER_FLAGS_6(sve_ldffhds_le_zss_mte, TCG_CALL_NO_WG,
217
+ void, env, ptr, ptr, ptr, tl, i32)
218
+DEF_HELPER_FLAGS_6(sve_ldffhds_be_zss_mte, TCG_CALL_NO_WG,
219
+ void, env, ptr, ptr, ptr, tl, i32)
220
+DEF_HELPER_FLAGS_6(sve_ldffsds_le_zss_mte, TCG_CALL_NO_WG,
221
+ void, env, ptr, ptr, ptr, tl, i32)
222
+DEF_HELPER_FLAGS_6(sve_ldffsds_be_zss_mte, TCG_CALL_NO_WG,
223
+ void, env, ptr, ptr, ptr, tl, i32)
224
+
225
+DEF_HELPER_FLAGS_6(sve_ldffbdu_zd_mte, TCG_CALL_NO_WG,
226
+ void, env, ptr, ptr, ptr, tl, i32)
227
+DEF_HELPER_FLAGS_6(sve_ldffhdu_le_zd_mte, TCG_CALL_NO_WG,
228
+ void, env, ptr, ptr, ptr, tl, i32)
229
+DEF_HELPER_FLAGS_6(sve_ldffhdu_be_zd_mte, TCG_CALL_NO_WG,
230
+ void, env, ptr, ptr, ptr, tl, i32)
231
+DEF_HELPER_FLAGS_6(sve_ldffsdu_le_zd_mte, TCG_CALL_NO_WG,
232
+ void, env, ptr, ptr, ptr, tl, i32)
233
+DEF_HELPER_FLAGS_6(sve_ldffsdu_be_zd_mte, TCG_CALL_NO_WG,
234
+ void, env, ptr, ptr, ptr, tl, i32)
235
+DEF_HELPER_FLAGS_6(sve_ldffdd_le_zd_mte, TCG_CALL_NO_WG,
236
+ void, env, ptr, ptr, ptr, tl, i32)
237
+DEF_HELPER_FLAGS_6(sve_ldffdd_be_zd_mte, TCG_CALL_NO_WG,
238
+ void, env, ptr, ptr, ptr, tl, i32)
239
+DEF_HELPER_FLAGS_6(sve_ldffbds_zd_mte, TCG_CALL_NO_WG,
240
+ void, env, ptr, ptr, ptr, tl, i32)
241
+DEF_HELPER_FLAGS_6(sve_ldffhds_le_zd_mte, TCG_CALL_NO_WG,
242
+ void, env, ptr, ptr, ptr, tl, i32)
243
+DEF_HELPER_FLAGS_6(sve_ldffhds_be_zd_mte, TCG_CALL_NO_WG,
244
+ void, env, ptr, ptr, ptr, tl, i32)
245
+DEF_HELPER_FLAGS_6(sve_ldffsds_le_zd_mte, TCG_CALL_NO_WG,
246
+ void, env, ptr, ptr, ptr, tl, i32)
247
+DEF_HELPER_FLAGS_6(sve_ldffsds_be_zd_mte, TCG_CALL_NO_WG,
248
+ void, env, ptr, ptr, ptr, tl, i32)
249
+
250
DEF_HELPER_FLAGS_6(sve_stbs_zsu, TCG_CALL_NO_WG,
251
void, env, ptr, ptr, ptr, tl, i32)
252
DEF_HELPER_FLAGS_6(sve_sths_le_zsu, TCG_CALL_NO_WG,
253
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_6(sve_stdd_le_zd, TCG_CALL_NO_WG,
254
DEF_HELPER_FLAGS_6(sve_stdd_be_zd, TCG_CALL_NO_WG,
255
void, env, ptr, ptr, ptr, tl, i32)
256
257
+DEF_HELPER_FLAGS_6(sve_stbs_zsu_mte, TCG_CALL_NO_WG,
258
+ void, env, ptr, ptr, ptr, tl, i32)
259
+DEF_HELPER_FLAGS_6(sve_sths_le_zsu_mte, TCG_CALL_NO_WG,
260
+ void, env, ptr, ptr, ptr, tl, i32)
261
+DEF_HELPER_FLAGS_6(sve_sths_be_zsu_mte, TCG_CALL_NO_WG,
262
+ void, env, ptr, ptr, ptr, tl, i32)
263
+DEF_HELPER_FLAGS_6(sve_stss_le_zsu_mte, TCG_CALL_NO_WG,
264
+ void, env, ptr, ptr, ptr, tl, i32)
265
+DEF_HELPER_FLAGS_6(sve_stss_be_zsu_mte, TCG_CALL_NO_WG,
266
+ void, env, ptr, ptr, ptr, tl, i32)
267
+
268
+DEF_HELPER_FLAGS_6(sve_stbs_zss_mte, TCG_CALL_NO_WG,
269
+ void, env, ptr, ptr, ptr, tl, i32)
270
+DEF_HELPER_FLAGS_6(sve_sths_le_zss_mte, TCG_CALL_NO_WG,
271
+ void, env, ptr, ptr, ptr, tl, i32)
272
+DEF_HELPER_FLAGS_6(sve_sths_be_zss_mte, TCG_CALL_NO_WG,
273
+ void, env, ptr, ptr, ptr, tl, i32)
274
+DEF_HELPER_FLAGS_6(sve_stss_le_zss_mte, TCG_CALL_NO_WG,
275
+ void, env, ptr, ptr, ptr, tl, i32)
276
+DEF_HELPER_FLAGS_6(sve_stss_be_zss_mte, TCG_CALL_NO_WG,
277
+ void, env, ptr, ptr, ptr, tl, i32)
278
+
279
+DEF_HELPER_FLAGS_6(sve_stbd_zsu_mte, TCG_CALL_NO_WG,
280
+ void, env, ptr, ptr, ptr, tl, i32)
281
+DEF_HELPER_FLAGS_6(sve_sthd_le_zsu_mte, TCG_CALL_NO_WG,
282
+ void, env, ptr, ptr, ptr, tl, i32)
283
+DEF_HELPER_FLAGS_6(sve_sthd_be_zsu_mte, TCG_CALL_NO_WG,
284
+ void, env, ptr, ptr, ptr, tl, i32)
285
+DEF_HELPER_FLAGS_6(sve_stsd_le_zsu_mte, TCG_CALL_NO_WG,
286
+ void, env, ptr, ptr, ptr, tl, i32)
287
+DEF_HELPER_FLAGS_6(sve_stsd_be_zsu_mte, TCG_CALL_NO_WG,
288
+ void, env, ptr, ptr, ptr, tl, i32)
289
+DEF_HELPER_FLAGS_6(sve_stdd_le_zsu_mte, TCG_CALL_NO_WG,
290
+ void, env, ptr, ptr, ptr, tl, i32)
291
+DEF_HELPER_FLAGS_6(sve_stdd_be_zsu_mte, TCG_CALL_NO_WG,
292
+ void, env, ptr, ptr, ptr, tl, i32)
293
+
294
+DEF_HELPER_FLAGS_6(sve_stbd_zss_mte, TCG_CALL_NO_WG,
295
+ void, env, ptr, ptr, ptr, tl, i32)
296
+DEF_HELPER_FLAGS_6(sve_sthd_le_zss_mte, TCG_CALL_NO_WG,
297
+ void, env, ptr, ptr, ptr, tl, i32)
298
+DEF_HELPER_FLAGS_6(sve_sthd_be_zss_mte, TCG_CALL_NO_WG,
299
+ void, env, ptr, ptr, ptr, tl, i32)
300
+DEF_HELPER_FLAGS_6(sve_stsd_le_zss_mte, TCG_CALL_NO_WG,
301
+ void, env, ptr, ptr, ptr, tl, i32)
302
+DEF_HELPER_FLAGS_6(sve_stsd_be_zss_mte, TCG_CALL_NO_WG,
303
+ void, env, ptr, ptr, ptr, tl, i32)
304
+DEF_HELPER_FLAGS_6(sve_stdd_le_zss_mte, TCG_CALL_NO_WG,
305
+ void, env, ptr, ptr, ptr, tl, i32)
306
+DEF_HELPER_FLAGS_6(sve_stdd_be_zss_mte, TCG_CALL_NO_WG,
307
+ void, env, ptr, ptr, ptr, tl, i32)
308
+
309
+DEF_HELPER_FLAGS_6(sve_stbd_zd_mte, TCG_CALL_NO_WG,
310
+ void, env, ptr, ptr, ptr, tl, i32)
311
+DEF_HELPER_FLAGS_6(sve_sthd_le_zd_mte, TCG_CALL_NO_WG,
312
+ void, env, ptr, ptr, ptr, tl, i32)
313
+DEF_HELPER_FLAGS_6(sve_sthd_be_zd_mte, TCG_CALL_NO_WG,
314
+ void, env, ptr, ptr, ptr, tl, i32)
315
+DEF_HELPER_FLAGS_6(sve_stsd_le_zd_mte, TCG_CALL_NO_WG,
316
+ void, env, ptr, ptr, ptr, tl, i32)
317
+DEF_HELPER_FLAGS_6(sve_stsd_be_zd_mte, TCG_CALL_NO_WG,
318
+ void, env, ptr, ptr, ptr, tl, i32)
319
+DEF_HELPER_FLAGS_6(sve_stdd_le_zd_mte, TCG_CALL_NO_WG,
320
+ void, env, ptr, ptr, ptr, tl, i32)
321
+DEF_HELPER_FLAGS_6(sve_stdd_be_zd_mte, TCG_CALL_NO_WG,
322
+ void, env, ptr, ptr, ptr, tl, i32)
323
+
324
DEF_HELPER_FLAGS_4(sve2_pmull_h, TCG_CALL_NO_RWG, void, ptr, ptr, ptr, i32)
325
diff --git a/target/arm/sve_helper.c b/target/arm/sve_helper.c
326
index XXXXXXX..XXXXXXX 100644
327
--- a/target/arm/sve_helper.c
328
+++ b/target/arm/sve_helper.c
329
@@ -XXX,XX +XXX,XX @@ static target_ulong off_zd_d(void *reg, intptr_t reg_ofs)
330
static inline QEMU_ALWAYS_INLINE
331
void sve_ld1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
332
target_ulong base, uint32_t desc, uintptr_t retaddr,
333
- int esize, int msize, zreg_off_fn *off_fn,
334
+ uint32_t mtedesc, int esize, int msize,
335
+ zreg_off_fn *off_fn,
336
sve_ldst1_host_fn *host_fn,
337
sve_ldst1_tlb_fn *tlb_fn)
338
{
339
@@ -XXX,XX +XXX,XX @@ void sve_ld1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
340
cpu_check_watchpoint(env_cpu(env), addr, msize,
341
info.attrs, BP_MEM_READ, retaddr);
342
}
343
- /* TODO: MTE check */
344
+ if (mtedesc && arm_tlb_mte_tagged(&info.attrs)) {
345
+ mte_check1(env, mtedesc, addr, retaddr);
346
+ }
347
host_fn(&scratch, reg_off, info.host);
348
} else {
349
/* Element crosses the page boundary. */
350
@@ -XXX,XX +XXX,XX @@ void sve_ld1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
351
msize, info.attrs,
352
BP_MEM_READ, retaddr);
353
}
354
- /* TODO: MTE check */
355
+ if (mtedesc && arm_tlb_mte_tagged(&info.attrs)) {
356
+ mte_check1(env, mtedesc, addr, retaddr);
357
+ }
358
tlb_fn(env, &scratch, reg_off, addr, retaddr);
359
}
360
}
361
@@ -XXX,XX +XXX,XX @@ void sve_ld1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
362
memcpy(vd, &scratch, reg_max);
42
}
363
}
43
364
44
+static MemoryRegion *make_dma(MPS2TZMachineState *mms, void *opaque,
365
+static inline QEMU_ALWAYS_INLINE
45
+ const char *name, hwaddr size)
366
+void sve_ld1_z_mte(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
367
+ target_ulong base, uint32_t desc, uintptr_t retaddr,
368
+ int esize, int msize, zreg_off_fn *off_fn,
369
+ sve_ldst1_host_fn *host_fn,
370
+ sve_ldst1_tlb_fn *tlb_fn)
46
+{
371
+{
47
+ PL080State *dma = opaque;
372
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
48
+ int i = dma - &mms->dma[0];
373
+ /* Remove mtedesc from the normal sve descriptor. */
49
+ SysBusDevice *s;
374
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
50
+ char *mscname = g_strdup_printf("%s-msc", name);
51
+ TZMSC *msc = &mms->msc[i];
52
+ DeviceState *iotkitdev = DEVICE(&mms->iotkit);
53
+ MemoryRegion *msc_upstream;
54
+ MemoryRegion *msc_downstream;
55
+
375
+
56
+ /*
376
+ /*
57
+ * Each DMA device is a PL081 whose transaction master interface
377
+ * ??? TODO: For the 32-bit offset extractions, base + ofs cannot
58
+ * is guarded by a Master Security Controller. The downstream end of
378
+ * offset base entirely over the address space hole to change the
59
+ * the MSC connects to the IoTKit AHB Slave Expansion port, so the
379
+ * pointer tag, or change the bit55 selector. So we could here
60
+ * DMA devices can see all devices and memory that the CPU does.
380
+ * examine TBI + TCMA like we do for sve_ldN_r_mte().
61
+ */
381
+ */
62
+ sysbus_init_child_obj(OBJECT(mms), mscname, msc, sizeof(*msc), TYPE_TZ_MSC);
382
+ sve_ld1_z(env, vd, vg, vm, base, desc, retaddr, mtedesc,
63
+ msc_downstream = sysbus_mmio_get_region(SYS_BUS_DEVICE(&mms->iotkit), 0);
383
+ esize, msize, off_fn, host_fn, tlb_fn);
64
+ object_property_set_link(OBJECT(msc), OBJECT(msc_downstream),
65
+ "downstream", &error_fatal);
66
+ object_property_set_link(OBJECT(msc), OBJECT(mms),
67
+ "idau", &error_fatal);
68
+ object_property_set_bool(OBJECT(msc), true, "realized", &error_fatal);
69
+
70
+ qdev_connect_gpio_out_named(DEVICE(msc), "irq", 0,
71
+ qdev_get_gpio_in_named(iotkitdev,
72
+ "mscexp_status", i));
73
+ qdev_connect_gpio_out_named(iotkitdev, "mscexp_clear", i,
74
+ qdev_get_gpio_in_named(DEVICE(msc),
75
+ "irq_clear", 0));
76
+ qdev_connect_gpio_out_named(iotkitdev, "mscexp_ns", i,
77
+ qdev_get_gpio_in_named(DEVICE(msc),
78
+ "cfg_nonsec", 0));
79
+ qdev_connect_gpio_out(DEVICE(&mms->sec_resp_splitter),
80
+ ARRAY_SIZE(mms->ppc) + i,
81
+ qdev_get_gpio_in_named(DEVICE(msc),
82
+ "cfg_sec_resp", 0));
83
+ msc_upstream = sysbus_mmio_get_region(SYS_BUS_DEVICE(msc), 0);
84
+
85
+ sysbus_init_child_obj(OBJECT(mms), name, dma, sizeof(*dma), TYPE_PL081);
86
+ object_property_set_link(OBJECT(dma), OBJECT(msc_upstream),
87
+ "downstream", &error_fatal);
88
+ object_property_set_bool(OBJECT(dma), true, "realized", &error_fatal);
89
+
90
+ s = SYS_BUS_DEVICE(dma);
91
+ /* Wire up DMACINTR, DMACINTERR, DMACINTTC */
92
+ sysbus_connect_irq(s, 0, qdev_get_gpio_in_named(iotkitdev,
93
+ "EXP_IRQ", 58 + i * 3));
94
+ sysbus_connect_irq(s, 1, qdev_get_gpio_in_named(iotkitdev,
95
+ "EXP_IRQ", 56 + i * 3));
96
+ sysbus_connect_irq(s, 2, qdev_get_gpio_in_named(iotkitdev,
97
+ "EXP_IRQ", 57 + i * 3));
98
+
99
+ return sysbus_mmio_get_region(s, 0);
100
+}
384
+}
101
+
385
+
102
static void mps2tz_common_init(MachineState *machine)
386
#define DO_LD1_ZPZ_S(MEM, OFS, MSZ) \
387
void HELPER(sve_ld##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
388
void *vm, target_ulong base, uint32_t desc) \
389
{ \
390
- sve_ld1_z(env, vd, vg, vm, base, desc, GETPC(), 4, 1 << MSZ, \
391
+ sve_ld1_z(env, vd, vg, vm, base, desc, GETPC(), 0, 4, 1 << MSZ, \
392
off_##OFS##_s, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
393
+} \
394
+void HELPER(sve_ld##MEM##_##OFS##_mte)(CPUARMState *env, void *vd, void *vg, \
395
+ void *vm, target_ulong base, uint32_t desc) \
396
+{ \
397
+ sve_ld1_z_mte(env, vd, vg, vm, base, desc, GETPC(), 4, 1 << MSZ, \
398
+ off_##OFS##_s, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
399
}
400
401
#define DO_LD1_ZPZ_D(MEM, OFS, MSZ) \
402
void HELPER(sve_ld##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
403
void *vm, target_ulong base, uint32_t desc) \
404
{ \
405
- sve_ld1_z(env, vd, vg, vm, base, desc, GETPC(), 8, 1 << MSZ, \
406
+ sve_ld1_z(env, vd, vg, vm, base, desc, GETPC(), 0, 8, 1 << MSZ, \
407
off_##OFS##_d, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
408
+} \
409
+void HELPER(sve_ld##MEM##_##OFS##_mte)(CPUARMState *env, void *vd, void *vg, \
410
+ void *vm, target_ulong base, uint32_t desc) \
411
+{ \
412
+ sve_ld1_z_mte(env, vd, vg, vm, base, desc, GETPC(), 8, 1 << MSZ, \
413
+ off_##OFS##_d, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
414
}
415
416
DO_LD1_ZPZ_S(bsu, zsu, MO_8)
417
@@ -XXX,XX +XXX,XX @@ DO_LD1_ZPZ_D(dd_be, zd, MO_64)
418
static inline QEMU_ALWAYS_INLINE
419
void sve_ldff1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
420
target_ulong base, uint32_t desc, uintptr_t retaddr,
421
- const int esz, const int msz, zreg_off_fn *off_fn,
422
+ uint32_t mtedesc, const int esz, const int msz,
423
+ zreg_off_fn *off_fn,
424
sve_ldst1_host_fn *host_fn,
425
sve_ldst1_tlb_fn *tlb_fn)
103
{
426
{
104
MPS2TZMachineState *mms = MPS2TZ_MACHINE(machine);
427
@@ -XXX,XX +XXX,XX @@ void sve_ldff1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
105
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
428
* Probe the first element, allowing faults.
106
&error_fatal);
107
108
/* The sec_resp_cfg output from the IoTKit must be split into multiple
109
- * lines, one for each of the PPCs we create here.
110
+ * lines, one for each of the PPCs we create here, plus one per MSC.
111
*/
429
*/
112
object_initialize(&mms->sec_resp_splitter, sizeof(mms->sec_resp_splitter),
430
addr = base + (off_fn(vm, reg_off) << scale);
113
TYPE_SPLIT_IRQ);
431
+ if (mtedesc) {
114
object_property_add_child(OBJECT(machine), "sec-resp-splitter",
432
+ mte_check1(env, mtedesc, addr, retaddr);
115
OBJECT(&mms->sec_resp_splitter), &error_abort);
433
+ }
116
- object_property_set_int(OBJECT(&mms->sec_resp_splitter), 5,
434
tlb_fn(env, vd, reg_off, addr, retaddr);
117
+ object_property_set_int(OBJECT(&mms->sec_resp_splitter),
435
118
+ ARRAY_SIZE(mms->ppc) + ARRAY_SIZE(mms->msc),
436
/* After any fault, zero the other elements. */
119
"num-lines", &error_fatal);
437
@@ -XXX,XX +XXX,XX @@ void sve_ldff1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
120
object_property_set_bool(OBJECT(&mms->sec_resp_splitter), true,
438
(env_cpu(env), addr, msize) & BP_MEM_READ)) {
121
"realized", &error_fatal);
439
goto fault;
122
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
440
}
123
}, {
441
- /* TODO: MTE check. */
124
.name = "ahb_ppcexp1",
442
+ if (mtedesc &&
125
.ports = {
443
+ arm_tlb_mte_tagged(&info.attrs) &&
126
- { "dma0", make_unimp_dev, &mms->dma[0], 0x40110000, 0x1000 },
444
+ !mte_probe1(env, mtedesc, addr)) {
127
- { "dma1", make_unimp_dev, &mms->dma[1], 0x40111000, 0x1000 },
445
+ goto fault;
128
- { "dma2", make_unimp_dev, &mms->dma[2], 0x40112000, 0x1000 },
446
+ }
129
- { "dma3", make_unimp_dev, &mms->dma[3], 0x40113000, 0x1000 },
447
130
+ { "dma0", make_dma, &mms->dma[0], 0x40110000, 0x1000 },
448
host_fn(vd, reg_off, info.host);
131
+ { "dma1", make_dma, &mms->dma[1], 0x40111000, 0x1000 },
449
}
132
+ { "dma2", make_dma, &mms->dma[2], 0x40112000, 0x1000 },
450
@@ -XXX,XX +XXX,XX @@ void sve_ldff1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
133
+ { "dma3", make_dma, &mms->dma[3], 0x40113000, 0x1000 },
451
record_fault(env, reg_off, reg_max);
134
},
135
},
136
};
137
@@ -XXX,XX +XXX,XX @@ static void mps2tz_common_init(MachineState *machine)
138
armv7m_load_kernel(ARM_CPU(first_cpu), machine->kernel_filename, 0x400000);
139
}
452
}
140
453
141
+static void mps2_tz_idau_check(IDAUInterface *ii, uint32_t address,
454
-#define DO_LDFF1_ZPZ_S(MEM, OFS, MSZ) \
142
+ int *iregion, bool *exempt, bool *ns, bool *nsc)
455
-void HELPER(sve_ldff##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
456
- void *vm, target_ulong base, uint32_t desc) \
457
-{ \
458
- sve_ldff1_z(env, vd, vg, vm, base, desc, GETPC(), MO_32, MSZ, \
459
- off_##OFS##_s, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
460
+static inline QEMU_ALWAYS_INLINE
461
+void sve_ldff1_z_mte(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
462
+ target_ulong base, uint32_t desc, uintptr_t retaddr,
463
+ const int esz, const int msz,
464
+ zreg_off_fn *off_fn,
465
+ sve_ldst1_host_fn *host_fn,
466
+ sve_ldst1_tlb_fn *tlb_fn)
143
+{
467
+{
468
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
469
+ /* Remove mtedesc from the normal sve descriptor. */
470
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
471
+
144
+ /*
472
+ /*
145
+ * The MPS2 TZ FPGA images have IDAUs in them which are connected to
473
+ * ??? TODO: For the 32-bit offset extractions, base + ofs cannot
146
+ * the Master Security Controllers. Thes have the same logic as
474
+ * offset base entirely over the address space hole to change the
147
+ * is used by the IoTKit for the IDAU connected to the CPU, except
475
+ * pointer tag, or change the bit55 selector. So we could here
148
+ * that MSCs don't care about the NSC attribute.
476
+ * examine TBI + TCMA like we do for sve_ldN_r_mte().
149
+ */
477
+ */
150
+ int region = extract32(address, 28, 4);
478
+ sve_ldff1_z(env, vd, vg, vm, base, desc, retaddr, mtedesc,
151
+
479
+ esz, msz, off_fn, host_fn, tlb_fn);
152
+ *ns = !(region & 1);
480
}
153
+ *nsc = false;
481
154
+ /* 0xe0000000..0xe00fffff and 0xf0000000..0xf00fffff are exempt */
482
-#define DO_LDFF1_ZPZ_D(MEM, OFS, MSZ) \
155
+ *exempt = (address & 0xeff00000) == 0xe0000000;
483
-void HELPER(sve_ldff##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
156
+ *iregion = region;
484
- void *vm, target_ulong base, uint32_t desc) \
485
-{ \
486
- sve_ldff1_z(env, vd, vg, vm, base, desc, GETPC(), MO_64, MSZ, \
487
- off_##OFS##_d, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
488
+#define DO_LDFF1_ZPZ_S(MEM, OFS, MSZ) \
489
+void HELPER(sve_ldff##MEM##_##OFS) \
490
+ (CPUARMState *env, void *vd, void *vg, \
491
+ void *vm, target_ulong base, uint32_t desc) \
492
+{ \
493
+ sve_ldff1_z(env, vd, vg, vm, base, desc, GETPC(), 0, MO_32, MSZ, \
494
+ off_##OFS##_s, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
495
+} \
496
+void HELPER(sve_ldff##MEM##_##OFS##_mte) \
497
+ (CPUARMState *env, void *vd, void *vg, \
498
+ void *vm, target_ulong base, uint32_t desc) \
499
+{ \
500
+ sve_ldff1_z_mte(env, vd, vg, vm, base, desc, GETPC(), MO_32, MSZ, \
501
+ off_##OFS##_s, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
157
+}
502
+}
158
+
503
+
159
static void mps2tz_class_init(ObjectClass *oc, void *data)
504
+#define DO_LDFF1_ZPZ_D(MEM, OFS, MSZ) \
505
+void HELPER(sve_ldff##MEM##_##OFS) \
506
+ (CPUARMState *env, void *vd, void *vg, \
507
+ void *vm, target_ulong base, uint32_t desc) \
508
+{ \
509
+ sve_ldff1_z(env, vd, vg, vm, base, desc, GETPC(), 0, MO_64, MSZ, \
510
+ off_##OFS##_d, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
511
+} \
512
+void HELPER(sve_ldff##MEM##_##OFS##_mte) \
513
+ (CPUARMState *env, void *vd, void *vg, \
514
+ void *vm, target_ulong base, uint32_t desc) \
515
+{ \
516
+ sve_ldff1_z_mte(env, vd, vg, vm, base, desc, GETPC(), MO_64, MSZ, \
517
+ off_##OFS##_d, sve_ld1##MEM##_host, sve_ld1##MEM##_tlb); \
518
}
519
520
DO_LDFF1_ZPZ_S(bsu, zsu, MO_8)
521
@@ -XXX,XX +XXX,XX @@ DO_LDFF1_ZPZ_D(dd_be, zd, MO_64)
522
static inline QEMU_ALWAYS_INLINE
523
void sve_st1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
524
target_ulong base, uint32_t desc, uintptr_t retaddr,
525
- int esize, int msize, zreg_off_fn *off_fn,
526
+ uint32_t mtedesc, int esize, int msize,
527
+ zreg_off_fn *off_fn,
528
sve_ldst1_host_fn *host_fn,
529
sve_ldst1_tlb_fn *tlb_fn)
160
{
530
{
161
MachineClass *mc = MACHINE_CLASS(oc);
531
@@ -XXX,XX +XXX,XX @@ void sve_st1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
162
+ IDAUInterfaceClass *iic = IDAU_INTERFACE_CLASS(oc);
532
cpu_check_watchpoint(env_cpu(env), addr, msize,
163
533
info.attrs, BP_MEM_WRITE, retaddr);
164
mc->init = mps2tz_common_init;
534
}
165
mc->max_cpus = 1;
535
- /* TODO: MTE check. */
166
+ iic->check = mps2_tz_idau_check;
536
+
537
+ if (mtedesc && arm_tlb_mte_tagged(&info.attrs)) {
538
+ mte_check1(env, mtedesc, addr, retaddr);
539
+ }
540
}
541
i += 1;
542
reg_off += esize;
543
@@ -XXX,XX +XXX,XX @@ void sve_st1_z(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
544
} while (reg_off < reg_max);
167
}
545
}
168
546
169
static void mps2tz_an505_class_init(ObjectClass *oc, void *data)
547
-#define DO_ST1_ZPZ_S(MEM, OFS, MSZ) \
170
@@ -XXX,XX +XXX,XX @@ static const TypeInfo mps2tz_info = {
548
-void HELPER(sve_st##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
171
.instance_size = sizeof(MPS2TZMachineState),
549
- void *vm, target_ulong base, uint32_t desc) \
172
.class_size = sizeof(MPS2TZMachineClass),
550
-{ \
173
.class_init = mps2tz_class_init,
551
- sve_st1_z(env, vd, vg, vm, base, desc, GETPC(), 4, 1 << MSZ, \
174
+ .interfaces = (InterfaceInfo[]) {
552
- off_##OFS##_s, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
175
+ { TYPE_IDAU_INTERFACE },
553
+static inline QEMU_ALWAYS_INLINE
176
+ { }
554
+void sve_st1_z_mte(CPUARMState *env, void *vd, uint64_t *vg, void *vm,
177
+ },
555
+ target_ulong base, uint32_t desc, uintptr_t retaddr,
556
+ int esize, int msize, zreg_off_fn *off_fn,
557
+ sve_ldst1_host_fn *host_fn,
558
+ sve_ldst1_tlb_fn *tlb_fn)
559
+{
560
+ uint32_t mtedesc = desc >> (SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
561
+ /* Remove mtedesc from the normal sve descriptor. */
562
+ desc = extract32(desc, 0, SIMD_DATA_SHIFT + SVE_MTEDESC_SHIFT);
563
+
564
+ /*
565
+ * ??? TODO: For the 32-bit offset extractions, base + ofs cannot
566
+ * offset base entirely over the address space hole to change the
567
+ * pointer tag, or change the bit55 selector. So we could here
568
+ * examine TBI + TCMA like we do for sve_ldN_r_mte().
569
+ */
570
+ sve_st1_z(env, vd, vg, vm, base, desc, retaddr, mtedesc,
571
+ esize, msize, off_fn, host_fn, tlb_fn);
572
}
573
574
-#define DO_ST1_ZPZ_D(MEM, OFS, MSZ) \
575
-void HELPER(sve_st##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
576
+#define DO_ST1_ZPZ_S(MEM, OFS, MSZ) \
577
+void HELPER(sve_st##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
578
void *vm, target_ulong base, uint32_t desc) \
579
-{ \
580
- sve_st1_z(env, vd, vg, vm, base, desc, GETPC(), 8, 1 << MSZ, \
581
- off_##OFS##_d, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
582
+{ \
583
+ sve_st1_z(env, vd, vg, vm, base, desc, GETPC(), 0, 4, 1 << MSZ, \
584
+ off_##OFS##_s, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
585
+} \
586
+void HELPER(sve_st##MEM##_##OFS##_mte)(CPUARMState *env, void *vd, void *vg, \
587
+ void *vm, target_ulong base, uint32_t desc) \
588
+{ \
589
+ sve_st1_z_mte(env, vd, vg, vm, base, desc, GETPC(), 4, 1 << MSZ, \
590
+ off_##OFS##_s, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
591
+}
592
+
593
+#define DO_ST1_ZPZ_D(MEM, OFS, MSZ) \
594
+void HELPER(sve_st##MEM##_##OFS)(CPUARMState *env, void *vd, void *vg, \
595
+ void *vm, target_ulong base, uint32_t desc) \
596
+{ \
597
+ sve_st1_z(env, vd, vg, vm, base, desc, GETPC(), 0, 8, 1 << MSZ, \
598
+ off_##OFS##_d, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
599
+} \
600
+void HELPER(sve_st##MEM##_##OFS##_mte)(CPUARMState *env, void *vd, void *vg, \
601
+ void *vm, target_ulong base, uint32_t desc) \
602
+{ \
603
+ sve_st1_z_mte(env, vd, vg, vm, base, desc, GETPC(), 8, 1 << MSZ, \
604
+ off_##OFS##_d, sve_st1##MEM##_host, sve_st1##MEM##_tlb); \
605
}
606
607
DO_ST1_ZPZ_S(bs, zsu, MO_8)
608
diff --git a/target/arm/translate-sve.c b/target/arm/translate-sve.c
609
index XXXXXXX..XXXXXXX 100644
610
--- a/target/arm/translate-sve.c
611
+++ b/target/arm/translate-sve.c
612
@@ -XXX,XX +XXX,XX @@ static bool trans_ST_zpri(DisasContext *s, arg_rpri_store *a)
613
*/
614
615
static void do_mem_zpz(DisasContext *s, int zt, int pg, int zm,
616
- int scale, TCGv_i64 scalar, int msz,
617
+ int scale, TCGv_i64 scalar, int msz, bool is_write,
618
gen_helper_gvec_mem_scatter *fn)
619
{
620
unsigned vsz = vec_full_reg_size(s);
621
@@ -XXX,XX +XXX,XX @@ static void do_mem_zpz(DisasContext *s, int zt, int pg, int zm,
622
TCGv_ptr t_pg = tcg_temp_new_ptr();
623
TCGv_ptr t_zt = tcg_temp_new_ptr();
624
TCGv_i32 t_desc;
625
- int desc;
626
+ int desc = 0;
627
628
+ if (s->mte_active[0]) {
629
+ desc = FIELD_DP32(desc, MTEDESC, MIDX, get_mem_index(s));
630
+ desc = FIELD_DP32(desc, MTEDESC, TBI, s->tbid);
631
+ desc = FIELD_DP32(desc, MTEDESC, TCMA, s->tcma);
632
+ desc = FIELD_DP32(desc, MTEDESC, WRITE, is_write);
633
+ desc = FIELD_DP32(desc, MTEDESC, ESIZE, 1 << msz);
634
+ desc <<= SVE_MTEDESC_SHIFT;
635
+ }
636
desc = simd_desc(vsz, vsz, scale);
637
t_desc = tcg_const_i32(desc);
638
639
@@ -XXX,XX +XXX,XX @@ static void do_mem_zpz(DisasContext *s, int zt, int pg, int zm,
640
tcg_temp_free_i32(t_desc);
641
}
642
643
-/* Indexed by [be][ff][xs][u][msz]. */
644
-static gen_helper_gvec_mem_scatter * const gather_load_fn32[2][2][2][2][3] = {
645
- /* Little-endian */
646
- { { { { gen_helper_sve_ldbss_zsu,
647
- gen_helper_sve_ldhss_le_zsu,
648
- NULL, },
649
- { gen_helper_sve_ldbsu_zsu,
650
- gen_helper_sve_ldhsu_le_zsu,
651
- gen_helper_sve_ldss_le_zsu, } },
652
- { { gen_helper_sve_ldbss_zss,
653
- gen_helper_sve_ldhss_le_zss,
654
- NULL, },
655
- { gen_helper_sve_ldbsu_zss,
656
- gen_helper_sve_ldhsu_le_zss,
657
- gen_helper_sve_ldss_le_zss, } } },
658
+/* Indexed by [mte][be][ff][xs][u][msz]. */
659
+static gen_helper_gvec_mem_scatter * const
660
+gather_load_fn32[2][2][2][2][2][3] = {
661
+ { /* MTE Inactive */
662
+ { /* Little-endian */
663
+ { { { gen_helper_sve_ldbss_zsu,
664
+ gen_helper_sve_ldhss_le_zsu,
665
+ NULL, },
666
+ { gen_helper_sve_ldbsu_zsu,
667
+ gen_helper_sve_ldhsu_le_zsu,
668
+ gen_helper_sve_ldss_le_zsu, } },
669
+ { { gen_helper_sve_ldbss_zss,
670
+ gen_helper_sve_ldhss_le_zss,
671
+ NULL, },
672
+ { gen_helper_sve_ldbsu_zss,
673
+ gen_helper_sve_ldhsu_le_zss,
674
+ gen_helper_sve_ldss_le_zss, } } },
675
676
- /* First-fault */
677
- { { { gen_helper_sve_ldffbss_zsu,
678
- gen_helper_sve_ldffhss_le_zsu,
679
- NULL, },
680
- { gen_helper_sve_ldffbsu_zsu,
681
- gen_helper_sve_ldffhsu_le_zsu,
682
- gen_helper_sve_ldffss_le_zsu, } },
683
- { { gen_helper_sve_ldffbss_zss,
684
- gen_helper_sve_ldffhss_le_zss,
685
- NULL, },
686
- { gen_helper_sve_ldffbsu_zss,
687
- gen_helper_sve_ldffhsu_le_zss,
688
- gen_helper_sve_ldffss_le_zss, } } } },
689
+ /* First-fault */
690
+ { { { gen_helper_sve_ldffbss_zsu,
691
+ gen_helper_sve_ldffhss_le_zsu,
692
+ NULL, },
693
+ { gen_helper_sve_ldffbsu_zsu,
694
+ gen_helper_sve_ldffhsu_le_zsu,
695
+ gen_helper_sve_ldffss_le_zsu, } },
696
+ { { gen_helper_sve_ldffbss_zss,
697
+ gen_helper_sve_ldffhss_le_zss,
698
+ NULL, },
699
+ { gen_helper_sve_ldffbsu_zss,
700
+ gen_helper_sve_ldffhsu_le_zss,
701
+ gen_helper_sve_ldffss_le_zss, } } } },
702
703
- /* Big-endian */
704
- { { { { gen_helper_sve_ldbss_zsu,
705
- gen_helper_sve_ldhss_be_zsu,
706
- NULL, },
707
- { gen_helper_sve_ldbsu_zsu,
708
- gen_helper_sve_ldhsu_be_zsu,
709
- gen_helper_sve_ldss_be_zsu, } },
710
- { { gen_helper_sve_ldbss_zss,
711
- gen_helper_sve_ldhss_be_zss,
712
- NULL, },
713
- { gen_helper_sve_ldbsu_zss,
714
- gen_helper_sve_ldhsu_be_zss,
715
- gen_helper_sve_ldss_be_zss, } } },
716
+ { /* Big-endian */
717
+ { { { gen_helper_sve_ldbss_zsu,
718
+ gen_helper_sve_ldhss_be_zsu,
719
+ NULL, },
720
+ { gen_helper_sve_ldbsu_zsu,
721
+ gen_helper_sve_ldhsu_be_zsu,
722
+ gen_helper_sve_ldss_be_zsu, } },
723
+ { { gen_helper_sve_ldbss_zss,
724
+ gen_helper_sve_ldhss_be_zss,
725
+ NULL, },
726
+ { gen_helper_sve_ldbsu_zss,
727
+ gen_helper_sve_ldhsu_be_zss,
728
+ gen_helper_sve_ldss_be_zss, } } },
729
730
- /* First-fault */
731
- { { { gen_helper_sve_ldffbss_zsu,
732
- gen_helper_sve_ldffhss_be_zsu,
733
- NULL, },
734
- { gen_helper_sve_ldffbsu_zsu,
735
- gen_helper_sve_ldffhsu_be_zsu,
736
- gen_helper_sve_ldffss_be_zsu, } },
737
- { { gen_helper_sve_ldffbss_zss,
738
- gen_helper_sve_ldffhss_be_zss,
739
- NULL, },
740
- { gen_helper_sve_ldffbsu_zss,
741
- gen_helper_sve_ldffhsu_be_zss,
742
- gen_helper_sve_ldffss_be_zss, } } } },
743
+ /* First-fault */
744
+ { { { gen_helper_sve_ldffbss_zsu,
745
+ gen_helper_sve_ldffhss_be_zsu,
746
+ NULL, },
747
+ { gen_helper_sve_ldffbsu_zsu,
748
+ gen_helper_sve_ldffhsu_be_zsu,
749
+ gen_helper_sve_ldffss_be_zsu, } },
750
+ { { gen_helper_sve_ldffbss_zss,
751
+ gen_helper_sve_ldffhss_be_zss,
752
+ NULL, },
753
+ { gen_helper_sve_ldffbsu_zss,
754
+ gen_helper_sve_ldffhsu_be_zss,
755
+ gen_helper_sve_ldffss_be_zss, } } } } },
756
+ { /* MTE Active */
757
+ { /* Little-endian */
758
+ { { { gen_helper_sve_ldbss_zsu_mte,
759
+ gen_helper_sve_ldhss_le_zsu_mte,
760
+ NULL, },
761
+ { gen_helper_sve_ldbsu_zsu_mte,
762
+ gen_helper_sve_ldhsu_le_zsu_mte,
763
+ gen_helper_sve_ldss_le_zsu_mte, } },
764
+ { { gen_helper_sve_ldbss_zss_mte,
765
+ gen_helper_sve_ldhss_le_zss_mte,
766
+ NULL, },
767
+ { gen_helper_sve_ldbsu_zss_mte,
768
+ gen_helper_sve_ldhsu_le_zss_mte,
769
+ gen_helper_sve_ldss_le_zss_mte, } } },
770
+
771
+ /* First-fault */
772
+ { { { gen_helper_sve_ldffbss_zsu_mte,
773
+ gen_helper_sve_ldffhss_le_zsu_mte,
774
+ NULL, },
775
+ { gen_helper_sve_ldffbsu_zsu_mte,
776
+ gen_helper_sve_ldffhsu_le_zsu_mte,
777
+ gen_helper_sve_ldffss_le_zsu_mte, } },
778
+ { { gen_helper_sve_ldffbss_zss_mte,
779
+ gen_helper_sve_ldffhss_le_zss_mte,
780
+ NULL, },
781
+ { gen_helper_sve_ldffbsu_zss_mte,
782
+ gen_helper_sve_ldffhsu_le_zss_mte,
783
+ gen_helper_sve_ldffss_le_zss_mte, } } } },
784
+
785
+ { /* Big-endian */
786
+ { { { gen_helper_sve_ldbss_zsu_mte,
787
+ gen_helper_sve_ldhss_be_zsu_mte,
788
+ NULL, },
789
+ { gen_helper_sve_ldbsu_zsu_mte,
790
+ gen_helper_sve_ldhsu_be_zsu_mte,
791
+ gen_helper_sve_ldss_be_zsu_mte, } },
792
+ { { gen_helper_sve_ldbss_zss_mte,
793
+ gen_helper_sve_ldhss_be_zss_mte,
794
+ NULL, },
795
+ { gen_helper_sve_ldbsu_zss_mte,
796
+ gen_helper_sve_ldhsu_be_zss_mte,
797
+ gen_helper_sve_ldss_be_zss_mte, } } },
798
+
799
+ /* First-fault */
800
+ { { { gen_helper_sve_ldffbss_zsu_mte,
801
+ gen_helper_sve_ldffhss_be_zsu_mte,
802
+ NULL, },
803
+ { gen_helper_sve_ldffbsu_zsu_mte,
804
+ gen_helper_sve_ldffhsu_be_zsu_mte,
805
+ gen_helper_sve_ldffss_be_zsu_mte, } },
806
+ { { gen_helper_sve_ldffbss_zss_mte,
807
+ gen_helper_sve_ldffhss_be_zss_mte,
808
+ NULL, },
809
+ { gen_helper_sve_ldffbsu_zss_mte,
810
+ gen_helper_sve_ldffhsu_be_zss_mte,
811
+ gen_helper_sve_ldffss_be_zss_mte, } } } } },
178
};
812
};
179
813
180
static const TypeInfo mps2tz_an505_info = {
814
/* Note that we overload xs=2 to indicate 64-bit offset. */
815
-static gen_helper_gvec_mem_scatter * const gather_load_fn64[2][2][3][2][4] = {
816
- /* Little-endian */
817
- { { { { gen_helper_sve_ldbds_zsu,
818
- gen_helper_sve_ldhds_le_zsu,
819
- gen_helper_sve_ldsds_le_zsu,
820
- NULL, },
821
- { gen_helper_sve_ldbdu_zsu,
822
- gen_helper_sve_ldhdu_le_zsu,
823
- gen_helper_sve_ldsdu_le_zsu,
824
- gen_helper_sve_lddd_le_zsu, } },
825
- { { gen_helper_sve_ldbds_zss,
826
- gen_helper_sve_ldhds_le_zss,
827
- gen_helper_sve_ldsds_le_zss,
828
- NULL, },
829
- { gen_helper_sve_ldbdu_zss,
830
- gen_helper_sve_ldhdu_le_zss,
831
- gen_helper_sve_ldsdu_le_zss,
832
- gen_helper_sve_lddd_le_zss, } },
833
- { { gen_helper_sve_ldbds_zd,
834
- gen_helper_sve_ldhds_le_zd,
835
- gen_helper_sve_ldsds_le_zd,
836
- NULL, },
837
- { gen_helper_sve_ldbdu_zd,
838
- gen_helper_sve_ldhdu_le_zd,
839
- gen_helper_sve_ldsdu_le_zd,
840
- gen_helper_sve_lddd_le_zd, } } },
841
+static gen_helper_gvec_mem_scatter * const
842
+gather_load_fn64[2][2][2][3][2][4] = {
843
+ { /* MTE Inactive */
844
+ { /* Little-endian */
845
+ { { { gen_helper_sve_ldbds_zsu,
846
+ gen_helper_sve_ldhds_le_zsu,
847
+ gen_helper_sve_ldsds_le_zsu,
848
+ NULL, },
849
+ { gen_helper_sve_ldbdu_zsu,
850
+ gen_helper_sve_ldhdu_le_zsu,
851
+ gen_helper_sve_ldsdu_le_zsu,
852
+ gen_helper_sve_lddd_le_zsu, } },
853
+ { { gen_helper_sve_ldbds_zss,
854
+ gen_helper_sve_ldhds_le_zss,
855
+ gen_helper_sve_ldsds_le_zss,
856
+ NULL, },
857
+ { gen_helper_sve_ldbdu_zss,
858
+ gen_helper_sve_ldhdu_le_zss,
859
+ gen_helper_sve_ldsdu_le_zss,
860
+ gen_helper_sve_lddd_le_zss, } },
861
+ { { gen_helper_sve_ldbds_zd,
862
+ gen_helper_sve_ldhds_le_zd,
863
+ gen_helper_sve_ldsds_le_zd,
864
+ NULL, },
865
+ { gen_helper_sve_ldbdu_zd,
866
+ gen_helper_sve_ldhdu_le_zd,
867
+ gen_helper_sve_ldsdu_le_zd,
868
+ gen_helper_sve_lddd_le_zd, } } },
869
870
- /* First-fault */
871
- { { { gen_helper_sve_ldffbds_zsu,
872
- gen_helper_sve_ldffhds_le_zsu,
873
- gen_helper_sve_ldffsds_le_zsu,
874
- NULL, },
875
- { gen_helper_sve_ldffbdu_zsu,
876
- gen_helper_sve_ldffhdu_le_zsu,
877
- gen_helper_sve_ldffsdu_le_zsu,
878
- gen_helper_sve_ldffdd_le_zsu, } },
879
- { { gen_helper_sve_ldffbds_zss,
880
- gen_helper_sve_ldffhds_le_zss,
881
- gen_helper_sve_ldffsds_le_zss,
882
- NULL, },
883
- { gen_helper_sve_ldffbdu_zss,
884
- gen_helper_sve_ldffhdu_le_zss,
885
- gen_helper_sve_ldffsdu_le_zss,
886
- gen_helper_sve_ldffdd_le_zss, } },
887
- { { gen_helper_sve_ldffbds_zd,
888
- gen_helper_sve_ldffhds_le_zd,
889
- gen_helper_sve_ldffsds_le_zd,
890
- NULL, },
891
- { gen_helper_sve_ldffbdu_zd,
892
- gen_helper_sve_ldffhdu_le_zd,
893
- gen_helper_sve_ldffsdu_le_zd,
894
- gen_helper_sve_ldffdd_le_zd, } } } },
895
+ /* First-fault */
896
+ { { { gen_helper_sve_ldffbds_zsu,
897
+ gen_helper_sve_ldffhds_le_zsu,
898
+ gen_helper_sve_ldffsds_le_zsu,
899
+ NULL, },
900
+ { gen_helper_sve_ldffbdu_zsu,
901
+ gen_helper_sve_ldffhdu_le_zsu,
902
+ gen_helper_sve_ldffsdu_le_zsu,
903
+ gen_helper_sve_ldffdd_le_zsu, } },
904
+ { { gen_helper_sve_ldffbds_zss,
905
+ gen_helper_sve_ldffhds_le_zss,
906
+ gen_helper_sve_ldffsds_le_zss,
907
+ NULL, },
908
+ { gen_helper_sve_ldffbdu_zss,
909
+ gen_helper_sve_ldffhdu_le_zss,
910
+ gen_helper_sve_ldffsdu_le_zss,
911
+ gen_helper_sve_ldffdd_le_zss, } },
912
+ { { gen_helper_sve_ldffbds_zd,
913
+ gen_helper_sve_ldffhds_le_zd,
914
+ gen_helper_sve_ldffsds_le_zd,
915
+ NULL, },
916
+ { gen_helper_sve_ldffbdu_zd,
917
+ gen_helper_sve_ldffhdu_le_zd,
918
+ gen_helper_sve_ldffsdu_le_zd,
919
+ gen_helper_sve_ldffdd_le_zd, } } } },
920
+ { /* Big-endian */
921
+ { { { gen_helper_sve_ldbds_zsu,
922
+ gen_helper_sve_ldhds_be_zsu,
923
+ gen_helper_sve_ldsds_be_zsu,
924
+ NULL, },
925
+ { gen_helper_sve_ldbdu_zsu,
926
+ gen_helper_sve_ldhdu_be_zsu,
927
+ gen_helper_sve_ldsdu_be_zsu,
928
+ gen_helper_sve_lddd_be_zsu, } },
929
+ { { gen_helper_sve_ldbds_zss,
930
+ gen_helper_sve_ldhds_be_zss,
931
+ gen_helper_sve_ldsds_be_zss,
932
+ NULL, },
933
+ { gen_helper_sve_ldbdu_zss,
934
+ gen_helper_sve_ldhdu_be_zss,
935
+ gen_helper_sve_ldsdu_be_zss,
936
+ gen_helper_sve_lddd_be_zss, } },
937
+ { { gen_helper_sve_ldbds_zd,
938
+ gen_helper_sve_ldhds_be_zd,
939
+ gen_helper_sve_ldsds_be_zd,
940
+ NULL, },
941
+ { gen_helper_sve_ldbdu_zd,
942
+ gen_helper_sve_ldhdu_be_zd,
943
+ gen_helper_sve_ldsdu_be_zd,
944
+ gen_helper_sve_lddd_be_zd, } } },
945
946
- /* Big-endian */
947
- { { { { gen_helper_sve_ldbds_zsu,
948
- gen_helper_sve_ldhds_be_zsu,
949
- gen_helper_sve_ldsds_be_zsu,
950
- NULL, },
951
- { gen_helper_sve_ldbdu_zsu,
952
- gen_helper_sve_ldhdu_be_zsu,
953
- gen_helper_sve_ldsdu_be_zsu,
954
- gen_helper_sve_lddd_be_zsu, } },
955
- { { gen_helper_sve_ldbds_zss,
956
- gen_helper_sve_ldhds_be_zss,
957
- gen_helper_sve_ldsds_be_zss,
958
- NULL, },
959
- { gen_helper_sve_ldbdu_zss,
960
- gen_helper_sve_ldhdu_be_zss,
961
- gen_helper_sve_ldsdu_be_zss,
962
- gen_helper_sve_lddd_be_zss, } },
963
- { { gen_helper_sve_ldbds_zd,
964
- gen_helper_sve_ldhds_be_zd,
965
- gen_helper_sve_ldsds_be_zd,
966
- NULL, },
967
- { gen_helper_sve_ldbdu_zd,
968
- gen_helper_sve_ldhdu_be_zd,
969
- gen_helper_sve_ldsdu_be_zd,
970
- gen_helper_sve_lddd_be_zd, } } },
971
+ /* First-fault */
972
+ { { { gen_helper_sve_ldffbds_zsu,
973
+ gen_helper_sve_ldffhds_be_zsu,
974
+ gen_helper_sve_ldffsds_be_zsu,
975
+ NULL, },
976
+ { gen_helper_sve_ldffbdu_zsu,
977
+ gen_helper_sve_ldffhdu_be_zsu,
978
+ gen_helper_sve_ldffsdu_be_zsu,
979
+ gen_helper_sve_ldffdd_be_zsu, } },
980
+ { { gen_helper_sve_ldffbds_zss,
981
+ gen_helper_sve_ldffhds_be_zss,
982
+ gen_helper_sve_ldffsds_be_zss,
983
+ NULL, },
984
+ { gen_helper_sve_ldffbdu_zss,
985
+ gen_helper_sve_ldffhdu_be_zss,
986
+ gen_helper_sve_ldffsdu_be_zss,
987
+ gen_helper_sve_ldffdd_be_zss, } },
988
+ { { gen_helper_sve_ldffbds_zd,
989
+ gen_helper_sve_ldffhds_be_zd,
990
+ gen_helper_sve_ldffsds_be_zd,
991
+ NULL, },
992
+ { gen_helper_sve_ldffbdu_zd,
993
+ gen_helper_sve_ldffhdu_be_zd,
994
+ gen_helper_sve_ldffsdu_be_zd,
995
+ gen_helper_sve_ldffdd_be_zd, } } } } },
996
+ { /* MTE Active */
997
+ { /* Little-endian */
998
+ { { { gen_helper_sve_ldbds_zsu_mte,
999
+ gen_helper_sve_ldhds_le_zsu_mte,
1000
+ gen_helper_sve_ldsds_le_zsu_mte,
1001
+ NULL, },
1002
+ { gen_helper_sve_ldbdu_zsu_mte,
1003
+ gen_helper_sve_ldhdu_le_zsu_mte,
1004
+ gen_helper_sve_ldsdu_le_zsu_mte,
1005
+ gen_helper_sve_lddd_le_zsu_mte, } },
1006
+ { { gen_helper_sve_ldbds_zss_mte,
1007
+ gen_helper_sve_ldhds_le_zss_mte,
1008
+ gen_helper_sve_ldsds_le_zss_mte,
1009
+ NULL, },
1010
+ { gen_helper_sve_ldbdu_zss_mte,
1011
+ gen_helper_sve_ldhdu_le_zss_mte,
1012
+ gen_helper_sve_ldsdu_le_zss_mte,
1013
+ gen_helper_sve_lddd_le_zss_mte, } },
1014
+ { { gen_helper_sve_ldbds_zd_mte,
1015
+ gen_helper_sve_ldhds_le_zd_mte,
1016
+ gen_helper_sve_ldsds_le_zd_mte,
1017
+ NULL, },
1018
+ { gen_helper_sve_ldbdu_zd_mte,
1019
+ gen_helper_sve_ldhdu_le_zd_mte,
1020
+ gen_helper_sve_ldsdu_le_zd_mte,
1021
+ gen_helper_sve_lddd_le_zd_mte, } } },
1022
1023
- /* First-fault */
1024
- { { { gen_helper_sve_ldffbds_zsu,
1025
- gen_helper_sve_ldffhds_be_zsu,
1026
- gen_helper_sve_ldffsds_be_zsu,
1027
- NULL, },
1028
- { gen_helper_sve_ldffbdu_zsu,
1029
- gen_helper_sve_ldffhdu_be_zsu,
1030
- gen_helper_sve_ldffsdu_be_zsu,
1031
- gen_helper_sve_ldffdd_be_zsu, } },
1032
- { { gen_helper_sve_ldffbds_zss,
1033
- gen_helper_sve_ldffhds_be_zss,
1034
- gen_helper_sve_ldffsds_be_zss,
1035
- NULL, },
1036
- { gen_helper_sve_ldffbdu_zss,
1037
- gen_helper_sve_ldffhdu_be_zss,
1038
- gen_helper_sve_ldffsdu_be_zss,
1039
- gen_helper_sve_ldffdd_be_zss, } },
1040
- { { gen_helper_sve_ldffbds_zd,
1041
- gen_helper_sve_ldffhds_be_zd,
1042
- gen_helper_sve_ldffsds_be_zd,
1043
- NULL, },
1044
- { gen_helper_sve_ldffbdu_zd,
1045
- gen_helper_sve_ldffhdu_be_zd,
1046
- gen_helper_sve_ldffsdu_be_zd,
1047
- gen_helper_sve_ldffdd_be_zd, } } } },
1048
+ /* First-fault */
1049
+ { { { gen_helper_sve_ldffbds_zsu_mte,
1050
+ gen_helper_sve_ldffhds_le_zsu_mte,
1051
+ gen_helper_sve_ldffsds_le_zsu_mte,
1052
+ NULL, },
1053
+ { gen_helper_sve_ldffbdu_zsu_mte,
1054
+ gen_helper_sve_ldffhdu_le_zsu_mte,
1055
+ gen_helper_sve_ldffsdu_le_zsu_mte,
1056
+ gen_helper_sve_ldffdd_le_zsu_mte, } },
1057
+ { { gen_helper_sve_ldffbds_zss_mte,
1058
+ gen_helper_sve_ldffhds_le_zss_mte,
1059
+ gen_helper_sve_ldffsds_le_zss_mte,
1060
+ NULL, },
1061
+ { gen_helper_sve_ldffbdu_zss_mte,
1062
+ gen_helper_sve_ldffhdu_le_zss_mte,
1063
+ gen_helper_sve_ldffsdu_le_zss_mte,
1064
+ gen_helper_sve_ldffdd_le_zss_mte, } },
1065
+ { { gen_helper_sve_ldffbds_zd_mte,
1066
+ gen_helper_sve_ldffhds_le_zd_mte,
1067
+ gen_helper_sve_ldffsds_le_zd_mte,
1068
+ NULL, },
1069
+ { gen_helper_sve_ldffbdu_zd_mte,
1070
+ gen_helper_sve_ldffhdu_le_zd_mte,
1071
+ gen_helper_sve_ldffsdu_le_zd_mte,
1072
+ gen_helper_sve_ldffdd_le_zd_mte, } } } },
1073
+ { /* Big-endian */
1074
+ { { { gen_helper_sve_ldbds_zsu_mte,
1075
+ gen_helper_sve_ldhds_be_zsu_mte,
1076
+ gen_helper_sve_ldsds_be_zsu_mte,
1077
+ NULL, },
1078
+ { gen_helper_sve_ldbdu_zsu_mte,
1079
+ gen_helper_sve_ldhdu_be_zsu_mte,
1080
+ gen_helper_sve_ldsdu_be_zsu_mte,
1081
+ gen_helper_sve_lddd_be_zsu_mte, } },
1082
+ { { gen_helper_sve_ldbds_zss_mte,
1083
+ gen_helper_sve_ldhds_be_zss_mte,
1084
+ gen_helper_sve_ldsds_be_zss_mte,
1085
+ NULL, },
1086
+ { gen_helper_sve_ldbdu_zss_mte,
1087
+ gen_helper_sve_ldhdu_be_zss_mte,
1088
+ gen_helper_sve_ldsdu_be_zss_mte,
1089
+ gen_helper_sve_lddd_be_zss_mte, } },
1090
+ { { gen_helper_sve_ldbds_zd_mte,
1091
+ gen_helper_sve_ldhds_be_zd_mte,
1092
+ gen_helper_sve_ldsds_be_zd_mte,
1093
+ NULL, },
1094
+ { gen_helper_sve_ldbdu_zd_mte,
1095
+ gen_helper_sve_ldhdu_be_zd_mte,
1096
+ gen_helper_sve_ldsdu_be_zd_mte,
1097
+ gen_helper_sve_lddd_be_zd_mte, } } },
1098
+
1099
+ /* First-fault */
1100
+ { { { gen_helper_sve_ldffbds_zsu_mte,
1101
+ gen_helper_sve_ldffhds_be_zsu_mte,
1102
+ gen_helper_sve_ldffsds_be_zsu_mte,
1103
+ NULL, },
1104
+ { gen_helper_sve_ldffbdu_zsu_mte,
1105
+ gen_helper_sve_ldffhdu_be_zsu_mte,
1106
+ gen_helper_sve_ldffsdu_be_zsu_mte,
1107
+ gen_helper_sve_ldffdd_be_zsu_mte, } },
1108
+ { { gen_helper_sve_ldffbds_zss_mte,
1109
+ gen_helper_sve_ldffhds_be_zss_mte,
1110
+ gen_helper_sve_ldffsds_be_zss_mte,
1111
+ NULL, },
1112
+ { gen_helper_sve_ldffbdu_zss_mte,
1113
+ gen_helper_sve_ldffhdu_be_zss_mte,
1114
+ gen_helper_sve_ldffsdu_be_zss_mte,
1115
+ gen_helper_sve_ldffdd_be_zss_mte, } },
1116
+ { { gen_helper_sve_ldffbds_zd_mte,
1117
+ gen_helper_sve_ldffhds_be_zd_mte,
1118
+ gen_helper_sve_ldffsds_be_zd_mte,
1119
+ NULL, },
1120
+ { gen_helper_sve_ldffbdu_zd_mte,
1121
+ gen_helper_sve_ldffhdu_be_zd_mte,
1122
+ gen_helper_sve_ldffsdu_be_zd_mte,
1123
+ gen_helper_sve_ldffdd_be_zd_mte, } } } } },
1124
};
1125
1126
static bool trans_LD1_zprz(DisasContext *s, arg_LD1_zprz *a)
1127
{
1128
gen_helper_gvec_mem_scatter *fn = NULL;
1129
- int be = s->be_data == MO_BE;
1130
+ bool be = s->be_data == MO_BE;
1131
+ bool mte = s->mte_active[0];
1132
1133
if (!sve_access_check(s)) {
1134
return true;
1135
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1_zprz(DisasContext *s, arg_LD1_zprz *a)
1136
1137
switch (a->esz) {
1138
case MO_32:
1139
- fn = gather_load_fn32[be][a->ff][a->xs][a->u][a->msz];
1140
+ fn = gather_load_fn32[mte][be][a->ff][a->xs][a->u][a->msz];
1141
break;
1142
case MO_64:
1143
- fn = gather_load_fn64[be][a->ff][a->xs][a->u][a->msz];
1144
+ fn = gather_load_fn64[mte][be][a->ff][a->xs][a->u][a->msz];
1145
break;
1146
}
1147
assert(fn != NULL);
1148
1149
do_mem_zpz(s, a->rd, a->pg, a->rm, a->scale * a->msz,
1150
- cpu_reg_sp(s, a->rn), a->msz, fn);
1151
+ cpu_reg_sp(s, a->rn), a->msz, false, fn);
1152
return true;
1153
}
1154
1155
static bool trans_LD1_zpiz(DisasContext *s, arg_LD1_zpiz *a)
1156
{
1157
gen_helper_gvec_mem_scatter *fn = NULL;
1158
- int be = s->be_data == MO_BE;
1159
+ bool be = s->be_data == MO_BE;
1160
+ bool mte = s->mte_active[0];
1161
TCGv_i64 imm;
1162
1163
if (a->esz < a->msz || (a->esz == a->msz && !a->u)) {
1164
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1_zpiz(DisasContext *s, arg_LD1_zpiz *a)
1165
1166
switch (a->esz) {
1167
case MO_32:
1168
- fn = gather_load_fn32[be][a->ff][0][a->u][a->msz];
1169
+ fn = gather_load_fn32[mte][be][a->ff][0][a->u][a->msz];
1170
break;
1171
case MO_64:
1172
- fn = gather_load_fn64[be][a->ff][2][a->u][a->msz];
1173
+ fn = gather_load_fn64[mte][be][a->ff][2][a->u][a->msz];
1174
break;
1175
}
1176
assert(fn != NULL);
1177
@@ -XXX,XX +XXX,XX @@ static bool trans_LD1_zpiz(DisasContext *s, arg_LD1_zpiz *a)
1178
* by loading the immediate into the scalar parameter.
1179
*/
1180
imm = tcg_const_i64(a->imm << a->msz);
1181
- do_mem_zpz(s, a->rd, a->pg, a->rn, 0, imm, a->msz, fn);
1182
+ do_mem_zpz(s, a->rd, a->pg, a->rn, 0, imm, a->msz, false, fn);
1183
tcg_temp_free_i64(imm);
1184
return true;
1185
}
1186
1187
-/* Indexed by [be][xs][msz]. */
1188
-static gen_helper_gvec_mem_scatter * const scatter_store_fn32[2][2][3] = {
1189
- /* Little-endian */
1190
- { { gen_helper_sve_stbs_zsu,
1191
- gen_helper_sve_sths_le_zsu,
1192
- gen_helper_sve_stss_le_zsu, },
1193
- { gen_helper_sve_stbs_zss,
1194
- gen_helper_sve_sths_le_zss,
1195
- gen_helper_sve_stss_le_zss, } },
1196
- /* Big-endian */
1197
- { { gen_helper_sve_stbs_zsu,
1198
- gen_helper_sve_sths_be_zsu,
1199
- gen_helper_sve_stss_be_zsu, },
1200
- { gen_helper_sve_stbs_zss,
1201
- gen_helper_sve_sths_be_zss,
1202
- gen_helper_sve_stss_be_zss, } },
1203
+/* Indexed by [mte][be][xs][msz]. */
1204
+static gen_helper_gvec_mem_scatter * const scatter_store_fn32[2][2][2][3] = {
1205
+ { /* MTE Inactive */
1206
+ { /* Little-endian */
1207
+ { gen_helper_sve_stbs_zsu,
1208
+ gen_helper_sve_sths_le_zsu,
1209
+ gen_helper_sve_stss_le_zsu, },
1210
+ { gen_helper_sve_stbs_zss,
1211
+ gen_helper_sve_sths_le_zss,
1212
+ gen_helper_sve_stss_le_zss, } },
1213
+ { /* Big-endian */
1214
+ { gen_helper_sve_stbs_zsu,
1215
+ gen_helper_sve_sths_be_zsu,
1216
+ gen_helper_sve_stss_be_zsu, },
1217
+ { gen_helper_sve_stbs_zss,
1218
+ gen_helper_sve_sths_be_zss,
1219
+ gen_helper_sve_stss_be_zss, } } },
1220
+ { /* MTE Active */
1221
+ { /* Little-endian */
1222
+ { gen_helper_sve_stbs_zsu_mte,
1223
+ gen_helper_sve_sths_le_zsu_mte,
1224
+ gen_helper_sve_stss_le_zsu_mte, },
1225
+ { gen_helper_sve_stbs_zss_mte,
1226
+ gen_helper_sve_sths_le_zss_mte,
1227
+ gen_helper_sve_stss_le_zss_mte, } },
1228
+ { /* Big-endian */
1229
+ { gen_helper_sve_stbs_zsu_mte,
1230
+ gen_helper_sve_sths_be_zsu_mte,
1231
+ gen_helper_sve_stss_be_zsu_mte, },
1232
+ { gen_helper_sve_stbs_zss_mte,
1233
+ gen_helper_sve_sths_be_zss_mte,
1234
+ gen_helper_sve_stss_be_zss_mte, } } },
1235
};
1236
1237
/* Note that we overload xs=2 to indicate 64-bit offset. */
1238
-static gen_helper_gvec_mem_scatter * const scatter_store_fn64[2][3][4] = {
1239
- /* Little-endian */
1240
- { { gen_helper_sve_stbd_zsu,
1241
- gen_helper_sve_sthd_le_zsu,
1242
- gen_helper_sve_stsd_le_zsu,
1243
- gen_helper_sve_stdd_le_zsu, },
1244
- { gen_helper_sve_stbd_zss,
1245
- gen_helper_sve_sthd_le_zss,
1246
- gen_helper_sve_stsd_le_zss,
1247
- gen_helper_sve_stdd_le_zss, },
1248
- { gen_helper_sve_stbd_zd,
1249
- gen_helper_sve_sthd_le_zd,
1250
- gen_helper_sve_stsd_le_zd,
1251
- gen_helper_sve_stdd_le_zd, } },
1252
- /* Big-endian */
1253
- { { gen_helper_sve_stbd_zsu,
1254
- gen_helper_sve_sthd_be_zsu,
1255
- gen_helper_sve_stsd_be_zsu,
1256
- gen_helper_sve_stdd_be_zsu, },
1257
- { gen_helper_sve_stbd_zss,
1258
- gen_helper_sve_sthd_be_zss,
1259
- gen_helper_sve_stsd_be_zss,
1260
- gen_helper_sve_stdd_be_zss, },
1261
- { gen_helper_sve_stbd_zd,
1262
- gen_helper_sve_sthd_be_zd,
1263
- gen_helper_sve_stsd_be_zd,
1264
- gen_helper_sve_stdd_be_zd, } },
1265
+static gen_helper_gvec_mem_scatter * const scatter_store_fn64[2][2][3][4] = {
1266
+ { /* MTE Inactive */
1267
+ { /* Little-endian */
1268
+ { gen_helper_sve_stbd_zsu,
1269
+ gen_helper_sve_sthd_le_zsu,
1270
+ gen_helper_sve_stsd_le_zsu,
1271
+ gen_helper_sve_stdd_le_zsu, },
1272
+ { gen_helper_sve_stbd_zss,
1273
+ gen_helper_sve_sthd_le_zss,
1274
+ gen_helper_sve_stsd_le_zss,
1275
+ gen_helper_sve_stdd_le_zss, },
1276
+ { gen_helper_sve_stbd_zd,
1277
+ gen_helper_sve_sthd_le_zd,
1278
+ gen_helper_sve_stsd_le_zd,
1279
+ gen_helper_sve_stdd_le_zd, } },
1280
+ { /* Big-endian */
1281
+ { gen_helper_sve_stbd_zsu,
1282
+ gen_helper_sve_sthd_be_zsu,
1283
+ gen_helper_sve_stsd_be_zsu,
1284
+ gen_helper_sve_stdd_be_zsu, },
1285
+ { gen_helper_sve_stbd_zss,
1286
+ gen_helper_sve_sthd_be_zss,
1287
+ gen_helper_sve_stsd_be_zss,
1288
+ gen_helper_sve_stdd_be_zss, },
1289
+ { gen_helper_sve_stbd_zd,
1290
+ gen_helper_sve_sthd_be_zd,
1291
+ gen_helper_sve_stsd_be_zd,
1292
+ gen_helper_sve_stdd_be_zd, } } },
1293
+ { /* MTE Inactive */
1294
+ { /* Little-endian */
1295
+ { gen_helper_sve_stbd_zsu_mte,
1296
+ gen_helper_sve_sthd_le_zsu_mte,
1297
+ gen_helper_sve_stsd_le_zsu_mte,
1298
+ gen_helper_sve_stdd_le_zsu_mte, },
1299
+ { gen_helper_sve_stbd_zss_mte,
1300
+ gen_helper_sve_sthd_le_zss_mte,
1301
+ gen_helper_sve_stsd_le_zss_mte,
1302
+ gen_helper_sve_stdd_le_zss_mte, },
1303
+ { gen_helper_sve_stbd_zd_mte,
1304
+ gen_helper_sve_sthd_le_zd_mte,
1305
+ gen_helper_sve_stsd_le_zd_mte,
1306
+ gen_helper_sve_stdd_le_zd_mte, } },
1307
+ { /* Big-endian */
1308
+ { gen_helper_sve_stbd_zsu_mte,
1309
+ gen_helper_sve_sthd_be_zsu_mte,
1310
+ gen_helper_sve_stsd_be_zsu_mte,
1311
+ gen_helper_sve_stdd_be_zsu_mte, },
1312
+ { gen_helper_sve_stbd_zss_mte,
1313
+ gen_helper_sve_sthd_be_zss_mte,
1314
+ gen_helper_sve_stsd_be_zss_mte,
1315
+ gen_helper_sve_stdd_be_zss_mte, },
1316
+ { gen_helper_sve_stbd_zd_mte,
1317
+ gen_helper_sve_sthd_be_zd_mte,
1318
+ gen_helper_sve_stsd_be_zd_mte,
1319
+ gen_helper_sve_stdd_be_zd_mte, } } },
1320
};
1321
1322
static bool trans_ST1_zprz(DisasContext *s, arg_ST1_zprz *a)
1323
{
1324
gen_helper_gvec_mem_scatter *fn;
1325
- int be = s->be_data == MO_BE;
1326
+ bool be = s->be_data == MO_BE;
1327
+ bool mte = s->mte_active[0];
1328
1329
if (a->esz < a->msz || (a->msz == 0 && a->scale)) {
1330
return false;
1331
@@ -XXX,XX +XXX,XX @@ static bool trans_ST1_zprz(DisasContext *s, arg_ST1_zprz *a)
1332
}
1333
switch (a->esz) {
1334
case MO_32:
1335
- fn = scatter_store_fn32[be][a->xs][a->msz];
1336
+ fn = scatter_store_fn32[mte][be][a->xs][a->msz];
1337
break;
1338
case MO_64:
1339
- fn = scatter_store_fn64[be][a->xs][a->msz];
1340
+ fn = scatter_store_fn64[mte][be][a->xs][a->msz];
1341
break;
1342
default:
1343
g_assert_not_reached();
1344
}
1345
do_mem_zpz(s, a->rd, a->pg, a->rm, a->scale * a->msz,
1346
- cpu_reg_sp(s, a->rn), a->msz, fn);
1347
+ cpu_reg_sp(s, a->rn), a->msz, true, fn);
1348
return true;
1349
}
1350
1351
static bool trans_ST1_zpiz(DisasContext *s, arg_ST1_zpiz *a)
1352
{
1353
gen_helper_gvec_mem_scatter *fn = NULL;
1354
- int be = s->be_data == MO_BE;
1355
+ bool be = s->be_data == MO_BE;
1356
+ bool mte = s->mte_active[0];
1357
TCGv_i64 imm;
1358
1359
if (a->esz < a->msz) {
1360
@@ -XXX,XX +XXX,XX @@ static bool trans_ST1_zpiz(DisasContext *s, arg_ST1_zpiz *a)
1361
1362
switch (a->esz) {
1363
case MO_32:
1364
- fn = scatter_store_fn32[be][0][a->msz];
1365
+ fn = scatter_store_fn32[mte][be][0][a->msz];
1366
break;
1367
case MO_64:
1368
- fn = scatter_store_fn64[be][2][a->msz];
1369
+ fn = scatter_store_fn64[mte][be][2][a->msz];
1370
break;
1371
}
1372
assert(fn != NULL);
1373
@@ -XXX,XX +XXX,XX @@ static bool trans_ST1_zpiz(DisasContext *s, arg_ST1_zpiz *a)
1374
* by loading the immediate into the scalar parameter.
1375
*/
1376
imm = tcg_const_i64(a->imm << a->msz);
1377
- do_mem_zpz(s, a->rd, a->pg, a->rn, 0, imm, a->msz, fn);
1378
+ do_mem_zpz(s, a->rd, a->pg, a->rn, 0, imm, a->msz, true, fn);
1379
tcg_temp_free_i64(imm);
1380
return true;
1381
}
181
--
1382
--
182
2.18.0
1383
2.20.1
183
1384
184
1385
diff view generated by jsdifflib
1
Validate the config settings that the guest tries to set.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
2
3
The wiki page documentation is not really accurate here:
3
There are a number of paths by which the TBI is still intact
4
generally rather than failing requests to set bad parameters,
4
for user-only in the SVE helpers.
5
the hardware will just clip them to something sensible.
6
5
7
Validate the most important parameters: sizes and
6
Because we currently always set TBI for user-only, we do not
8
the viewport offsets. This prevents the framebuffer
7
need to pass down the actual TBI setting from above, and we
9
code from trying to read out-of-range memory.
8
can remove the top byte in the inner-most primitives, so that
9
none are forgotten. Moreover, this keeps the "dirty" pointer
10
around at the higher levels, where we need it for any MTE checking.
10
11
11
In the property handling code, we validate the new parameters every
12
Since the normal case, especially for user-only, goes through
12
time we encounter a tag that sets them. This means we validate the
13
RAM, this clearing merely adds two insns per page lookup, which
13
config multiple times if the request includes multiple config-setting
14
will be completely in the noise.
14
tags, but the code would require significant restructuring to do a
15
validation only once but still return the clipped settings for
16
get-parameter tags and the buffer allocation tag.
17
15
18
Validation of settings made via the older bcm2835_fb_mbox_push()
16
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
19
function will be done in the next commit.
17
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
18
Message-id: 20200626033144.790098-39-richard.henderson@linaro.org
19
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
20
---
21
target/arm/cpu.c | 3 +++
22
target/arm/sve_helper.c | 19 +++++++++++++++++--
23
target/arm/translate-a64.c | 5 +++++
24
3 files changed, 25 insertions(+), 2 deletions(-)
20
25
21
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
26
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
22
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
23
Message-id: 20180814144436.679-8-peter.maydell@linaro.org
24
---
25
include/hw/display/bcm2835_fb.h | 8 +++++
26
hw/display/bcm2835_fb.c | 48 +++++++++++++++++++++++++++--
27
hw/misc/bcm2835_property.c | 54 ++++++++++++++++-----------------
28
3 files changed, 81 insertions(+), 29 deletions(-)
29
30
diff --git a/include/hw/display/bcm2835_fb.h b/include/hw/display/bcm2835_fb.h
31
index XXXXXXX..XXXXXXX 100644
27
index XXXXXXX..XXXXXXX 100644
32
--- a/include/hw/display/bcm2835_fb.h
28
--- a/target/arm/cpu.c
33
+++ b/include/hw/display/bcm2835_fb.h
29
+++ b/target/arm/cpu.c
34
@@ -XXX,XX +XXX,XX @@ static inline uint32_t bcm2835_fb_get_size(BCM2835FBConfig *config)
30
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_reset(DeviceState *dev)
35
return yres * bcm2835_fb_get_pitch(config);
31
* Enable TBI0 and TBI1. While the real kernel only enables TBI0,
32
* turning on both here will produce smaller code and otherwise
33
* make no difference to the user-level emulation.
34
+ *
35
+ * In sve_probe_page, we assume that this is set.
36
+ * Do not modify this without other changes.
37
*/
38
env->cp15.tcr_el[1].raw_tcr = (3ULL << 37);
39
#else
40
diff --git a/target/arm/sve_helper.c b/target/arm/sve_helper.c
41
index XXXXXXX..XXXXXXX 100644
42
--- a/target/arm/sve_helper.c
43
+++ b/target/arm/sve_helper.c
44
@@ -XXX,XX +XXX,XX @@ static void sve_##NAME##_host(void *vd, intptr_t reg_off, void *host) \
45
static void sve_##NAME##_tlb(CPUARMState *env, void *vd, intptr_t reg_off, \
46
target_ulong addr, uintptr_t ra) \
47
{ \
48
- *(TYPEE *)(vd + H(reg_off)) = (TYPEM)TLB(env, addr, ra); \
49
+ *(TYPEE *)(vd + H(reg_off)) = \
50
+ (TYPEM)TLB(env, useronly_clean_ptr(addr), ra); \
36
}
51
}
37
52
38
+/**
53
#define DO_ST_TLB(NAME, H, TYPEE, TYPEM, TLB) \
39
+ * bcm2835_fb_validate_config: check provided config
54
static void sve_##NAME##_tlb(CPUARMState *env, void *vd, intptr_t reg_off, \
40
+ *
55
target_ulong addr, uintptr_t ra) \
41
+ * Validates the configuration information provided by the guest and
56
{ \
42
+ * adjusts it if necessary.
57
- TLB(env, addr, (TYPEM)*(TYPEE *)(vd + H(reg_off)), ra); \
43
+ */
58
+ TLB(env, useronly_clean_ptr(addr), \
44
+void bcm2835_fb_validate_config(BCM2835FBConfig *config);
59
+ (TYPEM)*(TYPEE *)(vd + H(reg_off)), ra); \
45
+
46
#endif
47
diff --git a/hw/display/bcm2835_fb.c b/hw/display/bcm2835_fb.c
48
index XXXXXXX..XXXXXXX 100644
49
--- a/hw/display/bcm2835_fb.c
50
+++ b/hw/display/bcm2835_fb.c
51
@@ -XXX,XX +XXX,XX @@
52
#define DEFAULT_VCRAM_SIZE 0x4000000
53
#define BCM2835_FB_OFFSET 0x00100000
54
55
+/* Maximum permitted framebuffer size; experimentally determined on an rpi2 */
56
+#define XRES_MAX 3840
57
+#define YRES_MAX 2560
58
+/* Framebuffer size used if guest requests zero size */
59
+#define XRES_SMALL 592
60
+#define YRES_SMALL 488
61
+
62
static void fb_invalidate_display(void *opaque)
63
{
64
BCM2835FBState *s = BCM2835_FB(opaque);
65
@@ -XXX,XX +XXX,XX @@ static void fb_update_display(void *opaque)
66
s->invalidate = false;
67
}
60
}
68
61
69
+void bcm2835_fb_validate_config(BCM2835FBConfig *config)
62
#define DO_LD_PRIM_1(NAME, H, TE, TM) \
70
+{
63
@@ -XXX,XX +XXX,XX @@ static bool sve_probe_page(SVEHostPage *info, bool nofault,
71
+ /*
64
int flags;
72
+ * Validate the config, and clip any bogus values into range,
65
73
+ * as the hardware does. Note that fb_update_display() relies on
66
addr += mem_off;
74
+ * this happening to prevent it from performing out-of-range
75
+ * accesses on redraw.
76
+ */
77
+ config->xres = MIN(config->xres, XRES_MAX);
78
+ config->xres_virtual = MIN(config->xres_virtual, XRES_MAX);
79
+ config->yres = MIN(config->yres, YRES_MAX);
80
+ config->yres_virtual = MIN(config->yres_virtual, YRES_MAX);
81
+
67
+
82
+ /*
68
+ /*
83
+ * These are not minima: a 40x40 framebuffer will be accepted.
69
+ * User-only currently always issues with TBI. See the comment
84
+ * They're only used as defaults if the guest asks for zero size.
70
+ * above useronly_clean_ptr. Usually we clean this top byte away
71
+ * during translation, but we can't do that for e.g. vector + imm
72
+ * addressing modes.
73
+ *
74
+ * We currently always enable TBI for user-only, and do not provide
75
+ * a way to turn it off. So clean the pointer unconditionally here,
76
+ * rather than look it up here, or pass it down from above.
85
+ */
77
+ */
86
+ if (config->xres == 0) {
78
+ addr = useronly_clean_ptr(addr);
87
+ config->xres = XRES_SMALL;
88
+ }
89
+ if (config->yres == 0) {
90
+ config->yres = YRES_SMALL;
91
+ }
92
+ if (config->xres_virtual == 0) {
93
+ config->xres_virtual = config->xres;
94
+ }
95
+ if (config->yres_virtual == 0) {
96
+ config->yres_virtual = config->yres;
97
+ }
98
+
79
+
99
+ if (fb_use_offsets(config)) {
80
flags = probe_access_flags(env, addr, access_type, mmu_idx, nofault,
100
+ /* Clip the offsets so the viewport is within the physical screen */
81
&info->host, retaddr);
101
+ config->xoffset = MIN(config->xoffset,
82
info->flags = flags;
102
+ config->xres_virtual - config->xres);
83
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
103
+ config->yoffset = MIN(config->yoffset,
84
index XXXXXXX..XXXXXXX 100644
104
+ config->yres_virtual - config->yres);
85
--- a/target/arm/translate-a64.c
105
+ }
86
+++ b/target/arm/translate-a64.c
106
+}
87
@@ -XXX,XX +XXX,XX @@ static void aarch64_tr_init_disas_context(DisasContextBase *dcbase,
88
dc->features = env->features;
89
dc->dcz_blocksize = arm_cpu->dcz_blocksize;
90
91
+#ifdef CONFIG_USER_ONLY
92
+ /* In sve_probe_page, we assume TBI is enabled. */
93
+ tcg_debug_assert(dc->tbid & 1);
94
+#endif
107
+
95
+
108
static void bcm2835_fb_mbox_push(BCM2835FBState *s, uint32_t value)
96
/* Single step state. The code-generation logic here is:
109
{
97
* SS_ACTIVE == 0:
110
uint32_t pitch;
98
* generate code with no special handling for single-stepping (except
111
@@ -XXX,XX +XXX,XX @@ void bcm2835_fb_reconfigure(BCM2835FBState *s, BCM2835FBConfig *newconfig)
112
{
113
s->lock = true;
114
115
- /* TODO: input validation! */
116
-
117
s->config = *newconfig;
118
119
s->invalidate = true;
120
diff --git a/hw/misc/bcm2835_property.c b/hw/misc/bcm2835_property.c
121
index XXXXXXX..XXXXXXX 100644
122
--- a/hw/misc/bcm2835_property.c
123
+++ b/hw/misc/bcm2835_property.c
124
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
125
case 0x00040002: /* Blank screen */
126
resplen = 4;
127
break;
128
- case 0x00040003: /* Get physical display width/height */
129
- stl_le_phys(&s->dma_as, value + 12, fbconfig.xres);
130
- stl_le_phys(&s->dma_as, value + 16, fbconfig.yres);
131
- resplen = 8;
132
- break;
133
- case 0x00040004: /* Get virtual display width/height */
134
- stl_le_phys(&s->dma_as, value + 12, fbconfig.xres_virtual);
135
- stl_le_phys(&s->dma_as, value + 16, fbconfig.yres_virtual);
136
- resplen = 8;
137
- break;
138
case 0x00044003: /* Test physical display width/height */
139
case 0x00044004: /* Test virtual display width/height */
140
resplen = 8;
141
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
142
case 0x00048003: /* Set physical display width/height */
143
fbconfig.xres = ldl_le_phys(&s->dma_as, value + 12);
144
fbconfig.yres = ldl_le_phys(&s->dma_as, value + 16);
145
+ bcm2835_fb_validate_config(&fbconfig);
146
fbconfig_updated = true;
147
+ /* fall through */
148
+ case 0x00040003: /* Get physical display width/height */
149
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xres);
150
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yres);
151
resplen = 8;
152
break;
153
case 0x00048004: /* Set virtual display width/height */
154
fbconfig.xres_virtual = ldl_le_phys(&s->dma_as, value + 12);
155
fbconfig.yres_virtual = ldl_le_phys(&s->dma_as, value + 16);
156
+ bcm2835_fb_validate_config(&fbconfig);
157
fbconfig_updated = true;
158
+ /* fall through */
159
+ case 0x00040004: /* Get virtual display width/height */
160
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xres_virtual);
161
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yres_virtual);
162
resplen = 8;
163
break;
164
- case 0x00040005: /* Get depth */
165
- stl_le_phys(&s->dma_as, value + 12, fbconfig.bpp);
166
- resplen = 4;
167
- break;
168
case 0x00044005: /* Test depth */
169
resplen = 4;
170
break;
171
case 0x00048005: /* Set depth */
172
fbconfig.bpp = ldl_le_phys(&s->dma_as, value + 12);
173
+ bcm2835_fb_validate_config(&fbconfig);
174
fbconfig_updated = true;
175
- resplen = 4;
176
- break;
177
- case 0x00040006: /* Get pixel order */
178
- stl_le_phys(&s->dma_as, value + 12, fbconfig.pixo);
179
+ /* fall through */
180
+ case 0x00040005: /* Get depth */
181
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.bpp);
182
resplen = 4;
183
break;
184
case 0x00044006: /* Test pixel order */
185
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
186
break;
187
case 0x00048006: /* Set pixel order */
188
fbconfig.pixo = ldl_le_phys(&s->dma_as, value + 12);
189
+ bcm2835_fb_validate_config(&fbconfig);
190
fbconfig_updated = true;
191
- resplen = 4;
192
- break;
193
- case 0x00040007: /* Get alpha */
194
- stl_le_phys(&s->dma_as, value + 12, fbconfig.alpha);
195
+ /* fall through */
196
+ case 0x00040006: /* Get pixel order */
197
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.pixo);
198
resplen = 4;
199
break;
200
case 0x00044007: /* Test pixel alpha */
201
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
202
break;
203
case 0x00048007: /* Set alpha */
204
fbconfig.alpha = ldl_le_phys(&s->dma_as, value + 12);
205
+ bcm2835_fb_validate_config(&fbconfig);
206
fbconfig_updated = true;
207
+ /* fall through */
208
+ case 0x00040007: /* Get alpha */
209
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.alpha);
210
resplen = 4;
211
break;
212
case 0x00040008: /* Get pitch */
213
@@ -XXX,XX +XXX,XX @@ static void bcm2835_property_mbox_push(BCM2835PropertyState *s, uint32_t value)
214
bcm2835_fb_get_pitch(&fbconfig));
215
resplen = 4;
216
break;
217
- case 0x00040009: /* Get virtual offset */
218
- stl_le_phys(&s->dma_as, value + 12, fbconfig.xoffset);
219
- stl_le_phys(&s->dma_as, value + 16, fbconfig.yoffset);
220
- resplen = 8;
221
- break;
222
case 0x00044009: /* Test virtual offset */
223
resplen = 8;
224
break;
225
case 0x00048009: /* Set virtual offset */
226
fbconfig.xoffset = ldl_le_phys(&s->dma_as, value + 12);
227
fbconfig.yoffset = ldl_le_phys(&s->dma_as, value + 16);
228
+ bcm2835_fb_validate_config(&fbconfig);
229
fbconfig_updated = true;
230
+ /* fall through */
231
+ case 0x00040009: /* Get virtual offset */
232
+ stl_le_phys(&s->dma_as, value + 12, fbconfig.xoffset);
233
+ stl_le_phys(&s->dma_as, value + 16, fbconfig.yoffset);
234
resplen = 8;
235
break;
236
case 0x0004000a: /* Get/Test/Set overscan */
237
--
99
--
238
2.18.0
100
2.20.1
239
101
240
102
diff view generated by jsdifflib
1
Following the bulk conversion of the iwMMXt code, there are
1
From: Richard Henderson <richard.henderson@linaro.org>
2
just a handful of hard coded tabs in target/arm; fix them.
3
This is a whitespace-only patch.
4
2
3
This is DC GVA and DC GZVA, and the tag check for DC ZVA.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-40-richard.henderson@linaro.org
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Message-id: 20180821165215.29069-4-peter.maydell@linaro.org
7
---
9
---
8
target/arm/cpu.h | 16 ++++++++--------
10
target/arm/cpu.h | 4 +++-
9
target/arm/arm-semi.c | 2 +-
11
target/arm/helper.c | 16 ++++++++++++++++
10
2 files changed, 9 insertions(+), 9 deletions(-)
12
target/arm/translate-a64.c | 39 ++++++++++++++++++++++++++++++++++++++
13
3 files changed, 58 insertions(+), 1 deletion(-)
11
14
12
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
15
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
13
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
14
--- a/target/arm/cpu.h
17
--- a/target/arm/cpu.h
15
+++ b/target/arm/cpu.h
18
+++ b/target/arm/cpu.h
16
@@ -XXX,XX +XXX,XX @@ enum arm_cpu_mode {
19
@@ -XXX,XX +XXX,XX @@ static inline uint64_t cpreg_to_kvm_id(uint32_t cpregid)
17
#define ARM_VFP_FPINST2 10
20
#define ARM_CP_NZCV (ARM_CP_SPECIAL | 0x0300)
18
21
#define ARM_CP_CURRENTEL (ARM_CP_SPECIAL | 0x0400)
19
/* iwMMXt coprocessor control registers. */
22
#define ARM_CP_DC_ZVA (ARM_CP_SPECIAL | 0x0500)
20
-#define ARM_IWMMXT_wCID        0
23
-#define ARM_LAST_SPECIAL ARM_CP_DC_ZVA
21
-#define ARM_IWMMXT_wCon        1
24
+#define ARM_CP_DC_GVA (ARM_CP_SPECIAL | 0x0600)
22
-#define ARM_IWMMXT_wCSSF    2
25
+#define ARM_CP_DC_GZVA (ARM_CP_SPECIAL | 0x0700)
23
-#define ARM_IWMMXT_wCASF    3
26
+#define ARM_LAST_SPECIAL ARM_CP_DC_GZVA
24
-#define ARM_IWMMXT_wCGR0    8
27
#define ARM_CP_FPU 0x1000
25
-#define ARM_IWMMXT_wCGR1    9
28
#define ARM_CP_SVE 0x2000
26
-#define ARM_IWMMXT_wCGR2    10
29
#define ARM_CP_NO_GDB 0x4000
27
-#define ARM_IWMMXT_wCGR3    11
30
diff --git a/target/arm/helper.c b/target/arm/helper.c
28
+#define ARM_IWMMXT_wCID 0
29
+#define ARM_IWMMXT_wCon 1
30
+#define ARM_IWMMXT_wCSSF 2
31
+#define ARM_IWMMXT_wCASF 3
32
+#define ARM_IWMMXT_wCGR0 8
33
+#define ARM_IWMMXT_wCGR1 9
34
+#define ARM_IWMMXT_wCGR2 10
35
+#define ARM_IWMMXT_wCGR3 11
36
37
/* V7M CCR bits */
38
FIELD(V7M_CCR, NONBASETHRDENA, 0, 1)
39
diff --git a/target/arm/arm-semi.c b/target/arm/arm-semi.c
40
index XXXXXXX..XXXXXXX 100644
31
index XXXXXXX..XXXXXXX 100644
41
--- a/target/arm/arm-semi.c
32
--- a/target/arm/helper.c
42
+++ b/target/arm/arm-semi.c
33
+++ b/target/arm/helper.c
43
@@ -XXX,XX +XXX,XX @@ static void arm_semi_cb(CPUState *cs, target_ulong ret, target_ulong err)
34
@@ -XXX,XX +XXX,XX @@ static const ARMCPRegInfo mte_el0_cacheop_reginfo[] = {
44
#ifdef CONFIG_USER_ONLY
35
.opc0 = 1, .opc1 = 3, .crn = 7, .crm = 14, .opc2 = 5,
45
ts->swi_errno = err;
36
.type = ARM_CP_NOP, .access = PL0_W,
46
#else
37
.accessfn = aa64_cacheop_poc_access },
47
-    syscall_err = err;
38
+ { .name = "DC_GVA", .state = ARM_CP_STATE_AA64,
48
+ syscall_err = err;
39
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 4, .opc2 = 3,
49
#endif
40
+ .access = PL0_W, .type = ARM_CP_DC_GVA,
50
reg0 = ret;
41
+#ifndef CONFIG_USER_ONLY
51
} else {
42
+ /* Avoid overhead of an access check that always passes in user-mode */
43
+ .accessfn = aa64_zva_access,
44
+#endif
45
+ },
46
+ { .name = "DC_GZVA", .state = ARM_CP_STATE_AA64,
47
+ .opc0 = 1, .opc1 = 3, .crn = 7, .crm = 4, .opc2 = 4,
48
+ .access = PL0_W, .type = ARM_CP_DC_GZVA,
49
+#ifndef CONFIG_USER_ONLY
50
+ /* Avoid overhead of an access check that always passes in user-mode */
51
+ .accessfn = aa64_zva_access,
52
+#endif
53
+ },
54
REGINFO_SENTINEL
55
};
56
57
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
58
index XXXXXXX..XXXXXXX 100644
59
--- a/target/arm/translate-a64.c
60
+++ b/target/arm/translate-a64.c
61
@@ -XXX,XX +XXX,XX @@ static void handle_sys(DisasContext *s, uint32_t insn, bool isread,
62
}
63
gen_helper_dc_zva(cpu_env, tcg_rt);
64
return;
65
+ case ARM_CP_DC_GVA:
66
+ {
67
+ TCGv_i64 clean_addr, tag;
68
+
69
+ /*
70
+ * DC_GVA, like DC_ZVA, requires that we supply the original
71
+ * pointer for an invalid page. Probe that address first.
72
+ */
73
+ tcg_rt = cpu_reg(s, rt);
74
+ clean_addr = clean_data_tbi(s, tcg_rt);
75
+ gen_probe_access(s, clean_addr, MMU_DATA_STORE, MO_8);
76
+
77
+ if (s->ata) {
78
+ /* Extract the tag from the register to match STZGM. */
79
+ tag = tcg_temp_new_i64();
80
+ tcg_gen_shri_i64(tag, tcg_rt, 56);
81
+ gen_helper_stzgm_tags(cpu_env, clean_addr, tag);
82
+ tcg_temp_free_i64(tag);
83
+ }
84
+ }
85
+ return;
86
+ case ARM_CP_DC_GZVA:
87
+ {
88
+ TCGv_i64 clean_addr, tag;
89
+
90
+ /* For DC_GZVA, we can rely on DC_ZVA for the proper fault. */
91
+ tcg_rt = cpu_reg(s, rt);
92
+ clean_addr = clean_data_tbi(s, tcg_rt);
93
+ gen_helper_dc_zva(cpu_env, clean_addr);
94
+
95
+ if (s->ata) {
96
+ /* Extract the tag from the register to match STZGM. */
97
+ tag = tcg_temp_new_i64();
98
+ tcg_gen_shri_i64(tag, tcg_rt, 56);
99
+ gen_helper_stzgm_tags(cpu_env, clean_addr, tag);
100
+ tcg_temp_free_i64(tag);
101
+ }
102
+ }
103
+ return;
104
default:
105
break;
106
}
52
--
107
--
53
2.18.0
108
2.20.1
54
109
55
110
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Richard Henderson <richard.henderson@linaro.org>
2
2
3
D1.10 specifies that exception handlers begin with tag checks overridden.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
3
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
4
Message-id: 20180814002653.12828-4-richard.henderson@linaro.org
7
Message-id: 20200626033144.790098-41-richard.henderson@linaro.org
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
---
9
---
8
target/arm/helper.c | 29 +++++------------------------
10
target/arm/helper.c | 3 +++
9
1 file changed, 5 insertions(+), 24 deletions(-)
11
1 file changed, 3 insertions(+)
10
12
11
diff --git a/target/arm/helper.c b/target/arm/helper.c
13
diff --git a/target/arm/helper.c b/target/arm/helper.c
12
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
13
--- a/target/arm/helper.c
15
--- a/target/arm/helper.c
14
+++ b/target/arm/helper.c
16
+++ b/target/arm/helper.c
15
@@ -XXX,XX +XXX,XX @@ float32 VFP_HELPER(fcvts, d)(float64 x, CPUARMState *env)
17
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_do_interrupt_aarch64(CPUState *cs)
16
#define VFP_CONV_FIX_FLOAT(name, p, fsz, isz, itype) \
18
break;
17
float##fsz HELPER(vfp_##name##to##p)(uint##isz##_t x, uint32_t shift, \
19
}
18
void *fpstp) \
20
}
19
-{ \
21
+ if (cpu_isar_feature(aa64_mte, cpu)) {
20
- float_status *fpst = fpstp; \
22
+ new_mode |= PSTATE_TCO;
21
- float##fsz tmp; \
23
+ }
22
- tmp = itype##_to_##float##fsz(x, fpst); \
24
23
- return float##fsz##_scalbn(tmp, -(int)shift, fpst); \
25
pstate_write(env, PSTATE_DAIF | new_mode);
24
-}
26
env->aarch64 = 1;
25
+{ return itype##_to_##float##fsz##_scalbn(x, -shift, fpstp); }
26
27
/* Notice that we want only input-denormal exception flags from the
28
* scalbn operation: the other possible flags (overflow+inexact if
29
@@ -XXX,XX +XXX,XX @@ VFP_CONV_FIX_A64(uq, s, 32, 64, uint64)
30
#undef VFP_CONV_FLOAT_FIX_ROUND
31
#undef VFP_CONV_FIX_A64
32
33
-/* Conversion to/from f16 can overflow to infinity before/after scaling.
34
- * Therefore we convert to f64, scale, and then convert f64 to f16; or
35
- * vice versa for conversion to integer.
36
- *
37
- * For 16- and 32-bit integers, the conversion to f64 never rounds.
38
- * For 64-bit integers, any integer that would cause rounding will also
39
- * overflow to f16 infinity, so there is no double rounding problem.
40
- */
41
-
42
-static float16 do_postscale_fp16(float64 f, int shift, float_status *fpst)
43
-{
44
- return float64_to_float16(float64_scalbn(f, -shift, fpst), true, fpst);
45
-}
46
-
47
uint32_t HELPER(vfp_sltoh)(uint32_t x, uint32_t shift, void *fpst)
48
{
49
- return do_postscale_fp16(int32_to_float64(x, fpst), shift, fpst);
50
+ return int32_to_float16_scalbn(x, -shift, fpst);
51
}
52
53
uint32_t HELPER(vfp_ultoh)(uint32_t x, uint32_t shift, void *fpst)
54
{
55
- return do_postscale_fp16(uint32_to_float64(x, fpst), shift, fpst);
56
+ return uint32_to_float16_scalbn(x, -shift, fpst);
57
}
58
59
uint32_t HELPER(vfp_sqtoh)(uint64_t x, uint32_t shift, void *fpst)
60
{
61
- return do_postscale_fp16(int64_to_float64(x, fpst), shift, fpst);
62
+ return int64_to_float16_scalbn(x, -shift, fpst);
63
}
64
65
uint32_t HELPER(vfp_uqtoh)(uint64_t x, uint32_t shift, void *fpst)
66
{
67
- return do_postscale_fp16(uint64_to_float64(x, fpst), shift, fpst);
68
+ return uint64_to_float16_scalbn(x, -shift, fpst);
69
}
70
71
static float64 do_prescale_fp16(float16 f, int shift, float_status *fpst)
72
--
27
--
73
2.18.0
28
2.20.1
74
29
75
30
diff view generated by jsdifflib
New patch
1
1
From: Richard Henderson <richard.henderson@linaro.org>
2
3
We need to check the memattr of a page in order to determine
4
whether it is Tagged for MTE. Between Stage1 and Stage2,
5
this becomes simpler if we always collect this data, instead
6
of occasionally being presented with NULL.
7
8
Use the nonnull attribute to allow the compiler to check that
9
all pointer arguments are non-null.
10
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
13
Message-id: 20200626033144.790098-42-richard.henderson@linaro.org
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
16
target/arm/internals.h | 3 ++-
17
target/arm/helper.c | 60 ++++++++++++++++++++---------------------
18
target/arm/m_helper.c | 11 +++++---
19
target/arm/tlb_helper.c | 4 ++-
20
4 files changed, 42 insertions(+), 36 deletions(-)
21
22
diff --git a/target/arm/internals.h b/target/arm/internals.h
23
index XXXXXXX..XXXXXXX 100644
24
--- a/target/arm/internals.h
25
+++ b/target/arm/internals.h
26
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
27
MMUAccessType access_type, ARMMMUIdx mmu_idx,
28
hwaddr *phys_ptr, MemTxAttrs *attrs, int *prot,
29
target_ulong *page_size,
30
- ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs);
31
+ ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs)
32
+ __attribute__((nonnull));
33
34
void arm_log_exception(int idx);
35
36
diff --git a/target/arm/helper.c b/target/arm/helper.c
37
index XXXXXXX..XXXXXXX 100644
38
--- a/target/arm/helper.c
39
+++ b/target/arm/helper.c
40
@@ -XXX,XX +XXX,XX @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
41
bool s1_is_el0,
42
hwaddr *phys_ptr, MemTxAttrs *txattrs, int *prot,
43
target_ulong *page_size_ptr,
44
- ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs);
45
+ ARMMMUFaultInfo *fi, ARMCacheAttrs *cacheattrs)
46
+ __attribute__((nonnull));
47
#endif
48
49
static void switch_mode(CPUARMState *env, int mode);
50
@@ -XXX,XX +XXX,XX @@ static bool get_phys_addr_lpae(CPUARMState *env, target_ulong address,
51
arm_tlb_bti_gp(txattrs) = true;
52
}
53
54
- if (cacheattrs != NULL) {
55
- if (mmu_idx == ARMMMUIdx_Stage2) {
56
- cacheattrs->attrs = convert_stage2_attrs(env,
57
- extract32(attrs, 0, 4));
58
- } else {
59
- /* Index into MAIR registers for cache attributes */
60
- uint8_t attrindx = extract32(attrs, 0, 3);
61
- uint64_t mair = env->cp15.mair_el[regime_el(env, mmu_idx)];
62
- assert(attrindx <= 7);
63
- cacheattrs->attrs = extract64(mair, attrindx * 8, 8);
64
- }
65
- cacheattrs->shareability = extract32(attrs, 6, 2);
66
+ if (mmu_idx == ARMMMUIdx_Stage2) {
67
+ cacheattrs->attrs = convert_stage2_attrs(env, extract32(attrs, 0, 4));
68
+ } else {
69
+ /* Index into MAIR registers for cache attributes */
70
+ uint8_t attrindx = extract32(attrs, 0, 3);
71
+ uint64_t mair = env->cp15.mair_el[regime_el(env, mmu_idx)];
72
+ assert(attrindx <= 7);
73
+ cacheattrs->attrs = extract64(mair, attrindx * 8, 8);
74
}
75
+ cacheattrs->shareability = extract32(attrs, 6, 2);
76
77
*phys_ptr = descaddr;
78
*page_size_ptr = page_size;
79
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
80
ret = get_phys_addr_lpae(env, ipa, access_type, ARMMMUIdx_Stage2,
81
mmu_idx == ARMMMUIdx_E10_0,
82
phys_ptr, attrs, &s2_prot,
83
- page_size, fi,
84
- cacheattrs != NULL ? &cacheattrs2 : NULL);
85
+ page_size, fi, &cacheattrs2);
86
fi->s2addr = ipa;
87
/* Combine the S1 and S2 perms. */
88
*prot &= s2_prot;
89
90
- /* Combine the S1 and S2 cache attributes, if needed */
91
- if (!ret && cacheattrs != NULL) {
92
- if (env->cp15.hcr_el2 & HCR_DC) {
93
- /*
94
- * HCR.DC forces the first stage attributes to
95
- * Normal Non-Shareable,
96
- * Inner Write-Back Read-Allocate Write-Allocate,
97
- * Outer Write-Back Read-Allocate Write-Allocate.
98
- */
99
- cacheattrs->attrs = 0xff;
100
- cacheattrs->shareability = 0;
101
- }
102
- *cacheattrs = combine_cacheattrs(*cacheattrs, cacheattrs2);
103
+ /* If S2 fails, return early. */
104
+ if (ret) {
105
+ return ret;
106
}
107
108
- return ret;
109
+ /* Combine the S1 and S2 cache attributes. */
110
+ if (env->cp15.hcr_el2 & HCR_DC) {
111
+ /*
112
+ * HCR.DC forces the first stage attributes to
113
+ * Normal Non-Shareable,
114
+ * Inner Write-Back Read-Allocate Write-Allocate,
115
+ * Outer Write-Back Read-Allocate Write-Allocate.
116
+ */
117
+ cacheattrs->attrs = 0xff;
118
+ cacheattrs->shareability = 0;
119
+ }
120
+ *cacheattrs = combine_cacheattrs(*cacheattrs, cacheattrs2);
121
+ return 0;
122
} else {
123
/*
124
* For non-EL2 CPUs a stage1+stage2 translation is just stage 1.
125
@@ -XXX,XX +XXX,XX @@ hwaddr arm_cpu_get_phys_page_attrs_debug(CPUState *cs, vaddr addr,
126
bool ret;
127
ARMMMUFaultInfo fi = {};
128
ARMMMUIdx mmu_idx = arm_mmu_idx(env);
129
+ ARMCacheAttrs cacheattrs = {};
130
131
*attrs = (MemTxAttrs) {};
132
133
ret = get_phys_addr(env, addr, 0, mmu_idx, &phys_addr,
134
- attrs, &prot, &page_size, &fi, NULL);
135
+ attrs, &prot, &page_size, &fi, &cacheattrs);
136
137
if (ret) {
138
return -1;
139
diff --git a/target/arm/m_helper.c b/target/arm/m_helper.c
140
index XXXXXXX..XXXXXXX 100644
141
--- a/target/arm/m_helper.c
142
+++ b/target/arm/m_helper.c
143
@@ -XXX,XX +XXX,XX @@ static bool v7m_stack_write(ARMCPU *cpu, uint32_t addr, uint32_t value,
144
hwaddr physaddr;
145
int prot;
146
ARMMMUFaultInfo fi = {};
147
+ ARMCacheAttrs cacheattrs = {};
148
bool secure = mmu_idx & ARM_MMU_IDX_M_S;
149
int exc;
150
bool exc_secure;
151
152
if (get_phys_addr(env, addr, MMU_DATA_STORE, mmu_idx, &physaddr,
153
- &attrs, &prot, &page_size, &fi, NULL)) {
154
+ &attrs, &prot, &page_size, &fi, &cacheattrs)) {
155
/* MPU/SAU lookup failed */
156
if (fi.type == ARMFault_QEMU_SFault) {
157
if (mode == STACK_LAZYFP) {
158
@@ -XXX,XX +XXX,XX @@ static bool v7m_stack_read(ARMCPU *cpu, uint32_t *dest, uint32_t addr,
159
hwaddr physaddr;
160
int prot;
161
ARMMMUFaultInfo fi = {};
162
+ ARMCacheAttrs cacheattrs = {};
163
bool secure = mmu_idx & ARM_MMU_IDX_M_S;
164
int exc;
165
bool exc_secure;
166
uint32_t value;
167
168
if (get_phys_addr(env, addr, MMU_DATA_LOAD, mmu_idx, &physaddr,
169
- &attrs, &prot, &page_size, &fi, NULL)) {
170
+ &attrs, &prot, &page_size, &fi, &cacheattrs)) {
171
/* MPU/SAU lookup failed */
172
if (fi.type == ARMFault_QEMU_SFault) {
173
qemu_log_mask(CPU_LOG_INT,
174
@@ -XXX,XX +XXX,XX @@ static bool v7m_read_half_insn(ARMCPU *cpu, ARMMMUIdx mmu_idx,
175
V8M_SAttributes sattrs = {};
176
MemTxAttrs attrs = {};
177
ARMMMUFaultInfo fi = {};
178
+ ARMCacheAttrs cacheattrs = {};
179
MemTxResult txres;
180
target_ulong page_size;
181
hwaddr physaddr;
182
@@ -XXX,XX +XXX,XX @@ static bool v7m_read_half_insn(ARMCPU *cpu, ARMMMUIdx mmu_idx,
183
"...really SecureFault with SFSR.INVEP\n");
184
return false;
185
}
186
- if (get_phys_addr(env, addr, MMU_INST_FETCH, mmu_idx,
187
- &physaddr, &attrs, &prot, &page_size, &fi, NULL)) {
188
+ if (get_phys_addr(env, addr, MMU_INST_FETCH, mmu_idx, &physaddr,
189
+ &attrs, &prot, &page_size, &fi, &cacheattrs)) {
190
/* the MPU lookup failed */
191
env->v7m.cfsr[env->v7m.secure] |= R_V7M_CFSR_IACCVIOL_MASK;
192
armv7m_nvic_set_pending(env->nvic, ARMV7M_EXCP_MEM, env->v7m.secure);
193
diff --git a/target/arm/tlb_helper.c b/target/arm/tlb_helper.c
194
index XXXXXXX..XXXXXXX 100644
195
--- a/target/arm/tlb_helper.c
196
+++ b/target/arm/tlb_helper.c
197
@@ -XXX,XX +XXX,XX @@ bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
198
int prot, ret;
199
MemTxAttrs attrs = {};
200
ARMMMUFaultInfo fi = {};
201
+ ARMCacheAttrs cacheattrs = {};
202
203
/*
204
* Walk the page table and (if the mapping exists) add the page
205
@@ -XXX,XX +XXX,XX @@ bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
206
*/
207
ret = get_phys_addr(&cpu->env, address, access_type,
208
core_to_arm_mmu_idx(&cpu->env, mmu_idx),
209
- &phys_addr, &attrs, &prot, &page_size, &fi, NULL);
210
+ &phys_addr, &attrs, &prot, &page_size,
211
+ &fi, &cacheattrs);
212
if (likely(!ret)) {
213
/*
214
* Map a single [sub]page. Regions smaller than our declared
215
--
216
2.20.1
217
218
diff view generated by jsdifflib
1
Implement the necessary support code for taking exceptions
1
From: Richard Henderson <richard.henderson@linaro.org>
2
to Hyp mode in AArch32.
3
2
3
This "bit" is a particular value of the page's MemAttr.
4
5
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200626033144.790098-43-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180820153020.21478-5-peter.maydell@linaro.org
9
---
9
---
10
target/arm/helper.c | 82 +++++++++++++++++++++++++++++++++++++++++++++
10
target/arm/helper.c | 48 ++++++++++++++++++++++++++++++++++++++---
11
1 file changed, 82 insertions(+)
11
target/arm/tlb_helper.c | 5 +++++
12
2 files changed, 50 insertions(+), 3 deletions(-)
12
13
13
diff --git a/target/arm/helper.c b/target/arm/helper.c
14
diff --git a/target/arm/helper.c b/target/arm/helper.c
14
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
15
--- a/target/arm/helper.c
16
--- a/target/arm/helper.c
16
+++ b/target/arm/helper.c
17
+++ b/target/arm/helper.c
17
@@ -XXX,XX +XXX,XX @@ static void take_aarch32_exception(CPUARMState *env, int new_mode,
18
@@ -XXX,XX +XXX,XX @@ static uint8_t combine_cacheattr_nibble(uint8_t s1, uint8_t s2)
18
env->regs[15] = newpc;
19
*/
19
}
20
static ARMCacheAttrs combine_cacheattrs(ARMCacheAttrs s1, ARMCacheAttrs s2)
20
21
{
21
+static void arm_cpu_do_interrupt_aarch32_hyp(CPUState *cs)
22
- uint8_t s1lo = extract32(s1.attrs, 0, 4), s2lo = extract32(s2.attrs, 0, 4);
22
+{
23
- uint8_t s1hi = extract32(s1.attrs, 4, 4), s2hi = extract32(s2.attrs, 4, 4);
23
+ /*
24
+ uint8_t s1lo, s2lo, s1hi, s2hi;
24
+ * Handle exception entry to Hyp mode; this is sufficiently
25
ARMCacheAttrs ret;
25
+ * different to entry to other AArch32 modes that we handle it
26
+ bool tagged = false;
26
+ * separately here.
27
+ *
28
+ * The vector table entry used is always the 0x14 Hyp mode entry point,
29
+ * unless this is an UNDEF/HVC/abort taken from Hyp to Hyp.
30
+ * The offset applied to the preferred return address is always zero
31
+ * (see DDI0487C.a section G1.12.3).
32
+ * PSTATE A/I/F masks are set based only on the SCR.EA/IRQ/FIQ values.
33
+ */
34
+ uint32_t addr, mask;
35
+ ARMCPU *cpu = ARM_CPU(cs);
36
+ CPUARMState *env = &cpu->env;
37
+
27
+
38
+ switch (cs->exception_index) {
28
+ if (s1.attrs == 0xf0) {
39
+ case EXCP_UDEF:
29
+ tagged = true;
40
+ addr = 0x04;
30
+ s1.attrs = 0xff;
41
+ break;
42
+ case EXCP_SWI:
43
+ addr = 0x14;
44
+ break;
45
+ case EXCP_BKPT:
46
+ /* Fall through to prefetch abort. */
47
+ case EXCP_PREFETCH_ABORT:
48
+ env->cp15.ifar_s = env->exception.vaddress;
49
+ qemu_log_mask(CPU_LOG_INT, "...with HIFAR 0x%x\n",
50
+ (uint32_t)env->exception.vaddress);
51
+ addr = 0x0c;
52
+ break;
53
+ case EXCP_DATA_ABORT:
54
+ env->cp15.dfar_s = env->exception.vaddress;
55
+ qemu_log_mask(CPU_LOG_INT, "...with HDFAR 0x%x\n",
56
+ (uint32_t)env->exception.vaddress);
57
+ addr = 0x10;
58
+ break;
59
+ case EXCP_IRQ:
60
+ addr = 0x18;
61
+ break;
62
+ case EXCP_FIQ:
63
+ addr = 0x1c;
64
+ break;
65
+ case EXCP_HVC:
66
+ addr = 0x08;
67
+ break;
68
+ case EXCP_HYP_TRAP:
69
+ addr = 0x14;
70
+ default:
71
+ cpu_abort(cs, "Unhandled exception 0x%x\n", cs->exception_index);
72
+ }
31
+ }
73
+
32
+
74
+ if (cs->exception_index != EXCP_IRQ && cs->exception_index != EXCP_FIQ) {
33
+ s1lo = extract32(s1.attrs, 0, 4);
75
+ env->cp15.esr_el[2] = env->exception.syndrome;
34
+ s2lo = extract32(s2.attrs, 0, 4);
35
+ s1hi = extract32(s1.attrs, 4, 4);
36
+ s2hi = extract32(s2.attrs, 4, 4);
37
38
/* Combine shareability attributes (table D4-43) */
39
if (s1.shareability == 2 || s2.shareability == 2) {
40
@@ -XXX,XX +XXX,XX @@ static ARMCacheAttrs combine_cacheattrs(ARMCacheAttrs s1, ARMCacheAttrs s2)
41
}
42
}
43
44
+ /* TODO: CombineS1S2Desc does not consider transient, only WB, RWA. */
45
+ if (tagged && ret.attrs == 0xff) {
46
+ ret.attrs = 0xf0;
76
+ }
47
+ }
77
+
48
+
78
+ if (arm_current_el(env) != 2 && addr < 0x14) {
49
return ret;
79
+ addr = 0x14;
50
}
80
+ }
51
52
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
53
* Normal Non-Shareable,
54
* Inner Write-Back Read-Allocate Write-Allocate,
55
* Outer Write-Back Read-Allocate Write-Allocate.
56
+ * Do not overwrite Tagged within attrs.
57
*/
58
- cacheattrs->attrs = 0xff;
59
+ if (cacheattrs->attrs != 0xf0) {
60
+ cacheattrs->attrs = 0xff;
61
+ }
62
cacheattrs->shareability = 0;
63
}
64
*cacheattrs = combine_cacheattrs(*cacheattrs, cacheattrs2);
65
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
66
/* Definitely a real MMU, not an MPU */
67
68
if (regime_translation_disabled(env, mmu_idx)) {
69
+ uint64_t hcr;
70
+ uint8_t memattr;
81
+
71
+
82
+ mask = 0;
72
/*
83
+ if (!(env->cp15.scr_el3 & SCR_EA)) {
73
* MMU disabled. S1 addresses within aa64 translation regimes are
84
+ mask |= CPSR_A;
74
* still checked for bounds -- see AArch64.TranslateAddressS1Off.
85
+ }
75
@@ -XXX,XX +XXX,XX @@ bool get_phys_addr(CPUARMState *env, target_ulong address,
86
+ if (!(env->cp15.scr_el3 & SCR_IRQ)) {
76
*phys_ptr = address;
87
+ mask |= CPSR_I;
77
*prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
88
+ }
78
*page_size = TARGET_PAGE_SIZE;
89
+ if (!(env->cp15.scr_el3 & SCR_FIQ)) {
90
+ mask |= CPSR_F;
91
+ }
92
+
79
+
93
+ addr += env->cp15.hvbar;
80
+ /* Fill in cacheattr a-la AArch64.TranslateAddressS1Off. */
81
+ hcr = arm_hcr_el2_eff(env);
82
+ cacheattrs->shareability = 0;
83
+ if (hcr & HCR_DC) {
84
+ if (hcr & HCR_DCT) {
85
+ memattr = 0xf0; /* Tagged, Normal, WB, RWA */
86
+ } else {
87
+ memattr = 0xff; /* Normal, WB, RWA */
88
+ }
89
+ } else if (access_type == MMU_INST_FETCH) {
90
+ if (regime_sctlr(env, mmu_idx) & SCTLR_I) {
91
+ memattr = 0xee; /* Normal, WT, RA, NT */
92
+ } else {
93
+ memattr = 0x44; /* Normal, NC, No */
94
+ }
95
+ cacheattrs->shareability = 2; /* outer sharable */
96
+ } else {
97
+ memattr = 0x00; /* Device, nGnRnE */
98
+ }
99
+ cacheattrs->attrs = memattr;
100
return 0;
101
}
102
103
diff --git a/target/arm/tlb_helper.c b/target/arm/tlb_helper.c
104
index XXXXXXX..XXXXXXX 100644
105
--- a/target/arm/tlb_helper.c
106
+++ b/target/arm/tlb_helper.c
107
@@ -XXX,XX +XXX,XX @@ bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
108
phys_addr &= TARGET_PAGE_MASK;
109
address &= TARGET_PAGE_MASK;
110
}
111
+ /* Notice and record tagged memory. */
112
+ if (cpu_isar_feature(aa64_mte, cpu) && cacheattrs.attrs == 0xf0) {
113
+ arm_tlb_mte_tagged(&attrs) = true;
114
+ }
94
+
115
+
95
+ take_aarch32_exception(env, ARM_CPU_MODE_HYP, mask, 0, addr);
116
tlb_set_page_with_attrs(cs, address, phys_addr, attrs,
96
+}
117
prot, mmu_idx, page_size);
97
+
118
return true;
98
static void arm_cpu_do_interrupt_aarch32(CPUState *cs)
99
{
100
ARMCPU *cpu = ARM_CPU(cs);
101
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_do_interrupt_aarch32(CPUState *cs)
102
env->cp15.mdscr_el1 = deposit64(env->cp15.mdscr_el1, 2, 4, moe);
103
}
104
105
+ if (env->exception.target_el == 2) {
106
+ arm_cpu_do_interrupt_aarch32_hyp(cs);
107
+ return;
108
+ }
109
+
110
/* TODO: Vectored interrupt controller. */
111
switch (cs->exception_index) {
112
case EXCP_UDEF:
113
--
119
--
114
2.18.0
120
2.20.1
115
121
116
122
diff view generated by jsdifflib
1
The MPS2 FPGAIO block includes some simple free-running counters.
1
From: Richard Henderson <richard.henderson@linaro.org>
2
Implement these.
2
3
3
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
4
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
Message-id: 20200626033144.790098-44-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
6
Message-id: 20180820141116.9118-2-peter.maydell@linaro.org
7
---
7
---
8
include/hw/misc/mps2-fpgaio.h | 4 +++
8
target/arm/cpu.h | 6 ++++++
9
hw/misc/mps2-fpgaio.c | 53 ++++++++++++++++++++++++++++++++++-
9
hw/arm/virt.c | 55 ++++++++++++++++++++++++++++++++++++++++++++++--
10
2 files changed, 56 insertions(+), 1 deletion(-)
10
target/arm/cpu.c | 52 +++++++++++++++++++++++++++++++++++++++++----
11
11
3 files changed, 107 insertions(+), 6 deletions(-)
12
diff --git a/include/hw/misc/mps2-fpgaio.h b/include/hw/misc/mps2-fpgaio.h
12
13
diff --git a/target/arm/cpu.h b/target/arm/cpu.h
13
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
14
--- a/include/hw/misc/mps2-fpgaio.h
15
--- a/target/arm/cpu.h
15
+++ b/include/hw/misc/mps2-fpgaio.h
16
+++ b/target/arm/cpu.h
16
@@ -XXX,XX +XXX,XX @@ typedef struct {
17
@@ -XXX,XX +XXX,XX @@ struct ARMCPU {
17
uint32_t misc;
18
/* MemoryRegion to use for secure physical accesses */
18
19
MemoryRegion *secure_memory;
19
uint32_t prescale_clk;
20
20
+
21
+ /* MemoryRegion to use for allocation tag accesses */
21
+ /* These hold the CLOCK_VIRTUAL ns tick when the CLK1HZ/CLK100HZ was zero */
22
+ MemoryRegion *tag_memory;
22
+ int64_t clk1hz_tick_offset;
23
+ MemoryRegion *secure_tag_memory;
23
+ int64_t clk100hz_tick_offset;
24
+
24
} MPS2FPGAIO;
25
/* For v8M, pointer to the IDAU interface provided by board/SoC */
25
26
Object *idau;
26
#endif
27
27
diff --git a/hw/misc/mps2-fpgaio.c b/hw/misc/mps2-fpgaio.c
28
@@ -XXX,XX +XXX,XX @@ typedef enum ARMMMUIdxBit {
29
typedef enum ARMASIdx {
30
ARMASIdx_NS = 0,
31
ARMASIdx_S = 1,
32
+ ARMASIdx_TagNS = 2,
33
+ ARMASIdx_TagS = 3,
34
} ARMASIdx;
35
36
/* Return the Exception Level targeted by debug exceptions. */
37
diff --git a/hw/arm/virt.c b/hw/arm/virt.c
28
index XXXXXXX..XXXXXXX 100644
38
index XXXXXXX..XXXXXXX 100644
29
--- a/hw/misc/mps2-fpgaio.c
39
--- a/hw/arm/virt.c
30
+++ b/hw/misc/mps2-fpgaio.c
40
+++ b/hw/arm/virt.c
31
@@ -XXX,XX +XXX,XX @@
41
@@ -XXX,XX +XXX,XX @@ static void create_platform_bus(VirtMachineState *vms)
32
#include "hw/sysbus.h"
42
sysbus_mmio_get_region(s, 0));
33
#include "hw/registerfields.h"
43
}
34
#include "hw/misc/mps2-fpgaio.h"
44
35
+#include "qemu/timer.h"
45
+static void create_tag_ram(MemoryRegion *tag_sysmem,
36
46
+ hwaddr base, hwaddr size,
37
REG32(LED0, 0)
47
+ const char *name)
38
REG32(BUTTON, 8)
39
@@ -XXX,XX +XXX,XX @@ REG32(PRESCALE, 0x1c)
40
REG32(PSCNTR, 0x20)
41
REG32(MISC, 0x4c)
42
43
+static uint32_t counter_from_tickoff(int64_t now, int64_t tick_offset, int frq)
44
+{
48
+{
45
+ return muldiv64(now - tick_offset, frq, NANOSECONDS_PER_SECOND);
49
+ MemoryRegion *tagram = g_new(MemoryRegion, 1);
50
+
51
+ memory_region_init_ram(tagram, NULL, name, size / 32, &error_fatal);
52
+ memory_region_add_subregion(tag_sysmem, base / 32, tagram);
46
+}
53
+}
47
+
54
+
48
+static int64_t tickoff_from_counter(int64_t now, uint32_t count, int frq)
55
static void create_secure_ram(VirtMachineState *vms,
49
+{
56
- MemoryRegion *secure_sysmem)
50
+ return now - muldiv64(count, NANOSECONDS_PER_SECOND, frq);
57
+ MemoryRegion *secure_sysmem,
51
+}
58
+ MemoryRegion *secure_tag_sysmem)
52
+
53
static uint64_t mps2_fpgaio_read(void *opaque, hwaddr offset, unsigned size)
54
{
59
{
55
MPS2FPGAIO *s = MPS2_FPGAIO(opaque);
60
MemoryRegion *secram = g_new(MemoryRegion, 1);
56
uint64_t r;
61
char *nodename;
57
+ int64_t now;
62
@@ -XXX,XX +XXX,XX @@ static void create_secure_ram(VirtMachineState *vms,
58
63
qemu_fdt_setprop_string(vms->fdt, nodename, "status", "disabled");
59
switch (offset) {
64
qemu_fdt_setprop_string(vms->fdt, nodename, "secure-status", "okay");
60
case A_LED0:
65
61
@@ -XXX,XX +XXX,XX @@ static uint64_t mps2_fpgaio_read(void *opaque, hwaddr offset, unsigned size)
66
+ if (secure_tag_sysmem) {
62
r = s->misc;
67
+ create_tag_ram(secure_tag_sysmem, base, size, "mach-virt.secure-tag");
63
break;
68
+ }
64
case A_CLK1HZ:
69
+
65
+ now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
70
g_free(nodename);
66
+ r = counter_from_tickoff(now, s->clk1hz_tick_offset, 1);
67
+ break;
68
case A_CLK100HZ:
69
+ now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
70
+ r = counter_from_tickoff(now, s->clk100hz_tick_offset, 100);
71
+ break;
72
case A_COUNTER:
73
case A_PSCNTR:
74
- /* These are all upcounters of various frequencies. */
75
qemu_log_mask(LOG_UNIMP, "MPS2 FPGAIO: counters unimplemented\n");
76
r = 0;
77
break;
78
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_write(void *opaque, hwaddr offset, uint64_t value,
79
unsigned size)
80
{
81
MPS2FPGAIO *s = MPS2_FPGAIO(opaque);
82
+ int64_t now;
83
84
trace_mps2_fpgaio_write(offset, value, size);
85
86
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_write(void *opaque, hwaddr offset, uint64_t value,
87
"MPS2 FPGAIO: MISC control bits unimplemented\n");
88
s->misc = value;
89
break;
90
+ case A_CLK1HZ:
91
+ now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
92
+ s->clk1hz_tick_offset = tickoff_from_counter(now, value, 1);
93
+ break;
94
+ case A_CLK100HZ:
95
+ now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
96
+ s->clk100hz_tick_offset = tickoff_from_counter(now, value, 100);
97
+ break;
98
default:
99
qemu_log_mask(LOG_GUEST_ERROR,
100
"MPS2 FPGAIO write: bad offset 0x%x\n", (int) offset);
101
@@ -XXX,XX +XXX,XX @@ static const MemoryRegionOps mps2_fpgaio_ops = {
102
static void mps2_fpgaio_reset(DeviceState *dev)
103
{
104
MPS2FPGAIO *s = MPS2_FPGAIO(dev);
105
+ int64_t now = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
106
107
trace_mps2_fpgaio_reset();
108
s->led0 = 0;
109
s->prescale = 0;
110
s->misc = 0;
111
+ s->clk1hz_tick_offset = tickoff_from_counter(now, 0, 1);
112
+ s->clk100hz_tick_offset = tickoff_from_counter(now, 0, 100);
113
}
71
}
114
72
115
static void mps2_fpgaio_init(Object *obj)
73
@@ -XXX,XX +XXX,XX @@ static void machvirt_init(MachineState *machine)
116
@@ -XXX,XX +XXX,XX @@ static void mps2_fpgaio_init(Object *obj)
74
const CPUArchIdList *possible_cpus;
117
sysbus_init_mmio(sbd, &s->iomem);
75
MemoryRegion *sysmem = get_system_memory();
76
MemoryRegion *secure_sysmem = NULL;
77
+ MemoryRegion *tag_sysmem = NULL;
78
+ MemoryRegion *secure_tag_sysmem = NULL;
79
int n, virt_max_cpus;
80
bool firmware_loaded;
81
bool aarch64 = true;
82
@@ -XXX,XX +XXX,XX @@ static void machvirt_init(MachineState *machine)
83
"secure-memory", &error_abort);
84
}
85
86
+ /*
87
+ * The cpu adds the property if and only if MemTag is supported.
88
+ * If it is, we must allocate the ram to back that up.
89
+ */
90
+ if (object_property_find(cpuobj, "tag-memory", NULL)) {
91
+ if (!tag_sysmem) {
92
+ tag_sysmem = g_new(MemoryRegion, 1);
93
+ memory_region_init(tag_sysmem, OBJECT(machine),
94
+ "tag-memory", UINT64_MAX / 32);
95
+
96
+ if (vms->secure) {
97
+ secure_tag_sysmem = g_new(MemoryRegion, 1);
98
+ memory_region_init(secure_tag_sysmem, OBJECT(machine),
99
+ "secure-tag-memory", UINT64_MAX / 32);
100
+
101
+ /* As with ram, secure-tag takes precedence over tag. */
102
+ memory_region_add_subregion_overlap(secure_tag_sysmem, 0,
103
+ tag_sysmem, -1);
104
+ }
105
+ }
106
+
107
+ object_property_set_link(cpuobj, OBJECT(tag_sysmem),
108
+ "tag-memory", &error_abort);
109
+ if (vms->secure) {
110
+ object_property_set_link(cpuobj, OBJECT(secure_tag_sysmem),
111
+ "secure-tag-memory", &error_abort);
112
+ }
113
+ }
114
+
115
qdev_realize(DEVICE(cpuobj), NULL, &error_fatal);
116
object_unref(cpuobj);
117
}
118
@@ -XXX,XX +XXX,XX @@ static void machvirt_init(MachineState *machine)
119
create_uart(vms, VIRT_UART, sysmem, serial_hd(0));
120
121
if (vms->secure) {
122
- create_secure_ram(vms, secure_sysmem);
123
+ create_secure_ram(vms, secure_sysmem, secure_tag_sysmem);
124
create_uart(vms, VIRT_SECURE_UART, secure_sysmem, serial_hd(1));
125
}
126
127
+ if (tag_sysmem) {
128
+ create_tag_ram(tag_sysmem, vms->memmap[VIRT_MEM].base,
129
+ machine->ram_size, "mach-virt.tag");
130
+ }
131
+
132
vms->highmem_ecam &= vms->highmem && (!firmware_loaded || aarch64);
133
134
create_rtc(vms);
135
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
136
index XXXXXXX..XXXXXXX 100644
137
--- a/target/arm/cpu.c
138
+++ b/target/arm/cpu.c
139
@@ -XXX,XX +XXX,XX @@ void arm_cpu_post_init(Object *obj)
140
if (kvm_enabled()) {
141
kvm_arm_add_vcpu_properties(obj);
142
}
143
+
144
+#ifndef CONFIG_USER_ONLY
145
+ if (arm_feature(&cpu->env, ARM_FEATURE_AARCH64) &&
146
+ cpu_isar_feature(aa64_mte, cpu)) {
147
+ object_property_add_link(obj, "tag-memory",
148
+ TYPE_MEMORY_REGION,
149
+ (Object **)&cpu->tag_memory,
150
+ qdev_prop_allow_set_link_before_realize,
151
+ OBJ_PROP_LINK_STRONG);
152
+
153
+ if (arm_feature(&cpu->env, ARM_FEATURE_EL3)) {
154
+ object_property_add_link(obj, "secure-tag-memory",
155
+ TYPE_MEMORY_REGION,
156
+ (Object **)&cpu->secure_tag_memory,
157
+ qdev_prop_allow_set_link_before_realize,
158
+ OBJ_PROP_LINK_STRONG);
159
+ }
160
+ }
161
+#endif
118
}
162
}
119
163
120
+static bool mps2_fpgaio_counters_needed(void *opaque)
164
static void arm_cpu_finalizefn(Object *obj)
121
+{
165
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_realizefn(DeviceState *dev, Error **errp)
122
+ /* Currently vmstate.c insists all subsections have a 'needed' function */
166
#ifndef CONFIG_USER_ONLY
123
+ return true;
167
MachineState *ms = MACHINE(qdev_get_machine());
124
+}
168
unsigned int smp_cpus = ms->smp.cpus;
125
+
169
+ bool has_secure = cpu->has_el3 || arm_feature(env, ARM_FEATURE_M_SECURITY);
126
+static const VMStateDescription mps2_fpgaio_counters_vmstate = {
170
127
+ .name = "mps2-fpgaio/counters",
171
- if (cpu->has_el3 || arm_feature(env, ARM_FEATURE_M_SECURITY)) {
128
+ .version_id = 1,
172
- cs->num_ases = 2;
129
+ .minimum_version_id = 1,
173
+ /*
130
+ .needed = mps2_fpgaio_counters_needed,
174
+ * We must set cs->num_ases to the final value before
131
+ .fields = (VMStateField[]) {
175
+ * the first call to cpu_address_space_init.
132
+ VMSTATE_INT64(clk1hz_tick_offset, MPS2FPGAIO),
176
+ */
133
+ VMSTATE_INT64(clk100hz_tick_offset, MPS2FPGAIO),
177
+ if (cpu->tag_memory != NULL) {
134
+ VMSTATE_END_OF_LIST()
178
+ cs->num_ases = 3 + has_secure;
135
+ }
179
+ } else {
136
+};
180
+ cs->num_ases = 1 + has_secure;
137
+
181
+ }
138
static const VMStateDescription mps2_fpgaio_vmstate = {
182
139
.name = "mps2-fpgaio",
183
+ if (has_secure) {
140
.version_id = 1,
184
if (!cpu->secure_memory) {
141
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription mps2_fpgaio_vmstate = {
185
cpu->secure_memory = cs->memory;
142
VMSTATE_UINT32(prescale, MPS2FPGAIO),
186
}
143
VMSTATE_UINT32(misc, MPS2FPGAIO),
187
cpu_address_space_init(cs, ARMASIdx_S, "cpu-secure-memory",
144
VMSTATE_END_OF_LIST()
188
cpu->secure_memory);
145
+ },
189
- } else {
146
+ .subsections = (const VMStateDescription*[]) {
190
- cs->num_ases = 1;
147
+ &mps2_fpgaio_counters_vmstate,
191
}
148
+ NULL
192
+
149
}
193
+ if (cpu->tag_memory != NULL) {
150
};
194
+ cpu_address_space_init(cs, ARMASIdx_TagNS, "cpu-tag-memory",
151
195
+ cpu->tag_memory);
196
+ if (has_secure) {
197
+ cpu_address_space_init(cs, ARMASIdx_TagS, "cpu-tag-memory",
198
+ cpu->secure_tag_memory);
199
+ }
200
+ } else if (cpu_isar_feature(aa64_mte, cpu)) {
201
+ /*
202
+ * Since there is no tag memory, we can't meaningfully support MTE
203
+ * to its fullest. To avoid problems later, when we would come to
204
+ * use the tag memory, downgrade support to insns only.
205
+ */
206
+ cpu->isar.id_aa64pfr1 =
207
+ FIELD_DP64(cpu->isar.id_aa64pfr1, ID_AA64PFR1, MTE, 1);
208
+ }
209
+
210
cpu_address_space_init(cs, ARMASIdx_NS, "cpu-memory", cs->memory);
211
212
/* No core_count specified, default to smp_cpus. */
152
--
213
--
153
2.18.0
214
2.20.1
154
215
155
216
diff view generated by jsdifflib
1
The Arm Cortex-M System Design Kit includes a "dual-input timer module"
1
From: Richard Henderson <richard.henderson@linaro.org>
2
which combines two programmable down-counters. Implement a model
3
of this device.
4
2
3
Look up the physical address for the given virtual address,
4
convert that to a tag physical address, and finally return
5
the host address that backs it.
6
7
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20200626033144.790098-45-richard.henderson@linaro.org
9
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180820141116.9118-4-peter.maydell@linaro.org
8
---
11
---
9
hw/timer/Makefile.objs | 1 +
12
target/arm/mte_helper.c | 131 ++++++++++++++++++++++++++++++++++++++++
10
include/hw/timer/cmsdk-apb-dualtimer.h | 72 ++++
13
1 file changed, 131 insertions(+)
11
hw/timer/cmsdk-apb-dualtimer.c | 515 +++++++++++++++++++++++++
12
MAINTAINERS | 2 +
13
default-configs/arm-softmmu.mak | 1 +
14
hw/timer/trace-events | 5 +
15
6 files changed, 596 insertions(+)
16
create mode 100644 include/hw/timer/cmsdk-apb-dualtimer.h
17
create mode 100644 hw/timer/cmsdk-apb-dualtimer.c
18
14
19
diff --git a/hw/timer/Makefile.objs b/hw/timer/Makefile.objs
15
diff --git a/target/arm/mte_helper.c b/target/arm/mte_helper.c
20
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
21
--- a/hw/timer/Makefile.objs
17
--- a/target/arm/mte_helper.c
22
+++ b/hw/timer/Makefile.objs
18
+++ b/target/arm/mte_helper.c
23
@@ -XXX,XX +XXX,XX @@ common-obj-$(CONFIG_ASPEED_SOC) += aspeed_timer.o
24
25
common-obj-$(CONFIG_SUN4V_RTC) += sun4v-rtc.o
26
common-obj-$(CONFIG_CMSDK_APB_TIMER) += cmsdk-apb-timer.o
27
+common-obj-$(CONFIG_CMSDK_APB_DUALTIMER) += cmsdk-apb-dualtimer.o
28
common-obj-$(CONFIG_MSF2) += mss-timer.o
29
diff --git a/include/hw/timer/cmsdk-apb-dualtimer.h b/include/hw/timer/cmsdk-apb-dualtimer.h
30
new file mode 100644
31
index XXXXXXX..XXXXXXX
32
--- /dev/null
33
+++ b/include/hw/timer/cmsdk-apb-dualtimer.h
34
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@
35
+/*
20
#include "cpu.h"
36
+ * ARM CMSDK APB dual-timer emulation
21
#include "internals.h"
37
+ *
22
#include "exec/exec-all.h"
38
+ * Copyright (c) 2018 Linaro Limited
23
+#include "exec/ram_addr.h"
39
+ * Written by Peter Maydell
24
#include "exec/cpu_ldst.h"
40
+ *
25
#include "exec/helper-proto.h"
41
+ * This program is free software; you can redistribute it and/or modify
26
42
+ * it under the terms of the GNU General Public License version 2 or
27
@@ -XXX,XX +XXX,XX @@ static uint8_t *allocation_tag_mem(CPUARMState *env, int ptr_mmu_idx,
43
+ * (at your option) any later version.
28
int ptr_size, MMUAccessType tag_access,
44
+ */
29
int tag_size, uintptr_t ra)
30
{
31
+#ifdef CONFIG_USER_ONLY
32
/* Tag storage not implemented. */
33
return NULL;
34
+#else
35
+ uintptr_t index;
36
+ CPUIOTLBEntry *iotlbentry;
37
+ int in_page, flags;
38
+ ram_addr_t ptr_ra;
39
+ hwaddr ptr_paddr, tag_paddr, xlat;
40
+ MemoryRegion *mr;
41
+ ARMASIdx tag_asi;
42
+ AddressSpace *tag_as;
43
+ void *host;
45
+
44
+
46
+/*
45
+ /*
47
+ * This is a model of the "APB dual-input timer" which is part of the Cortex-M
46
+ * Probe the first byte of the virtual address. This raises an
48
+ * System Design Kit (CMSDK) and documented in the Cortex-M System
47
+ * exception for inaccessible pages, and resolves the virtual address
49
+ * Design Kit Technical Reference Manual (ARM DDI0479C):
48
+ * into the softmmu tlb.
50
+ * https://developer.arm.com/products/system-design/system-design-kits/cortex-m-system-design-kit
49
+ *
51
+ *
50
+ * When RA == 0, this is for mte_probe1. The page is expected to be
52
+ * QEMU interface:
51
+ * valid. Indicate to probe_access_flags no-fault, then assert that
53
+ * + QOM property "pclk-frq": frequency at which the timer is clocked
52
+ * we received a valid page.
54
+ * + sysbus MMIO region 0: the register bank
53
+ */
55
+ * + sysbus IRQ 0: combined timer interrupt TIMINTC
54
+ flags = probe_access_flags(env, ptr, ptr_access, ptr_mmu_idx,
56
+ * + sysbus IRO 1: timer block 1 interrupt TIMINT1
55
+ ra == 0, &host, ra);
57
+ * + sysbus IRQ 2: timer block 2 interrupt TIMINT2
56
+ assert(!(flags & TLB_INVALID_MASK));
58
+ */
59
+
57
+
60
+#ifndef CMSDK_APB_DUALTIMER_H
58
+ /*
61
+#define CMSDK_APB_DUALTIMER_H
59
+ * Find the iotlbentry for ptr. This *must* be present in the TLB
60
+ * because we just found the mapping.
61
+ * TODO: Perhaps there should be a cputlb helper that returns a
62
+ * matching tlb entry + iotlb entry.
63
+ */
64
+ index = tlb_index(env, ptr_mmu_idx, ptr);
65
+# ifdef CONFIG_DEBUG_TCG
66
+ {
67
+ CPUTLBEntry *entry = tlb_entry(env, ptr_mmu_idx, ptr);
68
+ target_ulong comparator = (ptr_access == MMU_DATA_LOAD
69
+ ? entry->addr_read
70
+ : tlb_addr_write(entry));
71
+ g_assert(tlb_hit(comparator, ptr));
72
+ }
73
+# endif
74
+ iotlbentry = &env_tlb(env)->d[ptr_mmu_idx].iotlb[index];
62
+
75
+
63
+#include "hw/sysbus.h"
76
+ /* If the virtual page MemAttr != Tagged, access unchecked. */
64
+#include "hw/ptimer.h"
77
+ if (!arm_tlb_mte_tagged(&iotlbentry->attrs)) {
65
+
78
+ return NULL;
66
+#define TYPE_CMSDK_APB_DUALTIMER "cmsdk-apb-dualtimer"
67
+#define CMSDK_APB_DUALTIMER(obj) OBJECT_CHECK(CMSDKAPBDualTimer, (obj), \
68
+ TYPE_CMSDK_APB_DUALTIMER)
69
+
70
+typedef struct CMSDKAPBDualTimer CMSDKAPBDualTimer;
71
+
72
+/* One of the two identical timer modules in the dual-timer module */
73
+typedef struct CMSDKAPBDualTimerModule {
74
+ CMSDKAPBDualTimer *parent;
75
+ struct ptimer_state *timer;
76
+ qemu_irq timerint;
77
+ /*
78
+ * We must track the guest LOAD and VALUE register state by hand
79
+ * rather than leaving this state only in the ptimer limit/count,
80
+ * because if CONTROL.SIZE is 0 then only the low 16 bits of the
81
+ * counter actually counts, but the high half is still guest
82
+ * accessible.
83
+ */
84
+ uint32_t load;
85
+ uint32_t value;
86
+ uint32_t control;
87
+ uint32_t intstatus;
88
+} CMSDKAPBDualTimerModule;
89
+
90
+#define CMSDK_APB_DUALTIMER_NUM_MODULES 2
91
+
92
+struct CMSDKAPBDualTimer {
93
+ /*< private >*/
94
+ SysBusDevice parent_obj;
95
+
96
+ /*< public >*/
97
+ MemoryRegion iomem;
98
+ qemu_irq timerintc;
99
+ uint32_t pclk_frq;
100
+
101
+ CMSDKAPBDualTimerModule timermod[CMSDK_APB_DUALTIMER_NUM_MODULES];
102
+ uint32_t timeritcr;
103
+ uint32_t timeritop;
104
+};
105
+
106
+#endif
107
diff --git a/hw/timer/cmsdk-apb-dualtimer.c b/hw/timer/cmsdk-apb-dualtimer.c
108
new file mode 100644
109
index XXXXXXX..XXXXXXX
110
--- /dev/null
111
+++ b/hw/timer/cmsdk-apb-dualtimer.c
112
@@ -XXX,XX +XXX,XX @@
113
+/*
114
+ * ARM CMSDK APB dual-timer emulation
115
+ *
116
+ * Copyright (c) 2018 Linaro Limited
117
+ * Written by Peter Maydell
118
+ *
119
+ * This program is free software; you can redistribute it and/or modify
120
+ * it under the terms of the GNU General Public License version 2 or
121
+ * (at your option) any later version.
122
+ */
123
+
124
+/*
125
+ * This is a model of the "APB dual-input timer" which is part of the Cortex-M
126
+ * System Design Kit (CMSDK) and documented in the Cortex-M System
127
+ * Design Kit Technical Reference Manual (ARM DDI0479C):
128
+ * https://developer.arm.com/products/system-design/system-design-kits/cortex-m-system-design-kit
129
+ */
130
+
131
+#include "qemu/osdep.h"
132
+#include "qemu/log.h"
133
+#include "trace.h"
134
+#include "qapi/error.h"
135
+#include "qemu/main-loop.h"
136
+#include "hw/sysbus.h"
137
+#include "hw/registerfields.h"
138
+#include "hw/timer/cmsdk-apb-dualtimer.h"
139
+
140
+REG32(TIMER1LOAD, 0x0)
141
+REG32(TIMER1VALUE, 0x4)
142
+REG32(TIMER1CONTROL, 0x8)
143
+ FIELD(CONTROL, ONESHOT, 0, 1)
144
+ FIELD(CONTROL, SIZE, 1, 1)
145
+ FIELD(CONTROL, PRESCALE, 2, 2)
146
+ FIELD(CONTROL, INTEN, 5, 1)
147
+ FIELD(CONTROL, MODE, 6, 1)
148
+ FIELD(CONTROL, ENABLE, 7, 1)
149
+#define R_CONTROL_VALID_MASK (R_CONTROL_ONESHOT_MASK | R_CONTROL_SIZE_MASK | \
150
+ R_CONTROL_PRESCALE_MASK | R_CONTROL_INTEN_MASK | \
151
+ R_CONTROL_MODE_MASK | R_CONTROL_ENABLE_MASK)
152
+REG32(TIMER1INTCLR, 0xc)
153
+REG32(TIMER1RIS, 0x10)
154
+REG32(TIMER1MIS, 0x14)
155
+REG32(TIMER1BGLOAD, 0x18)
156
+REG32(TIMER2LOAD, 0x20)
157
+REG32(TIMER2VALUE, 0x24)
158
+REG32(TIMER2CONTROL, 0x28)
159
+REG32(TIMER2INTCLR, 0x2c)
160
+REG32(TIMER2RIS, 0x30)
161
+REG32(TIMER2MIS, 0x34)
162
+REG32(TIMER2BGLOAD, 0x38)
163
+REG32(TIMERITCR, 0xf00)
164
+ FIELD(TIMERITCR, ENABLE, 0, 1)
165
+#define R_TIMERITCR_VALID_MASK R_TIMERITCR_ENABLE_MASK
166
+REG32(TIMERITOP, 0xf04)
167
+ FIELD(TIMERITOP, TIMINT1, 0, 1)
168
+ FIELD(TIMERITOP, TIMINT2, 1, 1)
169
+#define R_TIMERITOP_VALID_MASK (R_TIMERITOP_TIMINT1_MASK | \
170
+ R_TIMERITOP_TIMINT2_MASK)
171
+REG32(PID4, 0xfd0)
172
+REG32(PID5, 0xfd4)
173
+REG32(PID6, 0xfd8)
174
+REG32(PID7, 0xfdc)
175
+REG32(PID0, 0xfe0)
176
+REG32(PID1, 0xfe4)
177
+REG32(PID2, 0xfe8)
178
+REG32(PID3, 0xfec)
179
+REG32(CID0, 0xff0)
180
+REG32(CID1, 0xff4)
181
+REG32(CID2, 0xff8)
182
+REG32(CID3, 0xffc)
183
+
184
+/* PID/CID values */
185
+static const int timer_id[] = {
186
+ 0x04, 0x00, 0x00, 0x00, /* PID4..PID7 */
187
+ 0x23, 0xb8, 0x1b, 0x00, /* PID0..PID3 */
188
+ 0x0d, 0xf0, 0x05, 0xb1, /* CID0..CID3 */
189
+};
190
+
191
+static bool cmsdk_dualtimermod_intstatus(CMSDKAPBDualTimerModule *m)
192
+{
193
+ /* Return masked interrupt status for the timer module */
194
+ return m->intstatus && (m->control & R_CONTROL_INTEN_MASK);
195
+}
196
+
197
+static void cmsdk_apb_dualtimer_update(CMSDKAPBDualTimer *s)
198
+{
199
+ bool timint1, timint2, timintc;
200
+
201
+ if (s->timeritcr) {
202
+ /* Integration test mode: outputs driven directly from TIMERITOP bits */
203
+ timint1 = s->timeritop & R_TIMERITOP_TIMINT1_MASK;
204
+ timint2 = s->timeritop & R_TIMERITOP_TIMINT2_MASK;
205
+ } else {
206
+ timint1 = cmsdk_dualtimermod_intstatus(&s->timermod[0]);
207
+ timint2 = cmsdk_dualtimermod_intstatus(&s->timermod[1]);
208
+ }
79
+ }
209
+
80
+
210
+ timintc = timint1 || timint2;
81
+ /*
211
+
82
+ * If not backed by host ram, there is no tag storage: access unchecked.
212
+ qemu_set_irq(s->timermod[0].timerint, timint1);
83
+ * This is probably a guest os bug though, so log it.
213
+ qemu_set_irq(s->timermod[1].timerint, timint2);
84
+ */
214
+ qemu_set_irq(s->timerintc, timintc);
85
+ if (unlikely(flags & TLB_MMIO)) {
215
+}
86
+ qemu_log_mask(LOG_GUEST_ERROR,
216
+
87
+ "Page @ 0x%" PRIx64 " indicates Tagged Normal memory "
217
+static void cmsdk_dualtimermod_write_control(CMSDKAPBDualTimerModule *m,
88
+ "but is not backed by host ram\n", ptr);
218
+ uint32_t newctrl)
89
+ return NULL;
219
+{
220
+ /* Handle a write to the CONTROL register */
221
+ uint32_t changed;
222
+
223
+ newctrl &= R_CONTROL_VALID_MASK;
224
+
225
+ changed = m->control ^ newctrl;
226
+
227
+ if (changed & ~newctrl & R_CONTROL_ENABLE_MASK) {
228
+ /* ENABLE cleared, stop timer before any further changes */
229
+ ptimer_stop(m->timer);
230
+ }
90
+ }
231
+
91
+
232
+ if (changed & R_CONTROL_PRESCALE_MASK) {
92
+ /*
233
+ int divisor;
93
+ * The Normal memory access can extend to the next page. E.g. a single
234
+
94
+ * 8-byte access to the last byte of a page will check only the last
235
+ switch (FIELD_EX32(newctrl, CONTROL, PRESCALE)) {
95
+ * tag on the first page.
236
+ case 0:
96
+ * Any page access exception has priority over tag check exception.
237
+ divisor = 1;
97
+ */
238
+ break;
98
+ in_page = -(ptr | TARGET_PAGE_MASK);
239
+ case 1:
99
+ if (unlikely(ptr_size > in_page)) {
240
+ divisor = 16;
100
+ void *ignore;
241
+ break;
101
+ flags |= probe_access_flags(env, ptr + in_page, ptr_access,
242
+ case 2:
102
+ ptr_mmu_idx, ra == 0, &ignore, ra);
243
+ divisor = 256;
103
+ assert(!(flags & TLB_INVALID_MASK));
244
+ break;
245
+ case 3:
246
+ /* UNDEFINED; complain, and arbitrarily treat like 2 */
247
+ qemu_log_mask(LOG_GUEST_ERROR,
248
+ "CMSDK APB dual-timer: CONTROL.PRESCALE==0b11"
249
+ " is undefined behaviour\n");
250
+ divisor = 256;
251
+ break;
252
+ default:
253
+ g_assert_not_reached();
254
+ }
255
+ ptimer_set_freq(m->timer, m->parent->pclk_frq / divisor);
256
+ }
104
+ }
257
+
105
+
258
+ if (changed & R_CONTROL_MODE_MASK) {
106
+ /* Any debug exception has priority over a tag check exception. */
259
+ uint32_t load;
107
+ if (unlikely(flags & TLB_WATCHPOINT)) {
260
+ if (newctrl & R_CONTROL_MODE_MASK) {
108
+ int wp = ptr_access == MMU_DATA_LOAD ? BP_MEM_READ : BP_MEM_WRITE;
261
+ /* Periodic: the limit is the LOAD register value */
109
+ assert(ra != 0);
262
+ load = m->load;
110
+ cpu_check_watchpoint(env_cpu(env), ptr, ptr_size,
263
+ } else {
111
+ iotlbentry->attrs, wp, ra);
264
+ /* Free-running: counter wraps around */
265
+ load = ptimer_get_limit(m->timer);
266
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
267
+ load = deposit32(m->load, 0, 16, load);
268
+ }
269
+ m->load = load;
270
+ load = 0xffffffff;
271
+ }
272
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
273
+ load &= 0xffff;
274
+ }
275
+ ptimer_set_limit(m->timer, load, 0);
276
+ }
112
+ }
277
+
113
+
278
+ if (changed & R_CONTROL_SIZE_MASK) {
114
+ /*
279
+ /* Timer switched between 16 and 32 bit count */
115
+ * Find the physical address within the normal mem space.
280
+ uint32_t value, load;
116
+ * The memory region lookup must succeed because TLB_MMIO was
117
+ * not set in the cputlb lookup above.
118
+ */
119
+ mr = memory_region_from_host(host, &ptr_ra);
120
+ tcg_debug_assert(mr != NULL);
121
+ tcg_debug_assert(memory_region_is_ram(mr));
122
+ ptr_paddr = ptr_ra;
123
+ do {
124
+ ptr_paddr += mr->addr;
125
+ mr = mr->container;
126
+ } while (mr);
281
+
127
+
282
+ value = ptimer_get_count(m->timer);
128
+ /* Convert to the physical address in tag space. */
283
+ load = ptimer_get_limit(m->timer);
129
+ tag_paddr = ptr_paddr >> (LOG2_TAG_GRANULE + 1);
284
+ if (newctrl & R_CONTROL_SIZE_MASK) {
285
+ /* 16 -> 32, top half of VALUE is in struct field */
286
+ value = deposit32(m->value, 0, 16, value);
287
+ } else {
288
+ /* 32 -> 16: save top half to struct field and truncate */
289
+ m->value = value;
290
+ value &= 0xffff;
291
+ }
292
+
130
+
293
+ if (newctrl & R_CONTROL_MODE_MASK) {
131
+ /* Look up the address in tag space. */
294
+ /* Periodic, timer limit has LOAD value */
132
+ tag_asi = iotlbentry->attrs.secure ? ARMASIdx_TagS : ARMASIdx_TagNS;
295
+ if (newctrl & R_CONTROL_SIZE_MASK) {
133
+ tag_as = cpu_get_address_space(env_cpu(env), tag_asi);
296
+ load = deposit32(m->load, 0, 16, load);
134
+ mr = address_space_translate(tag_as, tag_paddr, &xlat, NULL,
297
+ } else {
135
+ tag_access == MMU_DATA_STORE,
298
+ m->load = load;
136
+ iotlbentry->attrs);
299
+ load &= 0xffff;
137
+
300
+ }
138
+ /*
301
+ } else {
139
+ * Note that @mr will never be NULL. If there is nothing in the address
302
+ /* Free-running, timer limit is set to give wraparound */
140
+ * space at @tag_paddr, the translation will return the unallocated memory
303
+ if (newctrl & R_CONTROL_SIZE_MASK) {
141
+ * region. For our purposes, the result must be ram.
304
+ load = 0xffffffff;
142
+ */
305
+ } else {
143
+ if (unlikely(!memory_region_is_ram(mr))) {
306
+ load = 0xffff;
144
+ /* ??? Failure is a board configuration error. */
307
+ }
145
+ qemu_log_mask(LOG_UNIMP,
308
+ }
146
+ "Tag Memory @ 0x%" HWADDR_PRIx " not found for "
309
+ ptimer_set_count(m->timer, value);
147
+ "Normal Memory @ 0x%" HWADDR_PRIx "\n",
310
+ ptimer_set_limit(m->timer, load, 0);
148
+ tag_paddr, ptr_paddr);
149
+ return NULL;
311
+ }
150
+ }
312
+
151
+
313
+ if (newctrl & R_CONTROL_ENABLE_MASK) {
152
+ /*
314
+ /*
153
+ * Ensure the tag memory is dirty on write, for migration.
315
+ * ENABLE is set; start the timer after all other changes.
154
+ * Tag memory can never contain code or display memory (vga).
316
+ * We start it even if the ENABLE bit didn't actually change,
155
+ */
317
+ * in case the timer was an expired one-shot timer that has
156
+ if (tag_access == MMU_DATA_STORE) {
318
+ * now been changed into a free-running or periodic timer.
157
+ ram_addr_t tag_ra = memory_region_get_ram_addr(mr) + xlat;
319
+ */
158
+ cpu_physical_memory_set_dirty_flag(tag_ra, DIRTY_MEMORY_MIGRATION);
320
+ ptimer_run(m->timer, !!(newctrl & R_CONTROL_ONESHOT_MASK));
321
+ }
159
+ }
322
+
160
+
323
+ m->control = newctrl;
161
+ return memory_region_get_ram_ptr(mr) + xlat;
324
+}
162
+#endif
325
+
163
}
326
+static uint64_t cmsdk_apb_dualtimer_read(void *opaque, hwaddr offset,
164
327
+ unsigned size)
165
uint64_t HELPER(irg)(CPUARMState *env, uint64_t rn, uint64_t rm)
328
+{
329
+ CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(opaque);
330
+ uint64_t r;
331
+
332
+ if (offset >= A_TIMERITCR) {
333
+ switch (offset) {
334
+ case A_TIMERITCR:
335
+ r = s->timeritcr;
336
+ break;
337
+ case A_PID4 ... A_CID3:
338
+ r = timer_id[(offset - A_PID4) / 4];
339
+ break;
340
+ default:
341
+ bad_offset:
342
+ qemu_log_mask(LOG_GUEST_ERROR,
343
+ "CMSDK APB dual-timer read: bad offset %x\n",
344
+ (int) offset);
345
+ r = 0;
346
+ break;
347
+ }
348
+ } else {
349
+ int timer = offset >> 5;
350
+ CMSDKAPBDualTimerModule *m;
351
+
352
+ if (timer >= ARRAY_SIZE(s->timermod)) {
353
+ goto bad_offset;
354
+ }
355
+
356
+ m = &s->timermod[timer];
357
+
358
+ switch (offset & 0x1F) {
359
+ case A_TIMER1LOAD:
360
+ case A_TIMER1BGLOAD:
361
+ if (m->control & R_CONTROL_MODE_MASK) {
362
+ /*
363
+ * Periodic: the ptimer limit is the LOAD register value, (or
364
+ * just the low 16 bits of it if the timer is in 16-bit mode)
365
+ */
366
+ r = ptimer_get_limit(m->timer);
367
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
368
+ r = deposit32(m->load, 0, 16, r);
369
+ }
370
+ } else {
371
+ /* Free-running: LOAD register value is just in m->load */
372
+ r = m->load;
373
+ }
374
+ break;
375
+ case A_TIMER1VALUE:
376
+ r = ptimer_get_count(m->timer);
377
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
378
+ r = deposit32(m->value, 0, 16, r);
379
+ }
380
+ break;
381
+ case A_TIMER1CONTROL:
382
+ r = m->control;
383
+ break;
384
+ case A_TIMER1RIS:
385
+ r = m->intstatus;
386
+ break;
387
+ case A_TIMER1MIS:
388
+ r = cmsdk_dualtimermod_intstatus(m);
389
+ break;
390
+ default:
391
+ goto bad_offset;
392
+ }
393
+ }
394
+
395
+ trace_cmsdk_apb_dualtimer_read(offset, r, size);
396
+ return r;
397
+}
398
+
399
+static void cmsdk_apb_dualtimer_write(void *opaque, hwaddr offset,
400
+ uint64_t value, unsigned size)
401
+{
402
+ CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(opaque);
403
+
404
+ trace_cmsdk_apb_dualtimer_write(offset, value, size);
405
+
406
+ if (offset >= A_TIMERITCR) {
407
+ switch (offset) {
408
+ case A_TIMERITCR:
409
+ s->timeritcr = value & R_TIMERITCR_VALID_MASK;
410
+ cmsdk_apb_dualtimer_update(s);
411
+ case A_TIMERITOP:
412
+ s->timeritop = value & R_TIMERITOP_VALID_MASK;
413
+ cmsdk_apb_dualtimer_update(s);
414
+ default:
415
+ bad_offset:
416
+ qemu_log_mask(LOG_GUEST_ERROR,
417
+ "CMSDK APB dual-timer write: bad offset %x\n",
418
+ (int) offset);
419
+ break;
420
+ }
421
+ } else {
422
+ int timer = offset >> 5;
423
+ CMSDKAPBDualTimerModule *m;
424
+
425
+ if (timer >= ARRAY_SIZE(s->timermod)) {
426
+ goto bad_offset;
427
+ }
428
+
429
+ m = &s->timermod[timer];
430
+
431
+ switch (offset & 0x1F) {
432
+ case A_TIMER1LOAD:
433
+ /* Set the limit, and immediately reload the count from it */
434
+ m->load = value;
435
+ m->value = value;
436
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
437
+ value &= 0xffff;
438
+ }
439
+ if (!(m->control & R_CONTROL_MODE_MASK)) {
440
+ /*
441
+ * In free-running mode this won't set the limit but will
442
+ * still change the current count value.
443
+ */
444
+ ptimer_set_count(m->timer, value);
445
+ } else {
446
+ if (!value) {
447
+ ptimer_stop(m->timer);
448
+ }
449
+ ptimer_set_limit(m->timer, value, 1);
450
+ if (value && (m->control & R_CONTROL_ENABLE_MASK)) {
451
+ /* Force possibly-expired oneshot timer to restart */
452
+ ptimer_run(m->timer, 1);
453
+ }
454
+ }
455
+ break;
456
+ case A_TIMER1BGLOAD:
457
+ /* Set the limit, but not the current count */
458
+ m->load = value;
459
+ if (!(m->control & R_CONTROL_MODE_MASK)) {
460
+ /* In free-running mode there is no limit */
461
+ break;
462
+ }
463
+ if (!(m->control & R_CONTROL_SIZE_MASK)) {
464
+ value &= 0xffff;
465
+ }
466
+ ptimer_set_limit(m->timer, value, 0);
467
+ break;
468
+ case A_TIMER1CONTROL:
469
+ cmsdk_dualtimermod_write_control(m, value);
470
+ cmsdk_apb_dualtimer_update(s);
471
+ break;
472
+ case A_TIMER1INTCLR:
473
+ m->intstatus = 0;
474
+ cmsdk_apb_dualtimer_update(s);
475
+ break;
476
+ default:
477
+ goto bad_offset;
478
+ }
479
+ }
480
+}
481
+
482
+static const MemoryRegionOps cmsdk_apb_dualtimer_ops = {
483
+ .read = cmsdk_apb_dualtimer_read,
484
+ .write = cmsdk_apb_dualtimer_write,
485
+ .endianness = DEVICE_LITTLE_ENDIAN,
486
+ /* byte/halfword accesses are just zero-padded on reads and writes */
487
+ .impl.min_access_size = 4,
488
+ .impl.max_access_size = 4,
489
+ .valid.min_access_size = 1,
490
+ .valid.max_access_size = 4,
491
+};
492
+
493
+static void cmsdk_dualtimermod_tick(void *opaque)
494
+{
495
+ CMSDKAPBDualTimerModule *m = opaque;
496
+
497
+ m->intstatus = 1;
498
+ cmsdk_apb_dualtimer_update(m->parent);
499
+}
500
+
501
+static void cmsdk_dualtimermod_reset(CMSDKAPBDualTimerModule *m)
502
+{
503
+ m->control = R_CONTROL_INTEN_MASK;
504
+ m->intstatus = 0;
505
+ m->load = 0;
506
+ m->value = 0xffffffff;
507
+ ptimer_stop(m->timer);
508
+ /*
509
+ * We start in free-running mode, with VALUE at 0xffffffff, and
510
+ * in 16-bit counter mode. This means that the ptimer count and
511
+ * limit must both be set to 0xffff, so we wrap at 16 bits.
512
+ */
513
+ ptimer_set_limit(m->timer, 0xffff, 1);
514
+ ptimer_set_freq(m->timer, m->parent->pclk_frq);
515
+}
516
+
517
+static void cmsdk_apb_dualtimer_reset(DeviceState *dev)
518
+{
519
+ CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(dev);
520
+ int i;
521
+
522
+ trace_cmsdk_apb_dualtimer_reset();
523
+
524
+ for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
525
+ cmsdk_dualtimermod_reset(&s->timermod[i]);
526
+ }
527
+ s->timeritcr = 0;
528
+ s->timeritop = 0;
529
+}
530
+
531
+static void cmsdk_apb_dualtimer_init(Object *obj)
532
+{
533
+ SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
534
+ CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(obj);
535
+ int i;
536
+
537
+ memory_region_init_io(&s->iomem, obj, &cmsdk_apb_dualtimer_ops,
538
+ s, "cmsdk-apb-dualtimer", 0x1000);
539
+ sysbus_init_mmio(sbd, &s->iomem);
540
+ sysbus_init_irq(sbd, &s->timerintc);
541
+
542
+ for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
543
+ sysbus_init_irq(sbd, &s->timermod[i].timerint);
544
+ }
545
+}
546
+
547
+static void cmsdk_apb_dualtimer_realize(DeviceState *dev, Error **errp)
548
+{
549
+ CMSDKAPBDualTimer *s = CMSDK_APB_DUALTIMER(dev);
550
+ int i;
551
+
552
+ if (s->pclk_frq == 0) {
553
+ error_setg(errp, "CMSDK APB timer: pclk-frq property must be set");
554
+ return;
555
+ }
556
+
557
+ for (i = 0; i < ARRAY_SIZE(s->timermod); i++) {
558
+ CMSDKAPBDualTimerModule *m = &s->timermod[i];
559
+ QEMUBH *bh = qemu_bh_new(cmsdk_dualtimermod_tick, m);
560
+
561
+ m->parent = s;
562
+ m->timer = ptimer_init(bh,
563
+ PTIMER_POLICY_WRAP_AFTER_ONE_PERIOD |
564
+ PTIMER_POLICY_TRIGGER_ONLY_ON_DECREMENT |
565
+ PTIMER_POLICY_NO_IMMEDIATE_RELOAD |
566
+ PTIMER_POLICY_NO_COUNTER_ROUND_DOWN);
567
+ }
568
+}
569
+
570
+static const VMStateDescription cmsdk_dualtimermod_vmstate = {
571
+ .name = "cmsdk-apb-dualtimer-module",
572
+ .version_id = 1,
573
+ .minimum_version_id = 1,
574
+ .fields = (VMStateField[]) {
575
+ VMSTATE_PTIMER(timer, CMSDKAPBDualTimerModule),
576
+ VMSTATE_UINT32(load, CMSDKAPBDualTimerModule),
577
+ VMSTATE_UINT32(value, CMSDKAPBDualTimerModule),
578
+ VMSTATE_UINT32(control, CMSDKAPBDualTimerModule),
579
+ VMSTATE_UINT32(intstatus, CMSDKAPBDualTimerModule),
580
+ VMSTATE_END_OF_LIST()
581
+ }
582
+};
583
+
584
+static const VMStateDescription cmsdk_apb_dualtimer_vmstate = {
585
+ .name = "cmsdk-apb-dualtimer",
586
+ .version_id = 1,
587
+ .minimum_version_id = 1,
588
+ .fields = (VMStateField[]) {
589
+ VMSTATE_STRUCT_ARRAY(timermod, CMSDKAPBDualTimer,
590
+ CMSDK_APB_DUALTIMER_NUM_MODULES,
591
+ 1, cmsdk_dualtimermod_vmstate,
592
+ CMSDKAPBDualTimerModule),
593
+ VMSTATE_UINT32(timeritcr, CMSDKAPBDualTimer),
594
+ VMSTATE_UINT32(timeritop, CMSDKAPBDualTimer),
595
+ VMSTATE_END_OF_LIST()
596
+ }
597
+};
598
+
599
+static Property cmsdk_apb_dualtimer_properties[] = {
600
+ DEFINE_PROP_UINT32("pclk-frq", CMSDKAPBDualTimer, pclk_frq, 0),
601
+ DEFINE_PROP_END_OF_LIST(),
602
+};
603
+
604
+static void cmsdk_apb_dualtimer_class_init(ObjectClass *klass, void *data)
605
+{
606
+ DeviceClass *dc = DEVICE_CLASS(klass);
607
+
608
+ dc->realize = cmsdk_apb_dualtimer_realize;
609
+ dc->vmsd = &cmsdk_apb_dualtimer_vmstate;
610
+ dc->reset = cmsdk_apb_dualtimer_reset;
611
+ dc->props = cmsdk_apb_dualtimer_properties;
612
+}
613
+
614
+static const TypeInfo cmsdk_apb_dualtimer_info = {
615
+ .name = TYPE_CMSDK_APB_DUALTIMER,
616
+ .parent = TYPE_SYS_BUS_DEVICE,
617
+ .instance_size = sizeof(CMSDKAPBDualTimer),
618
+ .instance_init = cmsdk_apb_dualtimer_init,
619
+ .class_init = cmsdk_apb_dualtimer_class_init,
620
+};
621
+
622
+static void cmsdk_apb_dualtimer_register_types(void)
623
+{
624
+ type_register_static(&cmsdk_apb_dualtimer_info);
625
+}
626
+
627
+type_init(cmsdk_apb_dualtimer_register_types);
628
diff --git a/MAINTAINERS b/MAINTAINERS
629
index XXXXXXX..XXXXXXX 100644
630
--- a/MAINTAINERS
631
+++ b/MAINTAINERS
632
@@ -XXX,XX +XXX,XX @@ F: hw/timer/pl031.c
633
F: include/hw/arm/primecell.h
634
F: hw/timer/cmsdk-apb-timer.c
635
F: include/hw/timer/cmsdk-apb-timer.h
636
+F: hw/timer/cmsdk-apb-dualtimer.c
637
+F: include/hw/timer/cmsdk-apb-dualtimer.h
638
F: hw/char/cmsdk-apb-uart.c
639
F: include/hw/char/cmsdk-apb-uart.h
640
F: hw/watchdog/cmsdk-apb-watchdog.c
641
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
642
index XXXXXXX..XXXXXXX 100644
643
--- a/default-configs/arm-softmmu.mak
644
+++ b/default-configs/arm-softmmu.mak
645
@@ -XXX,XX +XXX,XX @@ CONFIG_STM32F2XX_SPI=y
646
CONFIG_STM32F205_SOC=y
647
648
CONFIG_CMSDK_APB_TIMER=y
649
+CONFIG_CMSDK_APB_DUALTIMER=y
650
CONFIG_CMSDK_APB_UART=y
651
CONFIG_CMSDK_APB_WATCHDOG=y
652
653
diff --git a/hw/timer/trace-events b/hw/timer/trace-events
654
index XXXXXXX..XXXXXXX 100644
655
--- a/hw/timer/trace-events
656
+++ b/hw/timer/trace-events
657
@@ -XXX,XX +XXX,XX @@ cmsdk_apb_timer_read(uint64_t offset, uint64_t data, unsigned size) "CMSDK APB t
658
cmsdk_apb_timer_write(uint64_t offset, uint64_t data, unsigned size) "CMSDK APB timer write: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
659
cmsdk_apb_timer_reset(void) "CMSDK APB timer: reset"
660
661
+# hw/timer/cmsdk_apb_dualtimer.c
662
+cmsdk_apb_dualtimer_read(uint64_t offset, uint64_t data, unsigned size) "CMSDK APB dualtimer read: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
663
+cmsdk_apb_dualtimer_write(uint64_t offset, uint64_t data, unsigned size) "CMSDK APB dualtimer write: offset 0x%" PRIx64 " data 0x%" PRIx64 " size %u"
664
+cmsdk_apb_dualtimer_reset(void) "CMSDK APB dualtimer: reset"
665
+
666
# hw/timer/xlnx-zynqmp-rtc.c
667
xlnx_zynqmp_rtc_gettime(int year, int month, int day, int hour, int min, int sec) "Get time from host: %d-%d-%d %2d:%02d:%02d"
668
--
166
--
669
2.18.0
167
2.20.1
670
168
671
169
diff view generated by jsdifflib
1
The MPS2 FPGA images for the Cortex-M3 (mps2-an385 and mps2-511)
1
From: Richard Henderson <richard.henderson@linaro.org>
2
both include a CMSDK dual-timer module. Wire this up.
3
2
3
We now implement all of the components of MTE, without actually
4
supporting any tagged memory. All MTE instructions will work,
5
trivially, so we can enable support.
6
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
9
Message-id: 20200626033144.790098-46-richard.henderson@linaro.org
4
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20180820141116.9118-6-peter.maydell@linaro.org
8
---
11
---
9
hw/arm/mps2.c | 11 +++++++++++
12
target/arm/cpu64.c | 5 +++++
10
1 file changed, 11 insertions(+)
13
1 file changed, 5 insertions(+)
11
14
12
diff --git a/hw/arm/mps2.c b/hw/arm/mps2.c
15
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
13
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/arm/mps2.c
17
--- a/target/arm/cpu64.c
15
+++ b/hw/arm/mps2.c
18
+++ b/target/arm/cpu64.c
16
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
17
#include "hw/misc/unimp.h"
20
18
#include "hw/char/cmsdk-apb-uart.h"
21
t = cpu->isar.id_aa64pfr1;
19
#include "hw/timer/cmsdk-apb-timer.h"
22
t = FIELD_DP64(t, ID_AA64PFR1, BT, 1);
20
+#include "hw/timer/cmsdk-apb-dualtimer.h"
23
+ /*
21
#include "hw/misc/mps2-scc.h"
24
+ * Begin with full support for MTE; will be downgraded to MTE=1
22
#include "hw/devices.h"
25
+ * during realize if the board provides no tag memory.
23
#include "net/net.h"
26
+ */
24
@@ -XXX,XX +XXX,XX @@ typedef struct {
27
+ t = FIELD_DP64(t, ID_AA64PFR1, MTE, 2);
25
MemoryRegion blockram_m3;
28
cpu->isar.id_aa64pfr1 = t;
26
MemoryRegion sram;
29
27
MPS2SCC scc;
30
t = cpu->isar.id_aa64mmfr1;
28
+ CMSDKAPBDualTimer dualtimer;
29
} MPS2MachineState;
30
31
#define TYPE_MPS2_MACHINE "mps2"
32
@@ -XXX,XX +XXX,XX @@ static void mps2_common_init(MachineState *machine)
33
cmsdk_apb_timer_create(0x40000000, qdev_get_gpio_in(armv7m, 8), SYSCLK_FRQ);
34
cmsdk_apb_timer_create(0x40001000, qdev_get_gpio_in(armv7m, 9), SYSCLK_FRQ);
35
36
+ sysbus_init_child_obj(OBJECT(mms), "dualtimer", &mms->dualtimer,
37
+ sizeof(mms->dualtimer), TYPE_CMSDK_APB_DUALTIMER);
38
+ qdev_prop_set_uint32(DEVICE(&mms->dualtimer), "pclk-frq", SYSCLK_FRQ);
39
+ object_property_set_bool(OBJECT(&mms->dualtimer), true, "realized",
40
+ &error_fatal);
41
+ sysbus_connect_irq(SYS_BUS_DEVICE(&mms->dualtimer), 0,
42
+ qdev_get_gpio_in(armv7m, 10));
43
+ sysbus_mmio_map(SYS_BUS_DEVICE(&mms->dualtimer), 0, 0x40002000);
44
+
45
object_initialize(&mms->scc, sizeof(mms->scc), TYPE_MPS2_SCC);
46
sccdev = DEVICE(&mms->scc);
47
qdev_set_parent_bus(sccdev, sysbus_get_default());
48
--
31
--
49
2.18.0
32
2.20.1
50
33
51
34
diff view generated by jsdifflib