1
target-arm queue of various easier things that had piled
1
First arm pullreq of the 5.1 cycle; mostly bugfixes and some
2
up while I was on holiday.
2
cleanup patches. The new clock modelling framework is the big
3
thing here.
3
4
4
thanks
5
-- PMM
5
-- PMM
6
6
7
The following changes since commit 506e4a00de01e0b29fa83db5cbbc3d154253b4ea:
7
The following changes since commit 648db19685b7030aa558a4ddbd3a8e53d8c9a062:
8
8
9
Merge remote-tracking branch 'remotes/dgibson/tags/ppc-for-3.1-20180925' into staging (2018-09-25 13:30:45 +0100)
9
Merge remote-tracking branch 'remotes/armbru/tags/pull-misc-2020-04-29' into staging (2020-04-29 15:07:33 +0100)
10
10
11
are available in the Git repository at:
11
are available in the Git repository at:
12
12
13
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20180925
13
https://git.linaro.org/people/pmaydell/qemu-arm.git tags/pull-target-arm-20200430
14
14
15
for you to fetch changes up to 4a87106b160a3e72152443065fb92f8a1313c23d:
15
for you to fetch changes up to 1267437e593e85498f9105b3bdab796630d2e83f:
16
16
17
target/arm: Start AArch32 CPUs with EL2 but not EL3 in Hyp mode (2018-09-25 14:14:07 +0100)
17
hw/arm: xlnx-zcu102: Disable unsupported FDT firmware nodes (2020-04-30 11:52:29 +0100)
18
18
19
----------------------------------------------------------------
19
----------------------------------------------------------------
20
target-arm queue:
20
target-arm queue:
21
* target/arm: Fix cpu_get_tb_cpu_state() for non-SVE CPUs
21
* xlnx-zdma: Fix endianness handling of descriptor loading
22
* hw/arm/exynos4210: fix Exynos4210 UART support
22
* nrf51: Fix last GPIO CNF address
23
* hw/arm/virt-acpi-build: Add a check for memory-less NUMA nodes
23
* gicv3: Use gicr_typer in arm_gicv3_icc_reset
24
* arm: Add BBC micro:bit machine
24
* msf2: Add EMAC block to SmartFusion2 SoC
25
* aspeed/i2c: Fix interrupt handling bugs
25
* New clock modelling framework
26
* hw/arm/smmu-common: Fix the name of the iommu memory regions
26
* hw/arm: versal: Setup the ADMA with 128bit bus-width
27
* hw/arm/smmuv3: fix eventq recording and IRQ triggerring
27
* Cadence: gem: fix wraparound in 64bit descriptors
28
* hw/intc/arm_gic: Document QEMU interface
28
* cadence_gem: clear RX control descriptor
29
* hw/intc/arm_gic: Drop GIC_BASE_IRQ macro
29
* target/arm: Vectorize integer comparison vs zero
30
* hw/net/pcnet-pci: Convert away from old_mmio accessors
30
* hw/arm/virt: dt: add kaslr-seed property
31
* hw/timer/cmsdk-apb-dualtimer: Add missing 'break' statements
31
* hw/arm: xlnx-zcu102: Disable unsupported FDT firmware nodes
32
* aspeed/timer: fix compile breakage with clang 3.4.2
33
* hw/arm/aspeed: change the FMC flash model of the AST2500 evb
34
* hw/arm/aspeed: Minor code cleanups
35
* target/arm: Start AArch32 CPUs with EL2 but not EL3 in Hyp mode
36
32
37
----------------------------------------------------------------
33
----------------------------------------------------------------
38
Bartlomiej Zolnierkiewicz (1):
34
Cameron Esfahani (1):
39
hw/arm/exynos4210: fix Exynos4210 UART support
35
nrf51: Fix last GPIO CNF address
40
36
41
Cédric Le Goater (5):
37
Damien Hedde (7):
42
aspeed/i2c: interrupts should be cleared by software only
38
hw/core/clock-vmstate: define a vmstate entry for clock state
43
aspeed/timer: fix compile breakage with clang 3.4.2
39
qdev: add clock input&output support to devices.
44
hw/arm/aspeed: change the FMC flash model of the AST2500 evb
40
qdev-clock: introduce an init array to ease the device construction
45
hw/arm/aspeed: Add an Aspeed machine class
41
hw/misc/zynq_slcr: add clock generation for uarts
46
aspeed/smc: fix some alignment issues
42
hw/char/cadence_uart: add clock support
43
hw/arm/xilinx_zynq: connect uart clocks to slcr
44
qdev-monitor: print the device's clock with info qtree
47
45
48
Eric Auger (2):
46
Edgar E. Iglesias (7):
49
hw/arm/smmu-common: Fix the name of the iommu memory regions
47
dma/xlnx-zdma: Fix descriptor loading (MEM) wrt endianness
50
hw/arm/smmuv3: fix eventq recording and IRQ triggerring
48
dma/xlnx-zdma: Fix descriptor loading (REG) wrt endianness
49
hw/arm: versal: Setup the ADMA with 128bit bus-width
50
device_tree: Allow name wildcards in qemu_fdt_node_path()
51
device_tree: Constify compat in qemu_fdt_node_path()
52
hw/arm: xlnx-zcu102: Move arm_boot_info into XlnxZCU102
53
hw/arm: xlnx-zcu102: Disable unsupported FDT firmware nodes
51
54
52
Guenter Roeck (2):
55
Jerome Forissier (2):
53
aspeed/i2c: Handle receive command in separate function
56
hw/arm/virt: dt: move creation of /secure-chosen to create_fdt()
54
aspeed/i2c: Fix receive done interrupt handling
57
hw/arm/virt: dt: add kaslr-seed property
55
58
56
Joel Stanley (3):
59
Keqian Zhu (2):
57
MAINTAINERS: Add NRF51 entry
60
bugfix: Use gicr_typer in arm_gicv3_icc_reset
58
arm: Add Nordic Semiconductor nRF51 SoC
61
Typo: Correct the name of CPU hotplug memory region
59
arm: Add BBC micro:bit machine
60
62
61
Peter Maydell (6):
63
Peter Maydell (2):
62
hw/intc/arm_gic: Document QEMU interface
64
hw/core/clock: introduce clock object
63
hw/intc/arm_gic: Drop GIC_BASE_IRQ macro
65
docs/clocks: add device's clock documentation
64
hw/net/pcnet-pci: Convert away from old_mmio accessors
66
65
hw/net/pcnet-pci: Unify pcnet_ioport_read/write and pcnet_mmio_read/write
67
Philippe Mathieu-Daudé (3):
66
hw/timer/cmsdk-apb-dualtimer: Add missing 'break' statements
68
target/arm: Restrict the Address Translate write operation to TCG accel
67
target/arm: Start AArch32 CPUs with EL2 but not EL3 in Hyp mode
69
target/arm/cpu: Use ARRAY_SIZE() to iterate over ARMCPUInfo[]
70
target/arm/cpu: Update coding style to make checkpatch.pl happy
71
72
Ramon Fried (2):
73
Cadence: gem: fix wraparound in 64bit descriptors
74
net: cadence_gem: clear RX control descriptor
68
75
69
Richard Henderson (1):
76
Richard Henderson (1):
70
target/arm: Fix cpu_get_tb_cpu_state() for non-SVE CPUs
77
target/arm: Vectorize integer comparison vs zero
71
78
72
Shannon Zhao (1):
79
Subbaraya Sundeep (3):
73
hw/arm/virt-acpi-build: Add a check for memory-less NUMA nodes
80
hw/net: Add Smartfusion2 emac block
81
msf2: Add EMAC block to SmartFusion2 SoC
82
tests/boot_linux_console: Add ethernet test to SmartFusion2
74
83
75
hw/arm/Makefile.objs | 1 +
84
Thomas Huth (1):
76
hw/arm/smmuv3-internal.h | 26 ++---
85
target/arm: Make cpu_register() available for other files
77
hw/intc/gic_internal.h | 2 -
78
include/hw/arm/aspeed.h | 46 +++++++++
79
include/hw/arm/nrf51_soc.h | 41 ++++++++
80
include/hw/intc/arm_gic.h | 43 ++++++++
81
include/hw/timer/aspeed_timer.h | 3 +-
82
hw/arm/aspeed.c | 212 +++++++++++++---------------------------
83
hw/arm/exynos4210.c | 8 +-
84
hw/arm/microbit.c | 67 +++++++++++++
85
hw/arm/nrf51_soc.c | 133 +++++++++++++++++++++++++
86
hw/arm/smmu-common.c | 6 +-
87
hw/arm/smmuv3.c | 2 +-
88
hw/arm/virt-acpi-build.c | 10 +-
89
hw/i2c/aspeed_i2c.c | 63 ++++++++----
90
hw/intc/arm_gic.c | 31 +++---
91
hw/intc/arm_gic_common.c | 1 -
92
hw/net/pcnet-pci.c | 98 ++-----------------
93
hw/ssi/aspeed_smc.c | 8 +-
94
hw/timer/aspeed_timer.c | 1 -
95
hw/timer/cmsdk-apb-dualtimer.c | 2 +
96
target/arm/cpu.c | 14 ++-
97
target/arm/helper.c | 45 +++++----
98
MAINTAINERS | 8 ++
99
default-configs/arm-softmmu.mak | 1 +
100
hw/net/trace-events | 6 --
101
26 files changed, 542 insertions(+), 336 deletions(-)
102
create mode 100644 include/hw/arm/aspeed.h
103
create mode 100644 include/hw/arm/nrf51_soc.h
104
create mode 100644 hw/arm/microbit.c
105
create mode 100644 hw/arm/nrf51_soc.c
106
86
87
hw/core/Makefile.objs | 2 +
88
hw/net/Makefile.objs | 1 +
89
tests/Makefile.include | 1 +
90
include/hw/arm/msf2-soc.h | 2 +
91
include/hw/char/cadence_uart.h | 1 +
92
include/hw/clock.h | 225 +++++++++++++
93
include/hw/gpio/nrf51_gpio.h | 2 +-
94
include/hw/net/msf2-emac.h | 53 +++
95
include/hw/qdev-clock.h | 159 +++++++++
96
include/hw/qdev-core.h | 12 +
97
include/sysemu/device_tree.h | 5 +-
98
target/arm/cpu-qom.h | 9 +-
99
target/arm/helper.h | 27 +-
100
target/arm/translate.h | 5 +
101
device_tree.c | 4 +-
102
hw/acpi/cpu.c | 2 +-
103
hw/arm/msf2-soc.c | 26 +-
104
hw/arm/virt.c | 20 +-
105
hw/arm/xilinx_zynq.c | 57 +++-
106
hw/arm/xlnx-versal.c | 2 +
107
hw/arm/xlnx-zcu102.c | 39 ++-
108
hw/char/cadence_uart.c | 73 +++-
109
hw/core/clock-vmstate.c | 25 ++
110
hw/core/clock.c | 130 ++++++++
111
hw/core/qdev-clock.c | 185 +++++++++++
112
hw/core/qdev.c | 12 +
113
hw/dma/xlnx-zdma.c | 25 +-
114
hw/intc/arm_gicv3_kvm.c | 4 +-
115
hw/misc/zynq_slcr.c | 172 +++++++++-
116
hw/net/cadence_gem.c | 16 +-
117
hw/net/msf2-emac.c | 589 +++++++++++++++++++++++++++++++++
118
qdev-monitor.c | 9 +
119
target/arm/cpu.c | 25 +-
120
target/arm/cpu64.c | 16 +-
121
target/arm/helper.c | 17 +
122
target/arm/neon_helper.c | 24 --
123
target/arm/translate-a64.c | 64 +---
124
target/arm/translate.c | 256 ++++++++++++--
125
target/arm/vec_helper.c | 25 ++
126
MAINTAINERS | 2 +
127
docs/devel/clocks.rst | 391 ++++++++++++++++++++++
128
docs/devel/index.rst | 1 +
129
hw/char/trace-events | 3 +
130
hw/core/trace-events | 7 +
131
tests/acceptance/boot_linux_console.py | 15 +-
132
45 files changed, 2538 insertions(+), 202 deletions(-)
133
create mode 100644 include/hw/clock.h
134
create mode 100644 include/hw/net/msf2-emac.h
135
create mode 100644 include/hw/qdev-clock.h
136
create mode 100644 hw/core/clock-vmstate.c
137
create mode 100644 hw/core/clock.c
138
create mode 100644 hw/core/qdev-clock.c
139
create mode 100644 hw/net/msf2-emac.c
140
create mode 100644 docs/devel/clocks.rst
141
diff view generated by jsdifflib
1
From: Guenter Roeck <linux@roeck-us.net>
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
2
3
The AST2500 datasheet says:
3
Fix descriptor loading from memory wrt host endianness.
4
4
5
I2CD10 Interrupt Status Register
5
Reported-by: Peter Maydell <peter.maydell@linaro.org>
6
bit 2 Receive Done Interrupt status
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
S/W needs to clear this status bit to allow next data receiving
7
Reviewed-by: Francisco Iglesias <frasse.iglesias@gmail.com>
8
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
The Rx interrupt done interrupt status bit needs to be cleared
9
Message-id: 20200404122718.25111-2-edgar.iglesias@gmail.com
10
explicitly before the next byte can be received, and must therefore
11
not be auto-cleared. Also, receiving the next byte must be delayed
12
until the bit has been cleared.
13
14
Signed-off-by: Guenter Roeck <linux@roeck-us.net>
15
Signed-off-by: Cédric Le Goater <clg@kaod.org>
16
Message-id: 20180914063506.20815-4-clg@kaod.org
17
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
18
---
11
---
19
hw/i2c/aspeed_i2c.c | 10 +++++++++-
12
hw/dma/xlnx-zdma.c | 11 +++++++----
20
1 file changed, 9 insertions(+), 1 deletion(-)
13
1 file changed, 7 insertions(+), 4 deletions(-)
21
14
22
diff --git a/hw/i2c/aspeed_i2c.c b/hw/i2c/aspeed_i2c.c
15
diff --git a/hw/dma/xlnx-zdma.c b/hw/dma/xlnx-zdma.c
23
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
24
--- a/hw/i2c/aspeed_i2c.c
17
--- a/hw/dma/xlnx-zdma.c
25
+++ b/hw/i2c/aspeed_i2c.c
18
+++ b/hw/dma/xlnx-zdma.c
26
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_handle_cmd(AspeedI2CBus *bus, uint64_t value)
19
@@ -XXX,XX +XXX,XX @@ static void zdma_put_regaddr64(XlnxZDMA *s, unsigned int basereg, uint64_t addr)
27
aspeed_i2c_set_state(bus, I2CD_MACTIVE);
20
s->regs[basereg + 1] = addr >> 32;
21
}
22
23
-static bool zdma_load_descriptor(XlnxZDMA *s, uint64_t addr, void *buf)
24
+static bool zdma_load_descriptor(XlnxZDMA *s, uint64_t addr,
25
+ XlnxZDMADescr *descr)
26
{
27
/* ZDMA descriptors must be aligned to their own size. */
28
if (addr % sizeof(XlnxZDMADescr)) {
29
qemu_log_mask(LOG_GUEST_ERROR,
30
"zdma: unaligned descriptor at %" PRIx64,
31
addr);
32
- memset(buf, 0x0, sizeof(XlnxZDMADescr));
33
+ memset(descr, 0x0, sizeof(XlnxZDMADescr));
34
s->error = true;
35
return false;
28
}
36
}
29
37
30
- if (bus->cmd & (I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST)) {
38
- address_space_read(s->dma_as, addr, s->attr, buf, sizeof(XlnxZDMADescr));
31
+ if ((bus->cmd & (I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST)) &&
39
+ descr->addr = address_space_ldq_le(s->dma_as, addr, s->attr, NULL);
32
+ !(bus->intr_status & I2CD_INTR_RX_DONE)) {
40
+ descr->size = address_space_ldl_le(s->dma_as, addr + 8, s->attr, NULL);
33
aspeed_i2c_handle_rx_cmd(bus);
41
+ descr->attr = address_space_ldl_le(s->dma_as, addr + 12, s->attr, NULL);
42
return true;
43
}
44
45
@@ -XXX,XX +XXX,XX @@ static void zdma_update_descr_addr(XlnxZDMA *s, bool type,
46
} else {
47
addr = zdma_get_regaddr64(s, basereg);
48
addr += sizeof(s->dsc_dst);
49
- address_space_read(s->dma_as, addr, s->attr, (void *) &next, 8);
50
+ next = address_space_ldq_le(s->dma_as, addr, s->attr, NULL);
34
}
51
}
35
52
36
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_write(void *opaque, hwaddr offset,
53
zdma_put_regaddr64(s, basereg, next);
37
uint64_t value, unsigned size)
38
{
39
AspeedI2CBus *bus = opaque;
40
+ bool handle_rx;
41
42
switch (offset) {
43
case I2CD_FUN_CTRL_REG:
44
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_write(void *opaque, hwaddr offset,
45
bus->intr_ctrl = value & 0x7FFF;
46
break;
47
case I2CD_INTR_STS_REG:
48
+ handle_rx = (bus->intr_status & I2CD_INTR_RX_DONE) &&
49
+ (value & I2CD_INTR_RX_DONE);
50
bus->intr_status &= ~(value & 0x7FFF);
51
if (!bus->intr_status) {
52
bus->controller->intr_status &= ~(1 << bus->id);
53
qemu_irq_lower(bus->controller->irq);
54
}
55
+ if (handle_rx && (bus->cmd & (I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST))) {
56
+ aspeed_i2c_handle_rx_cmd(bus);
57
+ aspeed_i2c_bus_raise_interrupt(bus);
58
+ }
59
break;
60
case I2CD_DEV_ADDR_REG:
61
qemu_log_mask(LOG_UNIMP, "%s: slave mode not implemented\n",
62
--
54
--
63
2.19.0
55
2.20.1
64
56
65
57
diff view generated by jsdifflib
1
From: Guenter Roeck <linux@roeck-us.net>
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
2
3
Receive command handling may have to be deferred if a previous receive
3
Fix descriptor loading from registers wrt host endianness.
4
done interrupt was not yet acknowledged. Move receive command handling
5
into a separate function to prepare for the necessary changes.
6
4
7
Signed-off-by: Guenter Roeck <linux@roeck-us.net>
5
Reported-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Cédric Le Goater <clg@kaod.org>
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Message-id: 20180914063506.20815-3-clg@kaod.org
7
Reviewed-by: Francisco Iglesias <frasse.iglesias@gmail.com>
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Message-id: 20200404122718.25111-3-edgar.iglesias@gmail.com
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
10
---
13
hw/i2c/aspeed_i2c.c | 37 +++++++++++++++++++++----------------
11
hw/dma/xlnx-zdma.c | 14 ++++++++++----
14
1 file changed, 21 insertions(+), 16 deletions(-)
12
1 file changed, 10 insertions(+), 4 deletions(-)
15
13
16
diff --git a/hw/i2c/aspeed_i2c.c b/hw/i2c/aspeed_i2c.c
14
diff --git a/hw/dma/xlnx-zdma.c b/hw/dma/xlnx-zdma.c
17
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/i2c/aspeed_i2c.c
16
--- a/hw/dma/xlnx-zdma.c
19
+++ b/hw/i2c/aspeed_i2c.c
17
+++ b/hw/dma/xlnx-zdma.c
20
@@ -XXX,XX +XXX,XX @@ static uint8_t aspeed_i2c_get_state(AspeedI2CBus *bus)
18
@@ -XXX,XX +XXX,XX @@ static void zdma_put_regaddr64(XlnxZDMA *s, unsigned int basereg, uint64_t addr)
21
return (bus->cmd >> I2CD_TX_STATE_SHIFT) & I2CD_TX_STATE_MASK;
19
s->regs[basereg + 1] = addr >> 32;
22
}
20
}
23
21
24
+static void aspeed_i2c_handle_rx_cmd(AspeedI2CBus *bus)
22
+static void zdma_load_descriptor_reg(XlnxZDMA *s, unsigned int reg,
23
+ XlnxZDMADescr *descr)
25
+{
24
+{
26
+ int ret;
25
+ descr->addr = zdma_get_regaddr64(s, reg);
27
+
26
+ descr->size = s->regs[reg + 2];
28
+ aspeed_i2c_set_state(bus, I2CD_MRXD);
27
+ descr->attr = s->regs[reg + 3];
29
+ ret = i2c_recv(bus->bus);
30
+ if (ret < 0) {
31
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: read failed\n", __func__);
32
+ ret = 0xff;
33
+ } else {
34
+ bus->intr_status |= I2CD_INTR_RX_DONE;
35
+ }
36
+ bus->buf = (ret & I2CD_BYTE_BUF_RX_MASK) << I2CD_BYTE_BUF_RX_SHIFT;
37
+ if (bus->cmd & I2CD_M_S_RX_CMD_LAST) {
38
+ i2c_nack(bus->bus);
39
+ }
40
+ bus->cmd &= ~(I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST);
41
+ aspeed_i2c_set_state(bus, I2CD_MACTIVE);
42
+}
28
+}
43
+
29
+
44
/*
30
static bool zdma_load_descriptor(XlnxZDMA *s, uint64_t addr,
45
* The state machine needs some refinement. It is only used to track
31
XlnxZDMADescr *descr)
46
* invalid STOP commands for the moment.
32
{
47
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_handle_cmd(AspeedI2CBus *bus, uint64_t value)
33
@@ -XXX,XX +XXX,XX @@ static void zdma_load_src_descriptor(XlnxZDMA *s)
34
unsigned int ptype = ARRAY_FIELD_EX32(s->regs, ZDMA_CH_CTRL0, POINT_TYPE);
35
36
if (ptype == PT_REG) {
37
- memcpy(&s->dsc_src, &s->regs[R_ZDMA_CH_SRC_DSCR_WORD0],
38
- sizeof(s->dsc_src));
39
+ zdma_load_descriptor_reg(s, R_ZDMA_CH_SRC_DSCR_WORD0, &s->dsc_src);
40
return;
48
}
41
}
49
42
50
if (bus->cmd & (I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST)) {
43
@@ -XXX,XX +XXX,XX @@ static void zdma_load_dst_descriptor(XlnxZDMA *s)
51
- int ret;
44
bool dst_type;
52
-
45
53
- aspeed_i2c_set_state(bus, I2CD_MRXD);
46
if (ptype == PT_REG) {
54
- ret = i2c_recv(bus->bus);
47
- memcpy(&s->dsc_dst, &s->regs[R_ZDMA_CH_DST_DSCR_WORD0],
55
- if (ret < 0) {
48
- sizeof(s->dsc_dst));
56
- qemu_log_mask(LOG_GUEST_ERROR, "%s: read failed\n", __func__);
49
+ zdma_load_descriptor_reg(s, R_ZDMA_CH_DST_DSCR_WORD0, &s->dsc_dst);
57
- ret = 0xff;
50
return;
58
- } else {
59
- bus->intr_status |= I2CD_INTR_RX_DONE;
60
- }
61
- bus->buf = (ret & I2CD_BYTE_BUF_RX_MASK) << I2CD_BYTE_BUF_RX_SHIFT;
62
- if (bus->cmd & I2CD_M_S_RX_CMD_LAST) {
63
- i2c_nack(bus->bus);
64
- }
65
- bus->cmd &= ~(I2CD_M_RX_CMD | I2CD_M_S_RX_CMD_LAST);
66
- aspeed_i2c_set_state(bus, I2CD_MACTIVE);
67
+ aspeed_i2c_handle_rx_cmd(bus);
68
}
51
}
69
52
70
if (bus->cmd & I2CD_M_STOP_CMD) {
71
--
53
--
72
2.19.0
54
2.20.1
73
55
74
56
diff view generated by jsdifflib
New patch
1
From: Cameron Esfahani <dirty@apple.com>
1
2
3
NRF51_GPIO_REG_CNF_END doesn't actually refer to the start of the last
4
valid CNF register: it's referring to the last byte of the last valid
5
CNF register.
6
7
This hasn't been a problem up to now, as current implementation in
8
memory.c turns an unaligned 4-byte read from 0x77f to a single byte read
9
and the qtest only looks at the least-significant byte of the register.
10
11
But when running with patches which fix unaligned accesses in memory.c,
12
the qtest breaks.
13
14
Considering NRF51 doesn't support unaligned accesses, the simplest fix
15
is to actually set NRF51_GPIO_REG_CNF_END to the start of the last valid
16
CNF register: 0x77c.
17
18
Now, qtests work with or without the unaligned access patches.
19
20
Reviewed-by: Cédric Le Goater <clg@kaod.org>
21
Tested-by: Cédric Le Goater <clg@kaod.org>
22
Reviewed-by: Joel Stanley <joel@jms.id.au>
23
Signed-off-by: Cameron Esfahani <dirty@apple.com>
24
Message-id: 51b427f06838622da783d38ba56e3630d6d85c60.1586925392.git.dirty@apple.com
25
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
26
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
27
---
28
include/hw/gpio/nrf51_gpio.h | 2 +-
29
1 file changed, 1 insertion(+), 1 deletion(-)
30
31
diff --git a/include/hw/gpio/nrf51_gpio.h b/include/hw/gpio/nrf51_gpio.h
32
index XXXXXXX..XXXXXXX 100644
33
--- a/include/hw/gpio/nrf51_gpio.h
34
+++ b/include/hw/gpio/nrf51_gpio.h
35
@@ -XXX,XX +XXX,XX @@
36
#define NRF51_GPIO_REG_DIRSET 0x518
37
#define NRF51_GPIO_REG_DIRCLR 0x51C
38
#define NRF51_GPIO_REG_CNF_START 0x700
39
-#define NRF51_GPIO_REG_CNF_END 0x77F
40
+#define NRF51_GPIO_REG_CNF_END 0x77C
41
42
#define NRF51_GPIO_PULLDOWN 1
43
#define NRF51_GPIO_PULLUP 3
44
--
45
2.20.1
46
47
diff view generated by jsdifflib
New patch
1
From: Keqian Zhu <zhukeqian1@huawei.com>
1
2
3
The KVM_VGIC_ATTR macro expect the second parameter as gicr_typer,
4
of which high 32bit is constructed by mp_affinity. For most case,
5
the high 32bit of mp_affinity is zero, so it will always access the
6
ICC_CTLR_EL1 of CPU0.
7
8
Signed-off-by: Keqian Zhu <zhukeqian1@huawei.com>
9
Message-id: 20200413091552.62748-2-zhukeqian1@huawei.com
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
13
hw/intc/arm_gicv3_kvm.c | 4 +---
14
1 file changed, 1 insertion(+), 3 deletions(-)
15
16
diff --git a/hw/intc/arm_gicv3_kvm.c b/hw/intc/arm_gicv3_kvm.c
17
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/intc/arm_gicv3_kvm.c
19
+++ b/hw/intc/arm_gicv3_kvm.c
20
@@ -XXX,XX +XXX,XX @@ static void kvm_arm_gicv3_get(GICv3State *s)
21
22
static void arm_gicv3_icc_reset(CPUARMState *env, const ARMCPRegInfo *ri)
23
{
24
- ARMCPU *cpu;
25
GICv3State *s;
26
GICv3CPUState *c;
27
28
c = (GICv3CPUState *)env->gicv3state;
29
s = c->gic;
30
- cpu = ARM_CPU(c->cpu);
31
32
c->icc_pmr_el1 = 0;
33
c->icc_bpr[GICV3_G0] = GIC_MIN_BPR;
34
@@ -XXX,XX +XXX,XX @@ static void arm_gicv3_icc_reset(CPUARMState *env, const ARMCPRegInfo *ri)
35
36
/* Initialize to actual HW supported configuration */
37
kvm_device_access(s->dev_fd, KVM_DEV_ARM_VGIC_GRP_CPU_SYSREGS,
38
- KVM_VGIC_ATTR(ICC_CTLR_EL1, cpu->mp_affinity),
39
+ KVM_VGIC_ATTR(ICC_CTLR_EL1, c->gicr_typer),
40
&c->icc_ctlr_el1[GICV3_NS], false, &error_abort);
41
42
c->icc_ctlr_el1[GICV3_S] = c->icc_ctlr_el1[GICV3_NS];
43
--
44
2.20.1
45
46
diff view generated by jsdifflib
1
From: Cédric Le Goater <clg@kaod.org>
1
From: Keqian Zhu <zhukeqian1@huawei.com>
2
2
3
The AST2500 evb is shipped with a W25Q256 which has a non volatile bit
3
Replace "acpi-mem-hotplug" with "acpi-cpu-hotplug"
4
to make the chip operate in 4 Byte address mode at power up. This
5
should be an interesting feature to model as it will exercise a bit
6
more the SMC controllers and MMIO execution at boot time.
7
4
8
Signed-off-by: Cédric Le Goater <clg@kaod.org>
5
Signed-off-by: Keqian Zhu <zhukeqian1@huawei.com>
9
Message-id: 20180921161939.822-3-clg@kaod.org
6
Message-id: 20200413091552.62748-4-zhukeqian1@huawei.com
10
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
9
---
13
hw/arm/aspeed.c | 2 +-
10
hw/acpi/cpu.c | 2 +-
14
1 file changed, 1 insertion(+), 1 deletion(-)
11
1 file changed, 1 insertion(+), 1 deletion(-)
15
12
16
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
13
diff --git a/hw/acpi/cpu.c b/hw/acpi/cpu.c
17
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/arm/aspeed.c
15
--- a/hw/acpi/cpu.c
19
+++ b/hw/arm/aspeed.c
16
+++ b/hw/acpi/cpu.c
20
@@ -XXX,XX +XXX,XX @@ static const AspeedBoardConfig aspeed_boards[] = {
17
@@ -XXX,XX +XXX,XX @@ void cpu_hotplug_hw_init(MemoryRegion *as, Object *owner,
21
[AST2500_EVB] = {
18
state->devs[i].arch_id = id_list->cpus[i].arch_id;
22
.soc_name = "ast2500-a1",
19
}
23
.hw_strap1 = AST2500_EVB_HW_STRAP1,
20
memory_region_init_io(&state->ctrl_reg, owner, &cpu_hotplug_ops, state,
24
- .fmc_model = "n25q256a",
21
- "acpi-mem-hotplug", ACPI_CPU_HOTPLUG_REG_LEN);
25
+ .fmc_model = "w25q256",
22
+ "acpi-cpu-hotplug", ACPI_CPU_HOTPLUG_REG_LEN);
26
.spi_model = "mx25l25635e",
23
memory_region_add_subregion(as, base_addr, &state->ctrl_reg);
27
.num_cs = 1,
24
}
28
.i2c_init = ast2500_evb_i2c_init,
25
29
--
26
--
30
2.19.0
27
2.20.1
31
28
32
29
diff view generated by jsdifflib
1
From: Joel Stanley <joel@jms.id.au>
1
From: Subbaraya Sundeep <sundeep.lkml@gmail.com>
2
2
3
The nRF51 is a Cortex-M0 microcontroller with an on-board radio module,
3
Modelled Ethernet MAC of Smartfusion2 SoC.
4
plus other common ARM SoC peripherals.
4
Micrel KSZ8051 PHY is present on Emcraft's
5
SOM kit hence same PHY is emulated.
5
6
6
http://infocenter.nordicsemi.com/pdf/nRF51_RM_v3.0.pdf
7
Signed-off-by: Subbaraya Sundeep <sundeep.lkml@gmail.com>
7
8
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
This defines a basic model of the CPU and memory, with no peripherals
9
Tested-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
implemented at this stage.
10
Message-id: 1587048891-30493-2-git-send-email-sundeep.lkml@gmail.com
10
11
Signed-off-by: Joel Stanley <joel@jms.id.au>
12
Message-id: 20180831220920.27113-3-joel@jms.id.au
13
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
14
[PMM: wrapped a few long lines]
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
16
---
12
---
17
hw/arm/Makefile.objs | 1 +
13
hw/net/Makefile.objs | 1 +
18
include/hw/arm/nrf51_soc.h | 41 ++++++++++
14
include/hw/net/msf2-emac.h | 53 ++++
19
hw/arm/nrf51_soc.c | 133 ++++++++++++++++++++++++++++++++
15
hw/net/msf2-emac.c | 589 +++++++++++++++++++++++++++++++++++++
20
default-configs/arm-softmmu.mak | 1 +
16
MAINTAINERS | 2 +
21
4 files changed, 176 insertions(+)
17
4 files changed, 645 insertions(+)
22
create mode 100644 include/hw/arm/nrf51_soc.h
18
create mode 100644 include/hw/net/msf2-emac.h
23
create mode 100644 hw/arm/nrf51_soc.c
19
create mode 100644 hw/net/msf2-emac.c
24
20
25
diff --git a/hw/arm/Makefile.objs b/hw/arm/Makefile.objs
21
diff --git a/hw/net/Makefile.objs b/hw/net/Makefile.objs
26
index XXXXXXX..XXXXXXX 100644
22
index XXXXXXX..XXXXXXX 100644
27
--- a/hw/arm/Makefile.objs
23
--- a/hw/net/Makefile.objs
28
+++ b/hw/arm/Makefile.objs
24
+++ b/hw/net/Makefile.objs
29
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_IOTKIT) += iotkit.o
25
@@ -XXX,XX +XXX,XX @@ common-obj-$(CONFIG_ROCKER) += rocker/rocker.o rocker/rocker_fp.o \
30
obj-$(CONFIG_FSL_IMX7) += fsl-imx7.o mcimx7d-sabre.o
26
obj-$(call lnot,$(CONFIG_ROCKER)) += rocker/qmp-norocker.o
31
obj-$(CONFIG_ARM_SMMUV3) += smmu-common.o smmuv3.o
27
32
obj-$(CONFIG_FSL_IMX6UL) += fsl-imx6ul.o mcimx6ul-evk.o
28
common-obj-$(CONFIG_CAN_BUS) += can/
33
+obj-$(CONFIG_NRF51_SOC) += nrf51_soc.o
29
+common-obj-$(CONFIG_MSF2) += msf2-emac.o
34
diff --git a/include/hw/arm/nrf51_soc.h b/include/hw/arm/nrf51_soc.h
30
diff --git a/include/hw/net/msf2-emac.h b/include/hw/net/msf2-emac.h
35
new file mode 100644
31
new file mode 100644
36
index XXXXXXX..XXXXXXX
32
index XXXXXXX..XXXXXXX
37
--- /dev/null
33
--- /dev/null
38
+++ b/include/hw/arm/nrf51_soc.h
34
+++ b/include/hw/net/msf2-emac.h
39
@@ -XXX,XX +XXX,XX @@
35
@@ -XXX,XX +XXX,XX @@
40
+/*
36
+/*
41
+ * Nordic Semiconductor nRF51 SoC
37
+ * QEMU model of the Smartfusion2 Ethernet MAC.
42
+ *
38
+ *
43
+ * Copyright 2018 Joel Stanley <joel@jms.id.au>
39
+ * Copyright (c) 2020 Subbaraya Sundeep <sundeep.lkml@gmail.com>.
44
+ *
40
+ *
45
+ * This code is licensed under the GPL version 2 or later. See
41
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
46
+ * the COPYING file in the top-level directory.
42
+ * of this software and associated documentation files (the "Software"), to deal
43
+ * in the Software without restriction, including without limitation the rights
44
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
45
+ * copies of the Software, and to permit persons to whom the Software is
46
+ * furnished to do so, subject to the following conditions:
47
+ *
48
+ * The above copyright notice and this permission notice shall be included in
49
+ * all copies or substantial portions of the Software.
50
+ *
51
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
52
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
53
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
54
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
55
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
56
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
57
+ * THE SOFTWARE.
47
+ */
58
+ */
48
+
59
+
49
+#ifndef NRF51_SOC_H
50
+#define NRF51_SOC_H
51
+
52
+#include "hw/sysbus.h"
60
+#include "hw/sysbus.h"
53
+#include "hw/arm/armv7m.h"
61
+#include "exec/memory.h"
54
+
62
+#include "net/net.h"
55
+#define TYPE_NRF51_SOC "nrf51-soc"
63
+#include "net/eth.h"
56
+#define NRF51_SOC(obj) \
64
+
57
+ OBJECT_CHECK(NRF51State, (obj), TYPE_NRF51_SOC)
65
+#define TYPE_MSS_EMAC "msf2-emac"
58
+
66
+#define MSS_EMAC(obj) \
59
+typedef struct NRF51State {
67
+ OBJECT_CHECK(MSF2EmacState, (obj), TYPE_MSS_EMAC)
60
+ /*< private >*/
68
+
61
+ SysBusDevice parent_obj;
69
+#define R_MAX (0x1a0 / 4)
62
+
70
+#define PHY_MAX_REGS 32
63
+ /*< public >*/
71
+
64
+ ARMv7MState cpu;
72
+typedef struct MSF2EmacState {
65
+
73
+ SysBusDevice parent;
66
+ MemoryRegion iomem;
74
+
67
+ MemoryRegion sram;
75
+ MemoryRegion mmio;
68
+ MemoryRegion flash;
76
+ MemoryRegion *dma_mr;
69
+
77
+ AddressSpace dma_as;
70
+ uint32_t sram_size;
78
+
71
+ uint32_t flash_size;
79
+ qemu_irq irq;
72
+
80
+ NICState *nic;
73
+ MemoryRegion *board_memory;
81
+ NICConf conf;
74
+
82
+
75
+ MemoryRegion container;
83
+ uint8_t mac_addr[ETH_ALEN];
76
+
84
+ uint32_t rx_desc;
77
+} NRF51State;
85
+ uint16_t phy_regs[PHY_MAX_REGS];
78
+
86
+
79
+#endif
87
+ uint32_t regs[R_MAX];
80
+
88
+} MSF2EmacState;
81
diff --git a/hw/arm/nrf51_soc.c b/hw/arm/nrf51_soc.c
89
diff --git a/hw/net/msf2-emac.c b/hw/net/msf2-emac.c
82
new file mode 100644
90
new file mode 100644
83
index XXXXXXX..XXXXXXX
91
index XXXXXXX..XXXXXXX
84
--- /dev/null
92
--- /dev/null
85
+++ b/hw/arm/nrf51_soc.c
93
+++ b/hw/net/msf2-emac.c
86
@@ -XXX,XX +XXX,XX @@
94
@@ -XXX,XX +XXX,XX @@
87
+/*
95
+/*
88
+ * Nordic Semiconductor nRF51 SoC
96
+ * QEMU model of the Smartfusion2 Ethernet MAC.
89
+ * http://infocenter.nordicsemi.com/pdf/nRF51_RM_v3.0.1.pdf
97
+ *
90
+ *
98
+ * Copyright (c) 2020 Subbaraya Sundeep <sundeep.lkml@gmail.com>.
91
+ * Copyright 2018 Joel Stanley <joel@jms.id.au>
99
+ *
92
+ *
100
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
93
+ * This code is licensed under the GPL version 2 or later. See
101
+ * of this software and associated documentation files (the "Software"), to deal
94
+ * the COPYING file in the top-level directory.
102
+ * in the Software without restriction, including without limitation the rights
103
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
104
+ * copies of the Software, and to permit persons to whom the Software is
105
+ * furnished to do so, subject to the following conditions:
106
+ *
107
+ * The above copyright notice and this permission notice shall be included in
108
+ * all copies or substantial portions of the Software.
109
+ *
110
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
111
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
112
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
113
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
114
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
115
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
116
+ * THE SOFTWARE.
117
+ *
118
+ * Refer to section Ethernet MAC in the document:
119
+ * UG0331: SmartFusion2 Microcontroller Subsystem User Guide
120
+ * Datasheet URL:
121
+ * https://www.microsemi.com/document-portal/cat_view/56661-internal-documents/
122
+ * 56758-soc?lang=en&limit=20&limitstart=220
95
+ */
123
+ */
96
+
124
+
97
+#include "qemu/osdep.h"
125
+#include "qemu/osdep.h"
126
+#include "qemu-common.h"
127
+#include "qemu/log.h"
98
+#include "qapi/error.h"
128
+#include "qapi/error.h"
99
+#include "qemu-common.h"
100
+#include "hw/arm/arm.h"
101
+#include "hw/sysbus.h"
102
+#include "hw/boards.h"
103
+#include "hw/devices.h"
104
+#include "hw/misc/unimp.h"
105
+#include "exec/address-spaces.h"
129
+#include "exec/address-spaces.h"
106
+#include "sysemu/sysemu.h"
130
+#include "hw/registerfields.h"
107
+#include "qemu/log.h"
131
+#include "hw/net/msf2-emac.h"
108
+#include "cpu.h"
132
+#include "hw/net/mii.h"
109
+
133
+#include "hw/irq.h"
110
+#include "hw/arm/nrf51_soc.h"
134
+#include "hw/qdev-properties.h"
111
+
135
+#include "migration/vmstate.h"
112
+#define IOMEM_BASE 0x40000000
136
+
113
+#define IOMEM_SIZE 0x20000000
137
+REG32(CFG1, 0x0)
114
+
138
+ FIELD(CFG1, RESET, 31, 1)
115
+#define FICR_BASE 0x10000000
139
+ FIELD(CFG1, RX_EN, 2, 1)
116
+#define FICR_SIZE 0x000000fc
140
+ FIELD(CFG1, TX_EN, 0, 1)
117
+
141
+ FIELD(CFG1, LB_EN, 8, 1)
118
+#define FLASH_BASE 0x00000000
142
+REG32(CFG2, 0x4)
119
+#define SRAM_BASE 0x20000000
143
+REG32(IFG, 0x8)
120
+
144
+REG32(HALF_DUPLEX, 0xc)
121
+#define PRIVATE_BASE 0xF0000000
145
+REG32(MAX_FRAME_LENGTH, 0x10)
122
+#define PRIVATE_SIZE 0x10000000
146
+REG32(MII_CMD, 0x24)
123
+
147
+ FIELD(MII_CMD, READ, 0, 1)
124
+/*
148
+REG32(MII_ADDR, 0x28)
125
+ * The size and base is for the NRF51822 part. If other parts
149
+ FIELD(MII_ADDR, REGADDR, 0, 5)
126
+ * are supported in the future, add a sub-class of NRF51SoC for
150
+ FIELD(MII_ADDR, PHYADDR, 8, 5)
127
+ * the specific variants
151
+REG32(MII_CTL, 0x2c)
128
+ */
152
+REG32(MII_STS, 0x30)
129
+#define NRF51822_FLASH_SIZE (256 * 1024)
153
+REG32(STA1, 0x40)
130
+#define NRF51822_SRAM_SIZE (16 * 1024)
154
+REG32(STA2, 0x44)
131
+
155
+REG32(FIFO_CFG0, 0x48)
132
+static void nrf51_soc_realize(DeviceState *dev_soc, Error **errp)
156
+REG32(FIFO_CFG4, 0x58)
133
+{
157
+ FIELD(FIFO_CFG4, BCAST, 9, 1)
134
+ NRF51State *s = NRF51_SOC(dev_soc);
158
+ FIELD(FIFO_CFG4, MCAST, 8, 1)
135
+ Error *err = NULL;
159
+REG32(FIFO_CFG5, 0x5C)
136
+
160
+ FIELD(FIFO_CFG5, BCAST, 9, 1)
137
+ if (!s->board_memory) {
161
+ FIELD(FIFO_CFG5, MCAST, 8, 1)
138
+ error_setg(errp, "memory property was not set");
162
+REG32(DMA_TX_CTL, 0x180)
163
+ FIELD(DMA_TX_CTL, EN, 0, 1)
164
+REG32(DMA_TX_DESC, 0x184)
165
+REG32(DMA_TX_STATUS, 0x188)
166
+ FIELD(DMA_TX_STATUS, PKTCNT, 16, 8)
167
+ FIELD(DMA_TX_STATUS, UNDERRUN, 1, 1)
168
+ FIELD(DMA_TX_STATUS, PKT_SENT, 0, 1)
169
+REG32(DMA_RX_CTL, 0x18c)
170
+ FIELD(DMA_RX_CTL, EN, 0, 1)
171
+REG32(DMA_RX_DESC, 0x190)
172
+REG32(DMA_RX_STATUS, 0x194)
173
+ FIELD(DMA_RX_STATUS, PKTCNT, 16, 8)
174
+ FIELD(DMA_RX_STATUS, OVERFLOW, 2, 1)
175
+ FIELD(DMA_RX_STATUS, PKT_RCVD, 0, 1)
176
+REG32(DMA_IRQ_MASK, 0x198)
177
+REG32(DMA_IRQ, 0x19c)
178
+
179
+#define EMPTY_MASK (1 << 31)
180
+#define PKT_SIZE 0x7FF
181
+#define PHYADDR 0x1
182
+#define MAX_PKT_SIZE 2048
183
+
184
+typedef struct {
185
+ uint32_t pktaddr;
186
+ uint32_t pktsize;
187
+ uint32_t next;
188
+} EmacDesc;
189
+
190
+static uint32_t emac_get_isr(MSF2EmacState *s)
191
+{
192
+ uint32_t ier = s->regs[R_DMA_IRQ_MASK];
193
+ uint32_t tx = s->regs[R_DMA_TX_STATUS] & 0xF;
194
+ uint32_t rx = s->regs[R_DMA_RX_STATUS] & 0xF;
195
+ uint32_t isr = (rx << 4) | tx;
196
+
197
+ s->regs[R_DMA_IRQ] = ier & isr;
198
+ return s->regs[R_DMA_IRQ];
199
+}
200
+
201
+static void emac_update_irq(MSF2EmacState *s)
202
+{
203
+ bool intr = emac_get_isr(s);
204
+
205
+ qemu_set_irq(s->irq, intr);
206
+}
207
+
208
+static void emac_load_desc(MSF2EmacState *s, EmacDesc *d, hwaddr desc)
209
+{
210
+ address_space_read(&s->dma_as, desc, MEMTXATTRS_UNSPECIFIED, d, sizeof *d);
211
+ /* Convert from LE into host endianness. */
212
+ d->pktaddr = le32_to_cpu(d->pktaddr);
213
+ d->pktsize = le32_to_cpu(d->pktsize);
214
+ d->next = le32_to_cpu(d->next);
215
+}
216
+
217
+static void emac_store_desc(MSF2EmacState *s, EmacDesc *d, hwaddr desc)
218
+{
219
+ /* Convert from host endianness into LE. */
220
+ d->pktaddr = cpu_to_le32(d->pktaddr);
221
+ d->pktsize = cpu_to_le32(d->pktsize);
222
+ d->next = cpu_to_le32(d->next);
223
+
224
+ address_space_write(&s->dma_as, desc, MEMTXATTRS_UNSPECIFIED, d, sizeof *d);
225
+}
226
+
227
+static void msf2_dma_tx(MSF2EmacState *s)
228
+{
229
+ NetClientState *nc = qemu_get_queue(s->nic);
230
+ hwaddr desc = s->regs[R_DMA_TX_DESC];
231
+ uint8_t buf[MAX_PKT_SIZE];
232
+ EmacDesc d;
233
+ int size;
234
+ uint8_t pktcnt;
235
+ uint32_t status;
236
+
237
+ if (!(s->regs[R_CFG1] & R_CFG1_TX_EN_MASK)) {
139
+ return;
238
+ return;
140
+ }
239
+ }
141
+
240
+
142
+ object_property_set_link(OBJECT(&s->cpu), OBJECT(&s->container), "memory",
241
+ while (1) {
143
+ &err);
242
+ emac_load_desc(s, &d, desc);
144
+ if (err) {
243
+ if (d.pktsize & EMPTY_MASK) {
145
+ error_propagate(errp, err);
244
+ break;
245
+ }
246
+ size = d.pktsize & PKT_SIZE;
247
+ address_space_read(&s->dma_as, d.pktaddr, MEMTXATTRS_UNSPECIFIED,
248
+ buf, size);
249
+ /*
250
+ * This is very basic way to send packets. Ideally there should be
251
+ * a FIFO and packets should be sent out from FIFO only when
252
+ * R_CFG1 bit 0 is set.
253
+ */
254
+ if (s->regs[R_CFG1] & R_CFG1_LB_EN_MASK) {
255
+ nc->info->receive(nc, buf, size);
256
+ } else {
257
+ qemu_send_packet(nc, buf, size);
258
+ }
259
+ d.pktsize |= EMPTY_MASK;
260
+ emac_store_desc(s, &d, desc);
261
+ /* update sent packets count */
262
+ status = s->regs[R_DMA_TX_STATUS];
263
+ pktcnt = FIELD_EX32(status, DMA_TX_STATUS, PKTCNT);
264
+ pktcnt++;
265
+ s->regs[R_DMA_TX_STATUS] = FIELD_DP32(status, DMA_TX_STATUS,
266
+ PKTCNT, pktcnt);
267
+ s->regs[R_DMA_TX_STATUS] |= R_DMA_TX_STATUS_PKT_SENT_MASK;
268
+ desc = d.next;
269
+ }
270
+ s->regs[R_DMA_TX_STATUS] |= R_DMA_TX_STATUS_UNDERRUN_MASK;
271
+ s->regs[R_DMA_TX_CTL] &= ~R_DMA_TX_CTL_EN_MASK;
272
+}
273
+
274
+static void msf2_phy_update_link(MSF2EmacState *s)
275
+{
276
+ /* Autonegotiation status mirrors link status. */
277
+ if (qemu_get_queue(s->nic)->link_down) {
278
+ s->phy_regs[MII_BMSR] &= ~(MII_BMSR_AN_COMP |
279
+ MII_BMSR_LINK_ST);
280
+ } else {
281
+ s->phy_regs[MII_BMSR] |= (MII_BMSR_AN_COMP |
282
+ MII_BMSR_LINK_ST);
283
+ }
284
+}
285
+
286
+static void msf2_phy_reset(MSF2EmacState *s)
287
+{
288
+ memset(&s->phy_regs[0], 0, sizeof(s->phy_regs));
289
+ s->phy_regs[MII_BMCR] = 0x1140;
290
+ s->phy_regs[MII_BMSR] = 0x7968;
291
+ s->phy_regs[MII_PHYID1] = 0x0022;
292
+ s->phy_regs[MII_PHYID2] = 0x1550;
293
+ s->phy_regs[MII_ANAR] = 0x01E1;
294
+ s->phy_regs[MII_ANLPAR] = 0xCDE1;
295
+
296
+ msf2_phy_update_link(s);
297
+}
298
+
299
+static void write_to_phy(MSF2EmacState *s)
300
+{
301
+ uint8_t reg_addr = s->regs[R_MII_ADDR] & R_MII_ADDR_REGADDR_MASK;
302
+ uint8_t phy_addr = (s->regs[R_MII_ADDR] >> R_MII_ADDR_PHYADDR_SHIFT) &
303
+ R_MII_ADDR_REGADDR_MASK;
304
+ uint16_t data = s->regs[R_MII_CTL] & 0xFFFF;
305
+
306
+ if (phy_addr != PHYADDR) {
146
+ return;
307
+ return;
147
+ }
308
+ }
148
+ object_property_set_bool(OBJECT(&s->cpu), true, "realized", &err);
309
+
149
+ if (err) {
310
+ switch (reg_addr) {
150
+ error_propagate(errp, err);
311
+ case MII_BMCR:
312
+ if (data & MII_BMCR_RESET) {
313
+ /* Phy reset */
314
+ msf2_phy_reset(s);
315
+ data &= ~MII_BMCR_RESET;
316
+ }
317
+ if (data & MII_BMCR_AUTOEN) {
318
+ /* Complete autonegotiation immediately */
319
+ data &= ~MII_BMCR_AUTOEN;
320
+ s->phy_regs[MII_BMSR] |= MII_BMSR_AN_COMP;
321
+ }
322
+ break;
323
+ }
324
+
325
+ s->phy_regs[reg_addr] = data;
326
+}
327
+
328
+static uint16_t read_from_phy(MSF2EmacState *s)
329
+{
330
+ uint8_t reg_addr = s->regs[R_MII_ADDR] & R_MII_ADDR_REGADDR_MASK;
331
+ uint8_t phy_addr = (s->regs[R_MII_ADDR] >> R_MII_ADDR_PHYADDR_SHIFT) &
332
+ R_MII_ADDR_REGADDR_MASK;
333
+
334
+ if (phy_addr == PHYADDR) {
335
+ return s->phy_regs[reg_addr];
336
+ } else {
337
+ return 0xFFFF;
338
+ }
339
+}
340
+
341
+static void msf2_emac_do_reset(MSF2EmacState *s)
342
+{
343
+ memset(&s->regs[0], 0, sizeof(s->regs));
344
+ s->regs[R_CFG1] = 0x80000000;
345
+ s->regs[R_CFG2] = 0x00007000;
346
+ s->regs[R_IFG] = 0x40605060;
347
+ s->regs[R_HALF_DUPLEX] = 0x00A1F037;
348
+ s->regs[R_MAX_FRAME_LENGTH] = 0x00000600;
349
+ s->regs[R_FIFO_CFG5] = 0X3FFFF;
350
+
351
+ msf2_phy_reset(s);
352
+}
353
+
354
+static uint64_t emac_read(void *opaque, hwaddr addr, unsigned int size)
355
+{
356
+ MSF2EmacState *s = opaque;
357
+ uint32_t r = 0;
358
+
359
+ addr >>= 2;
360
+
361
+ switch (addr) {
362
+ case R_DMA_IRQ:
363
+ r = emac_get_isr(s);
364
+ break;
365
+ default:
366
+ if (addr >= ARRAY_SIZE(s->regs)) {
367
+ qemu_log_mask(LOG_GUEST_ERROR,
368
+ "%s: Bad offset 0x%" HWADDR_PRIx "\n", __func__,
369
+ addr * 4);
370
+ return r;
371
+ }
372
+ r = s->regs[addr];
373
+ break;
374
+ }
375
+ return r;
376
+}
377
+
378
+static void emac_write(void *opaque, hwaddr addr, uint64_t val64,
379
+ unsigned int size)
380
+{
381
+ MSF2EmacState *s = opaque;
382
+ uint32_t value = val64;
383
+ uint32_t enreqbits;
384
+ uint8_t pktcnt;
385
+
386
+ addr >>= 2;
387
+ switch (addr) {
388
+ case R_DMA_TX_CTL:
389
+ s->regs[addr] = value;
390
+ if (value & R_DMA_TX_CTL_EN_MASK) {
391
+ msf2_dma_tx(s);
392
+ }
393
+ break;
394
+ case R_DMA_RX_CTL:
395
+ s->regs[addr] = value;
396
+ if (value & R_DMA_RX_CTL_EN_MASK) {
397
+ s->rx_desc = s->regs[R_DMA_RX_DESC];
398
+ qemu_flush_queued_packets(qemu_get_queue(s->nic));
399
+ }
400
+ break;
401
+ case R_CFG1:
402
+ s->regs[addr] = value;
403
+ if (value & R_CFG1_RESET_MASK) {
404
+ msf2_emac_do_reset(s);
405
+ }
406
+ break;
407
+ case R_FIFO_CFG0:
408
+ /*
409
+ * For our implementation, turning on modules is instantaneous,
410
+ * so the states requested via the *ENREQ bits appear in the
411
+ * *ENRPLY bits immediately. Also the reset bits to reset PE-MCXMAC
412
+ * module are not emulated here since it deals with start of frames,
413
+ * inter-packet gap and control frames.
414
+ */
415
+ enreqbits = extract32(value, 8, 5);
416
+ s->regs[addr] = deposit32(value, 16, 5, enreqbits);
417
+ break;
418
+ case R_DMA_TX_DESC:
419
+ if (value & 0x3) {
420
+ qemu_log_mask(LOG_GUEST_ERROR, "Tx Descriptor address should be"
421
+ " 32 bit aligned\n");
422
+ }
423
+ /* Ignore [1:0] bits */
424
+ s->regs[addr] = value & ~3;
425
+ break;
426
+ case R_DMA_RX_DESC:
427
+ if (value & 0x3) {
428
+ qemu_log_mask(LOG_GUEST_ERROR, "Rx Descriptor address should be"
429
+ " 32 bit aligned\n");
430
+ }
431
+ /* Ignore [1:0] bits */
432
+ s->regs[addr] = value & ~3;
433
+ break;
434
+ case R_DMA_TX_STATUS:
435
+ if (value & R_DMA_TX_STATUS_UNDERRUN_MASK) {
436
+ s->regs[addr] &= ~R_DMA_TX_STATUS_UNDERRUN_MASK;
437
+ }
438
+ if (value & R_DMA_TX_STATUS_PKT_SENT_MASK) {
439
+ pktcnt = FIELD_EX32(s->regs[addr], DMA_TX_STATUS, PKTCNT);
440
+ pktcnt--;
441
+ s->regs[addr] = FIELD_DP32(s->regs[addr], DMA_TX_STATUS,
442
+ PKTCNT, pktcnt);
443
+ if (pktcnt == 0) {
444
+ s->regs[addr] &= ~R_DMA_TX_STATUS_PKT_SENT_MASK;
445
+ }
446
+ }
447
+ break;
448
+ case R_DMA_RX_STATUS:
449
+ if (value & R_DMA_RX_STATUS_OVERFLOW_MASK) {
450
+ s->regs[addr] &= ~R_DMA_RX_STATUS_OVERFLOW_MASK;
451
+ }
452
+ if (value & R_DMA_RX_STATUS_PKT_RCVD_MASK) {
453
+ pktcnt = FIELD_EX32(s->regs[addr], DMA_RX_STATUS, PKTCNT);
454
+ pktcnt--;
455
+ s->regs[addr] = FIELD_DP32(s->regs[addr], DMA_RX_STATUS,
456
+ PKTCNT, pktcnt);
457
+ if (pktcnt == 0) {
458
+ s->regs[addr] &= ~R_DMA_RX_STATUS_PKT_RCVD_MASK;
459
+ }
460
+ }
461
+ break;
462
+ case R_DMA_IRQ:
463
+ break;
464
+ case R_MII_CMD:
465
+ if (value & R_MII_CMD_READ_MASK) {
466
+ s->regs[R_MII_STS] = read_from_phy(s);
467
+ }
468
+ break;
469
+ case R_MII_CTL:
470
+ s->regs[addr] = value;
471
+ write_to_phy(s);
472
+ break;
473
+ case R_STA1:
474
+ s->regs[addr] = value;
475
+ /*
476
+ * R_STA1 [31:24] : octet 1 of mac address
477
+ * R_STA1 [23:16] : octet 2 of mac address
478
+ * R_STA1 [15:8] : octet 3 of mac address
479
+ * R_STA1 [7:0] : octet 4 of mac address
480
+ */
481
+ stl_be_p(s->mac_addr, value);
482
+ break;
483
+ case R_STA2:
484
+ s->regs[addr] = value;
485
+ /*
486
+ * R_STA2 [31:24] : octet 5 of mac address
487
+ * R_STA2 [23:16] : octet 6 of mac address
488
+ */
489
+ stw_be_p(s->mac_addr + 4, value >> 16);
490
+ break;
491
+ default:
492
+ if (addr >= ARRAY_SIZE(s->regs)) {
493
+ qemu_log_mask(LOG_GUEST_ERROR,
494
+ "%s: Bad offset 0x%" HWADDR_PRIx "\n", __func__,
495
+ addr * 4);
496
+ return;
497
+ }
498
+ s->regs[addr] = value;
499
+ break;
500
+ }
501
+ emac_update_irq(s);
502
+}
503
+
504
+static const MemoryRegionOps emac_ops = {
505
+ .read = emac_read,
506
+ .write = emac_write,
507
+ .endianness = DEVICE_NATIVE_ENDIAN,
508
+ .impl = {
509
+ .min_access_size = 4,
510
+ .max_access_size = 4
511
+ }
512
+};
513
+
514
+static bool emac_can_rx(NetClientState *nc)
515
+{
516
+ MSF2EmacState *s = qemu_get_nic_opaque(nc);
517
+
518
+ return (s->regs[R_CFG1] & R_CFG1_RX_EN_MASK) &&
519
+ (s->regs[R_DMA_RX_CTL] & R_DMA_RX_CTL_EN_MASK);
520
+}
521
+
522
+static bool addr_filter_ok(MSF2EmacState *s, const uint8_t *buf)
523
+{
524
+ /* The broadcast MAC address: FF:FF:FF:FF:FF:FF */
525
+ const uint8_t broadcast_addr[] = { 0xFF, 0xFF, 0xFF, 0xFF,
526
+ 0xFF, 0xFF };
527
+ bool bcast_en = true;
528
+ bool mcast_en = true;
529
+
530
+ if (s->regs[R_FIFO_CFG5] & R_FIFO_CFG5_BCAST_MASK) {
531
+ bcast_en = true; /* Broadcast dont care for drop circuitry */
532
+ } else if (s->regs[R_FIFO_CFG4] & R_FIFO_CFG4_BCAST_MASK) {
533
+ bcast_en = false;
534
+ }
535
+
536
+ if (s->regs[R_FIFO_CFG5] & R_FIFO_CFG5_MCAST_MASK) {
537
+ mcast_en = true; /* Multicast dont care for drop circuitry */
538
+ } else if (s->regs[R_FIFO_CFG4] & R_FIFO_CFG4_MCAST_MASK) {
539
+ mcast_en = false;
540
+ }
541
+
542
+ if (!memcmp(buf, broadcast_addr, sizeof(broadcast_addr))) {
543
+ return bcast_en;
544
+ }
545
+
546
+ if (buf[0] & 1) {
547
+ return mcast_en;
548
+ }
549
+
550
+ return !memcmp(buf, s->mac_addr, sizeof(s->mac_addr));
551
+}
552
+
553
+static ssize_t emac_rx(NetClientState *nc, const uint8_t *buf, size_t size)
554
+{
555
+ MSF2EmacState *s = qemu_get_nic_opaque(nc);
556
+ EmacDesc d;
557
+ uint8_t pktcnt;
558
+ uint32_t status;
559
+
560
+ if (size > (s->regs[R_MAX_FRAME_LENGTH] & 0xFFFF)) {
561
+ return size;
562
+ }
563
+ if (!addr_filter_ok(s, buf)) {
564
+ return size;
565
+ }
566
+
567
+ emac_load_desc(s, &d, s->rx_desc);
568
+
569
+ if (d.pktsize & EMPTY_MASK) {
570
+ address_space_write(&s->dma_as, d.pktaddr, MEMTXATTRS_UNSPECIFIED,
571
+ buf, size & PKT_SIZE);
572
+ d.pktsize = size & PKT_SIZE;
573
+ emac_store_desc(s, &d, s->rx_desc);
574
+ /* update received packets count */
575
+ status = s->regs[R_DMA_RX_STATUS];
576
+ pktcnt = FIELD_EX32(status, DMA_RX_STATUS, PKTCNT);
577
+ pktcnt++;
578
+ s->regs[R_DMA_RX_STATUS] = FIELD_DP32(status, DMA_RX_STATUS,
579
+ PKTCNT, pktcnt);
580
+ s->regs[R_DMA_RX_STATUS] |= R_DMA_RX_STATUS_PKT_RCVD_MASK;
581
+ s->rx_desc = d.next;
582
+ } else {
583
+ s->regs[R_DMA_RX_CTL] &= ~R_DMA_RX_CTL_EN_MASK;
584
+ s->regs[R_DMA_RX_STATUS] |= R_DMA_RX_STATUS_OVERFLOW_MASK;
585
+ }
586
+ emac_update_irq(s);
587
+ return size;
588
+}
589
+
590
+static void msf2_emac_reset(DeviceState *dev)
591
+{
592
+ MSF2EmacState *s = MSS_EMAC(dev);
593
+
594
+ msf2_emac_do_reset(s);
595
+}
596
+
597
+static void emac_set_link(NetClientState *nc)
598
+{
599
+ MSF2EmacState *s = qemu_get_nic_opaque(nc);
600
+
601
+ msf2_phy_update_link(s);
602
+}
603
+
604
+static NetClientInfo net_msf2_emac_info = {
605
+ .type = NET_CLIENT_DRIVER_NIC,
606
+ .size = sizeof(NICState),
607
+ .can_receive = emac_can_rx,
608
+ .receive = emac_rx,
609
+ .link_status_changed = emac_set_link,
610
+};
611
+
612
+static void msf2_emac_realize(DeviceState *dev, Error **errp)
613
+{
614
+ MSF2EmacState *s = MSS_EMAC(dev);
615
+
616
+ if (!s->dma_mr) {
617
+ error_setg(errp, "MSS_EMAC 'ahb-bus' link not set");
151
+ return;
618
+ return;
152
+ }
619
+ }
153
+
620
+
154
+ memory_region_add_subregion_overlap(&s->container, 0, s->board_memory, -1);
621
+ address_space_init(&s->dma_as, s->dma_mr, "emac-ahb");
155
+
622
+
156
+ memory_region_init_rom(&s->flash, OBJECT(s), "nrf51.flash", s->flash_size,
623
+ qemu_macaddr_default_if_unset(&s->conf.macaddr);
157
+ &err);
624
+ s->nic = qemu_new_nic(&net_msf2_emac_info, &s->conf,
158
+ if (err) {
625
+ object_get_typename(OBJECT(dev)), dev->id, s);
159
+ error_propagate(errp, err);
626
+ qemu_format_nic_info_str(qemu_get_queue(s->nic), s->conf.macaddr.a);
160
+ return;
627
+}
161
+ }
628
+
162
+ memory_region_add_subregion(&s->container, FLASH_BASE, &s->flash);
629
+static void msf2_emac_init(Object *obj)
163
+
630
+{
164
+ memory_region_init_ram(&s->sram, NULL, "nrf51.sram", s->sram_size, &err);
631
+ MSF2EmacState *s = MSS_EMAC(obj);
165
+ if (err) {
632
+
166
+ error_propagate(errp, err);
633
+ sysbus_init_irq(SYS_BUS_DEVICE(obj), &s->irq);
167
+ return;
634
+
168
+ }
635
+ memory_region_init_io(&s->mmio, obj, &emac_ops, s,
169
+ memory_region_add_subregion(&s->container, SRAM_BASE, &s->sram);
636
+ "msf2-emac", R_MAX * 4);
170
+
637
+ sysbus_init_mmio(SYS_BUS_DEVICE(obj), &s->mmio);
171
+ create_unimplemented_device("nrf51_soc.io", IOMEM_BASE, IOMEM_SIZE);
638
+}
172
+ create_unimplemented_device("nrf51_soc.ficr", FICR_BASE, FICR_SIZE);
639
+
173
+ create_unimplemented_device("nrf51_soc.private",
640
+static Property msf2_emac_properties[] = {
174
+ PRIVATE_BASE, PRIVATE_SIZE);
641
+ DEFINE_PROP_LINK("ahb-bus", MSF2EmacState, dma_mr,
175
+}
642
+ TYPE_MEMORY_REGION, MemoryRegion *),
176
+
643
+ DEFINE_NIC_PROPERTIES(MSF2EmacState, conf),
177
+static void nrf51_soc_init(Object *obj)
178
+{
179
+ NRF51State *s = NRF51_SOC(obj);
180
+
181
+ memory_region_init(&s->container, obj, "nrf51-container", UINT64_MAX);
182
+
183
+ sysbus_init_child_obj(OBJECT(s), "armv6m", OBJECT(&s->cpu), sizeof(s->cpu),
184
+ TYPE_ARMV7M);
185
+ qdev_prop_set_string(DEVICE(&s->cpu), "cpu-type",
186
+ ARM_CPU_TYPE_NAME("cortex-m0"));
187
+ qdev_prop_set_uint32(DEVICE(&s->cpu), "num-irq", 32);
188
+}
189
+
190
+static Property nrf51_soc_properties[] = {
191
+ DEFINE_PROP_LINK("memory", NRF51State, board_memory, TYPE_MEMORY_REGION,
192
+ MemoryRegion *),
193
+ DEFINE_PROP_UINT32("sram-size", NRF51State, sram_size, NRF51822_SRAM_SIZE),
194
+ DEFINE_PROP_UINT32("flash-size", NRF51State, flash_size,
195
+ NRF51822_FLASH_SIZE),
196
+ DEFINE_PROP_END_OF_LIST(),
644
+ DEFINE_PROP_END_OF_LIST(),
197
+};
645
+};
198
+
646
+
199
+static void nrf51_soc_class_init(ObjectClass *klass, void *data)
647
+static const VMStateDescription vmstate_msf2_emac = {
648
+ .name = TYPE_MSS_EMAC,
649
+ .version_id = 1,
650
+ .minimum_version_id = 1,
651
+ .fields = (VMStateField[]) {
652
+ VMSTATE_UINT8_ARRAY(mac_addr, MSF2EmacState, ETH_ALEN),
653
+ VMSTATE_UINT32(rx_desc, MSF2EmacState),
654
+ VMSTATE_UINT16_ARRAY(phy_regs, MSF2EmacState, PHY_MAX_REGS),
655
+ VMSTATE_UINT32_ARRAY(regs, MSF2EmacState, R_MAX),
656
+ VMSTATE_END_OF_LIST()
657
+ }
658
+};
659
+
660
+static void msf2_emac_class_init(ObjectClass *klass, void *data)
200
+{
661
+{
201
+ DeviceClass *dc = DEVICE_CLASS(klass);
662
+ DeviceClass *dc = DEVICE_CLASS(klass);
202
+
663
+
203
+ dc->realize = nrf51_soc_realize;
664
+ dc->realize = msf2_emac_realize;
204
+ dc->props = nrf51_soc_properties;
665
+ dc->reset = msf2_emac_reset;
205
+}
666
+ dc->vmsd = &vmstate_msf2_emac;
206
+
667
+ device_class_set_props(dc, msf2_emac_properties);
207
+static const TypeInfo nrf51_soc_info = {
668
+}
208
+ .name = TYPE_NRF51_SOC,
669
+
670
+static const TypeInfo msf2_emac_info = {
671
+ .name = TYPE_MSS_EMAC,
209
+ .parent = TYPE_SYS_BUS_DEVICE,
672
+ .parent = TYPE_SYS_BUS_DEVICE,
210
+ .instance_size = sizeof(NRF51State),
673
+ .instance_size = sizeof(MSF2EmacState),
211
+ .instance_init = nrf51_soc_init,
674
+ .instance_init = msf2_emac_init,
212
+ .class_init = nrf51_soc_class_init,
675
+ .class_init = msf2_emac_class_init,
213
+};
676
+};
214
+
677
+
215
+static void nrf51_soc_types(void)
678
+static void msf2_emac_register_types(void)
216
+{
679
+{
217
+ type_register_static(&nrf51_soc_info);
680
+ type_register_static(&msf2_emac_info);
218
+}
681
+}
219
+type_init(nrf51_soc_types)
682
+
220
diff --git a/default-configs/arm-softmmu.mak b/default-configs/arm-softmmu.mak
683
+type_init(msf2_emac_register_types)
684
diff --git a/MAINTAINERS b/MAINTAINERS
221
index XXXXXXX..XXXXXXX 100644
685
index XXXXXXX..XXXXXXX 100644
222
--- a/default-configs/arm-softmmu.mak
686
--- a/MAINTAINERS
223
+++ b/default-configs/arm-softmmu.mak
687
+++ b/MAINTAINERS
224
@@ -XXX,XX +XXX,XX @@ CONFIG_STM32F2XX_SYSCFG=y
688
@@ -XXX,XX +XXX,XX @@ F: include/hw/arm/msf2-soc.h
225
CONFIG_STM32F2XX_ADC=y
689
F: include/hw/misc/msf2-sysreg.h
226
CONFIG_STM32F2XX_SPI=y
690
F: include/hw/timer/mss-timer.h
227
CONFIG_STM32F205_SOC=y
691
F: include/hw/ssi/mss-spi.h
228
+CONFIG_NRF51_SOC=y
692
+F: hw/net/msf2-emac.c
229
693
+F: include/hw/net/msf2-emac.h
230
CONFIG_CMSDK_APB_TIMER=y
694
231
CONFIG_CMSDK_APB_DUALTIMER=y
695
Emcraft M2S-FG484
696
M: Subbaraya Sundeep <sundeep.lkml@gmail.com>
232
--
697
--
233
2.19.0
698
2.20.1
234
699
235
700
diff view generated by jsdifflib
New patch
1
From: Subbaraya Sundeep <sundeep.lkml@gmail.com>
1
2
3
With SmartFusion2 Ethernet MAC model in
4
place this patch adds the same to SoC.
5
6
Signed-off-by: Subbaraya Sundeep <sundeep.lkml@gmail.com>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Tested-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Message-id: 1587048891-30493-3-git-send-email-sundeep.lkml@gmail.com
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
include/hw/arm/msf2-soc.h | 2 ++
13
hw/arm/msf2-soc.c | 26 ++++++++++++++++++++++++--
14
2 files changed, 26 insertions(+), 2 deletions(-)
15
16
diff --git a/include/hw/arm/msf2-soc.h b/include/hw/arm/msf2-soc.h
17
index XXXXXXX..XXXXXXX 100644
18
--- a/include/hw/arm/msf2-soc.h
19
+++ b/include/hw/arm/msf2-soc.h
20
@@ -XXX,XX +XXX,XX @@
21
#include "hw/timer/mss-timer.h"
22
#include "hw/misc/msf2-sysreg.h"
23
#include "hw/ssi/mss-spi.h"
24
+#include "hw/net/msf2-emac.h"
25
26
#define TYPE_MSF2_SOC "msf2-soc"
27
#define MSF2_SOC(obj) OBJECT_CHECK(MSF2State, (obj), TYPE_MSF2_SOC)
28
@@ -XXX,XX +XXX,XX @@ typedef struct MSF2State {
29
MSF2SysregState sysreg;
30
MSSTimerState timer;
31
MSSSpiState spi[MSF2_NUM_SPIS];
32
+ MSF2EmacState emac;
33
} MSF2State;
34
35
#endif
36
diff --git a/hw/arm/msf2-soc.c b/hw/arm/msf2-soc.c
37
index XXXXXXX..XXXXXXX 100644
38
--- a/hw/arm/msf2-soc.c
39
+++ b/hw/arm/msf2-soc.c
40
@@ -XXX,XX +XXX,XX @@
41
/*
42
* SmartFusion2 SoC emulation.
43
*
44
- * Copyright (c) 2017 Subbaraya Sundeep <sundeep.lkml@gmail.com>
45
+ * Copyright (c) 2017-2020 Subbaraya Sundeep <sundeep.lkml@gmail.com>
46
*
47
* Permission is hereby granted, free of charge, to any person obtaining a copy
48
* of this software and associated documentation files (the "Software"), to deal
49
@@ -XXX,XX +XXX,XX @@
50
51
#define MSF2_TIMER_BASE 0x40004000
52
#define MSF2_SYSREG_BASE 0x40038000
53
+#define MSF2_EMAC_BASE 0x40041000
54
55
#define ENVM_BASE_ADDRESS 0x60000000
56
57
#define SRAM_BASE_ADDRESS 0x20000000
58
59
+#define MSF2_EMAC_IRQ 12
60
+
61
#define MSF2_ENVM_MAX_SIZE (512 * KiB)
62
63
/*
64
@@ -XXX,XX +XXX,XX @@ static void m2sxxx_soc_initfn(Object *obj)
65
sysbus_init_child_obj(obj, "spi[*]", &s->spi[i], sizeof(s->spi[i]),
66
TYPE_MSS_SPI);
67
}
68
+
69
+ sysbus_init_child_obj(obj, "emac", &s->emac, sizeof(s->emac),
70
+ TYPE_MSS_EMAC);
71
+ if (nd_table[0].used) {
72
+ qemu_check_nic_model(&nd_table[0], TYPE_MSS_EMAC);
73
+ qdev_set_nic_properties(DEVICE(&s->emac), &nd_table[0]);
74
+ }
75
}
76
77
static void m2sxxx_soc_realize(DeviceState *dev_soc, Error **errp)
78
@@ -XXX,XX +XXX,XX @@ static void m2sxxx_soc_realize(DeviceState *dev_soc, Error **errp)
79
g_free(bus_name);
80
}
81
82
+ dev = DEVICE(&s->emac);
83
+ object_property_set_link(OBJECT(&s->emac), OBJECT(get_system_memory()),
84
+ "ahb-bus", &error_abort);
85
+ object_property_set_bool(OBJECT(&s->emac), true, "realized", &err);
86
+ if (err != NULL) {
87
+ error_propagate(errp, err);
88
+ return;
89
+ }
90
+ busdev = SYS_BUS_DEVICE(dev);
91
+ sysbus_mmio_map(busdev, 0, MSF2_EMAC_BASE);
92
+ sysbus_connect_irq(busdev, 0,
93
+ qdev_get_gpio_in(armv7m, MSF2_EMAC_IRQ));
94
+
95
/* Below devices are not modelled yet. */
96
create_unimplemented_device("i2c_0", 0x40002000, 0x1000);
97
create_unimplemented_device("dma", 0x40003000, 0x1000);
98
@@ -XXX,XX +XXX,XX @@ static void m2sxxx_soc_realize(DeviceState *dev_soc, Error **errp)
99
create_unimplemented_device("can", 0x40015000, 0x1000);
100
create_unimplemented_device("rtc", 0x40017000, 0x1000);
101
create_unimplemented_device("apb_config", 0x40020000, 0x10000);
102
- create_unimplemented_device("emac", 0x40041000, 0x1000);
103
create_unimplemented_device("usb", 0x40043000, 0x1000);
104
}
105
106
--
107
2.20.1
108
109
diff view generated by jsdifflib
New patch
1
From: Subbaraya Sundeep <sundeep.lkml@gmail.com>
1
2
3
In addition to simple serial test this patch uses ping
4
to test the ethernet block modelled in SmartFusion2 SoC.
5
6
Signed-off-by: Subbaraya Sundeep <sundeep.lkml@gmail.com>
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
8
Tested-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
9
Message-id: 1587048891-30493-4-git-send-email-sundeep.lkml@gmail.com
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
12
tests/acceptance/boot_linux_console.py | 15 ++++++++++-----
13
1 file changed, 10 insertions(+), 5 deletions(-)
14
15
diff --git a/tests/acceptance/boot_linux_console.py b/tests/acceptance/boot_linux_console.py
16
index XXXXXXX..XXXXXXX 100644
17
--- a/tests/acceptance/boot_linux_console.py
18
+++ b/tests/acceptance/boot_linux_console.py
19
@@ -XXX,XX +XXX,XX @@ class BootLinuxConsole(Test):
20
"""
21
uboot_url = ('https://raw.githubusercontent.com/'
22
'Subbaraya-Sundeep/qemu-test-binaries/'
23
- 'fa030bd77a014a0b8e360d3b7011df89283a2f0b/u-boot')
24
- uboot_hash = 'abba5d9c24cdd2d49cdc2a8aa92976cf20737eff'
25
+ 'fe371d32e50ca682391e1e70ab98c2942aeffb01/u-boot')
26
+ uboot_hash = 'cbb8cbab970f594bf6523b9855be209c08374ae2'
27
uboot_path = self.fetch_asset(uboot_url, asset_hash=uboot_hash)
28
spi_url = ('https://raw.githubusercontent.com/'
29
'Subbaraya-Sundeep/qemu-test-binaries/'
30
- 'fa030bd77a014a0b8e360d3b7011df89283a2f0b/spi.bin')
31
- spi_hash = '85f698329d38de63aea6e884a86fbde70890a78a'
32
+ 'fe371d32e50ca682391e1e70ab98c2942aeffb01/spi.bin')
33
+ spi_hash = '65523a1835949b6f4553be96dec1b6a38fb05501'
34
spi_path = self.fetch_asset(spi_url, asset_hash=spi_hash)
35
36
self.vm.set_console()
37
@@ -XXX,XX +XXX,XX @@ class BootLinuxConsole(Test):
38
'-drive', 'file=' + spi_path + ',if=mtd,format=raw',
39
'-no-reboot')
40
self.vm.launch()
41
- self.wait_for_console_pattern('init started: BusyBox')
42
+ self.wait_for_console_pattern('Enter \'help\' for a list')
43
+
44
+ exec_command_and_wait_for_pattern(self, 'ifconfig eth0 10.0.2.15',
45
+ 'eth0: link becomes ready')
46
+ exec_command_and_wait_for_pattern(self, 'ping -c 3 10.0.2.2',
47
+ '3 packets transmitted, 3 packets received, 0% packet loss')
48
49
def do_test_arm_raspi2(self, uart_id):
50
"""
51
--
52
2.20.1
53
54
diff view generated by jsdifflib
New patch
1
1
This object may be used to represent a clock inside a clock tree.
2
3
A clock may be connected to another clock so that it receives update,
4
through a callback, whenever the source/parent clock is updated.
5
6
Although only the root clock of a clock tree controls the values
7
(represented as periods) of all clocks in tree, each clock holds
8
a local state containing the current value so that it can be fetched
9
independently. It will allows us to fullfill migration requirements
10
by migrating each clock independently of others.
11
12
This is based on the original work of Frederic Konrad.
13
14
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
15
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
16
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
17
Message-id: 20200406135251.157596-2-damien.hedde@greensocs.com
18
[PMM: Use uint64_t rather than unsigned long long in trace events;
19
the dtrace backend can't handle the latter]
20
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
21
---
22
hw/core/Makefile.objs | 1 +
23
include/hw/clock.h | 216 ++++++++++++++++++++++++++++++++++++++++++
24
hw/core/clock.c | 130 +++++++++++++++++++++++++
25
hw/core/trace-events | 7 ++
26
4 files changed, 354 insertions(+)
27
create mode 100644 include/hw/clock.h
28
create mode 100644 hw/core/clock.c
29
30
diff --git a/hw/core/Makefile.objs b/hw/core/Makefile.objs
31
index XXXXXXX..XXXXXXX 100644
32
--- a/hw/core/Makefile.objs
33
+++ b/hw/core/Makefile.objs
34
@@ -XXX,XX +XXX,XX @@ common-obj-y += hotplug.o
35
common-obj-y += vmstate-if.o
36
# irq.o needed for qdev GPIO handling:
37
common-obj-y += irq.o
38
+common-obj-y += clock.o
39
40
common-obj-$(CONFIG_SOFTMMU) += reset.o
41
common-obj-$(CONFIG_SOFTMMU) += qdev-fw.o
42
diff --git a/include/hw/clock.h b/include/hw/clock.h
43
new file mode 100644
44
index XXXXXXX..XXXXXXX
45
--- /dev/null
46
+++ b/include/hw/clock.h
47
@@ -XXX,XX +XXX,XX @@
48
+/*
49
+ * Hardware Clocks
50
+ *
51
+ * Copyright GreenSocs 2016-2020
52
+ *
53
+ * Authors:
54
+ * Frederic Konrad
55
+ * Damien Hedde
56
+ *
57
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
58
+ * See the COPYING file in the top-level directory.
59
+ */
60
+
61
+#ifndef QEMU_HW_CLOCK_H
62
+#define QEMU_HW_CLOCK_H
63
+
64
+#include "qom/object.h"
65
+#include "qemu/queue.h"
66
+
67
+#define TYPE_CLOCK "clock"
68
+#define CLOCK(obj) OBJECT_CHECK(Clock, (obj), TYPE_CLOCK)
69
+
70
+typedef void ClockCallback(void *opaque);
71
+
72
+/*
73
+ * clock store a value representing the clock's period in 2^-32ns unit.
74
+ * It can represent:
75
+ * + periods from 2^-32ns up to 4seconds
76
+ * + frequency from ~0.25Hz 2e10Ghz
77
+ * Resolution of frequency representation decreases with frequency:
78
+ * + at 100MHz, resolution is ~2mHz
79
+ * + at 1Ghz, resolution is ~0.2Hz
80
+ * + at 10Ghz, resolution is ~20Hz
81
+ */
82
+#define CLOCK_SECOND (1000000000llu << 32)
83
+
84
+/*
85
+ * macro helpers to convert to hertz / nanosecond
86
+ */
87
+#define CLOCK_PERIOD_FROM_NS(ns) ((ns) * (CLOCK_SECOND / 1000000000llu))
88
+#define CLOCK_PERIOD_TO_NS(per) ((per) / (CLOCK_SECOND / 1000000000llu))
89
+#define CLOCK_PERIOD_FROM_HZ(hz) (((hz) != 0) ? CLOCK_SECOND / (hz) : 0u)
90
+#define CLOCK_PERIOD_TO_HZ(per) (((per) != 0) ? CLOCK_SECOND / (per) : 0u)
91
+
92
+/**
93
+ * Clock:
94
+ * @parent_obj: parent class
95
+ * @period: unsigned integer representing the period of the clock
96
+ * @canonical_path: clock path string cache (used for trace purpose)
97
+ * @callback: called when clock changes
98
+ * @callback_opaque: argument for @callback
99
+ * @source: source (or parent in clock tree) of the clock
100
+ * @children: list of clocks connected to this one (it is their source)
101
+ * @sibling: structure used to form a clock list
102
+ */
103
+
104
+typedef struct Clock Clock;
105
+
106
+struct Clock {
107
+ /*< private >*/
108
+ Object parent_obj;
109
+
110
+ /* all fields are private and should not be modified directly */
111
+
112
+ /* fields */
113
+ uint64_t period;
114
+ char *canonical_path;
115
+ ClockCallback *callback;
116
+ void *callback_opaque;
117
+
118
+ /* Clocks are organized in a clock tree */
119
+ Clock *source;
120
+ QLIST_HEAD(, Clock) children;
121
+ QLIST_ENTRY(Clock) sibling;
122
+};
123
+
124
+/**
125
+ * clock_setup_canonical_path:
126
+ * @clk: clock
127
+ *
128
+ * compute the canonical path of the clock (used by log messages)
129
+ */
130
+void clock_setup_canonical_path(Clock *clk);
131
+
132
+/**
133
+ * clock_set_callback:
134
+ * @clk: the clock to register the callback into
135
+ * @cb: the callback function
136
+ * @opaque: the argument to the callback
137
+ *
138
+ * Register a callback called on every clock update.
139
+ */
140
+void clock_set_callback(Clock *clk, ClockCallback *cb, void *opaque);
141
+
142
+/**
143
+ * clock_clear_callback:
144
+ * @clk: the clock to delete the callback from
145
+ *
146
+ * Unregister the callback registered with clock_set_callback.
147
+ */
148
+void clock_clear_callback(Clock *clk);
149
+
150
+/**
151
+ * clock_set_source:
152
+ * @clk: the clock.
153
+ * @src: the source clock
154
+ *
155
+ * Setup @src as the clock source of @clk. The current @src period
156
+ * value is also copied to @clk and its subtree but no callback is
157
+ * called.
158
+ * Further @src update will be propagated to @clk and its subtree.
159
+ */
160
+void clock_set_source(Clock *clk, Clock *src);
161
+
162
+/**
163
+ * clock_set:
164
+ * @clk: the clock to initialize.
165
+ * @value: the clock's value, 0 means unclocked
166
+ *
167
+ * Set the local cached period value of @clk to @value.
168
+ */
169
+void clock_set(Clock *clk, uint64_t value);
170
+
171
+static inline void clock_set_hz(Clock *clk, unsigned hz)
172
+{
173
+ clock_set(clk, CLOCK_PERIOD_FROM_HZ(hz));
174
+}
175
+
176
+static inline void clock_set_ns(Clock *clk, unsigned ns)
177
+{
178
+ clock_set(clk, CLOCK_PERIOD_FROM_NS(ns));
179
+}
180
+
181
+/**
182
+ * clock_propagate:
183
+ * @clk: the clock
184
+ *
185
+ * Propagate the clock period that has been previously configured using
186
+ * @clock_set(). This will update recursively all connected clocks.
187
+ * It is an error to call this function on a clock which has a source.
188
+ * Note: this function must not be called during device inititialization
189
+ * or migration.
190
+ */
191
+void clock_propagate(Clock *clk);
192
+
193
+/**
194
+ * clock_update:
195
+ * @clk: the clock to update.
196
+ * @value: the new clock's value, 0 means unclocked
197
+ *
198
+ * Update the @clk to the new @value. All connected clocks will be informed
199
+ * of this update. This is equivalent to call @clock_set() then
200
+ * @clock_propagate().
201
+ */
202
+static inline void clock_update(Clock *clk, uint64_t value)
203
+{
204
+ clock_set(clk, value);
205
+ clock_propagate(clk);
206
+}
207
+
208
+static inline void clock_update_hz(Clock *clk, unsigned hz)
209
+{
210
+ clock_update(clk, CLOCK_PERIOD_FROM_HZ(hz));
211
+}
212
+
213
+static inline void clock_update_ns(Clock *clk, unsigned ns)
214
+{
215
+ clock_update(clk, CLOCK_PERIOD_FROM_NS(ns));
216
+}
217
+
218
+/**
219
+ * clock_get:
220
+ * @clk: the clk to fetch the clock
221
+ *
222
+ * @return: the current period.
223
+ */
224
+static inline uint64_t clock_get(const Clock *clk)
225
+{
226
+ return clk->period;
227
+}
228
+
229
+static inline unsigned clock_get_hz(Clock *clk)
230
+{
231
+ return CLOCK_PERIOD_TO_HZ(clock_get(clk));
232
+}
233
+
234
+static inline unsigned clock_get_ns(Clock *clk)
235
+{
236
+ return CLOCK_PERIOD_TO_NS(clock_get(clk));
237
+}
238
+
239
+/**
240
+ * clock_is_enabled:
241
+ * @clk: a clock
242
+ *
243
+ * @return: true if the clock is running.
244
+ */
245
+static inline bool clock_is_enabled(const Clock *clk)
246
+{
247
+ return clock_get(clk) != 0;
248
+}
249
+
250
+static inline void clock_init(Clock *clk, uint64_t value)
251
+{
252
+ clock_set(clk, value);
253
+}
254
+static inline void clock_init_hz(Clock *clk, uint64_t value)
255
+{
256
+ clock_set_hz(clk, value);
257
+}
258
+static inline void clock_init_ns(Clock *clk, uint64_t value)
259
+{
260
+ clock_set_ns(clk, value);
261
+}
262
+
263
+#endif /* QEMU_HW_CLOCK_H */
264
diff --git a/hw/core/clock.c b/hw/core/clock.c
265
new file mode 100644
266
index XXXXXXX..XXXXXXX
267
--- /dev/null
268
+++ b/hw/core/clock.c
269
@@ -XXX,XX +XXX,XX @@
270
+/*
271
+ * Hardware Clocks
272
+ *
273
+ * Copyright GreenSocs 2016-2020
274
+ *
275
+ * Authors:
276
+ * Frederic Konrad
277
+ * Damien Hedde
278
+ *
279
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
280
+ * See the COPYING file in the top-level directory.
281
+ */
282
+
283
+#include "qemu/osdep.h"
284
+#include "hw/clock.h"
285
+#include "trace.h"
286
+
287
+#define CLOCK_PATH(_clk) (_clk->canonical_path)
288
+
289
+void clock_setup_canonical_path(Clock *clk)
290
+{
291
+ g_free(clk->canonical_path);
292
+ clk->canonical_path = object_get_canonical_path(OBJECT(clk));
293
+}
294
+
295
+void clock_set_callback(Clock *clk, ClockCallback *cb, void *opaque)
296
+{
297
+ clk->callback = cb;
298
+ clk->callback_opaque = opaque;
299
+}
300
+
301
+void clock_clear_callback(Clock *clk)
302
+{
303
+ clock_set_callback(clk, NULL, NULL);
304
+}
305
+
306
+void clock_set(Clock *clk, uint64_t period)
307
+{
308
+ trace_clock_set(CLOCK_PATH(clk), CLOCK_PERIOD_TO_NS(clk->period),
309
+ CLOCK_PERIOD_TO_NS(period));
310
+ clk->period = period;
311
+}
312
+
313
+static void clock_propagate_period(Clock *clk, bool call_callbacks)
314
+{
315
+ Clock *child;
316
+
317
+ QLIST_FOREACH(child, &clk->children, sibling) {
318
+ if (child->period != clk->period) {
319
+ child->period = clk->period;
320
+ trace_clock_update(CLOCK_PATH(child), CLOCK_PATH(clk),
321
+ CLOCK_PERIOD_TO_NS(clk->period),
322
+ call_callbacks);
323
+ if (call_callbacks && child->callback) {
324
+ child->callback(child->callback_opaque);
325
+ }
326
+ clock_propagate_period(child, call_callbacks);
327
+ }
328
+ }
329
+}
330
+
331
+void clock_propagate(Clock *clk)
332
+{
333
+ assert(clk->source == NULL);
334
+ trace_clock_propagate(CLOCK_PATH(clk));
335
+ clock_propagate_period(clk, true);
336
+}
337
+
338
+void clock_set_source(Clock *clk, Clock *src)
339
+{
340
+ /* changing clock source is not supported */
341
+ assert(!clk->source);
342
+
343
+ trace_clock_set_source(CLOCK_PATH(clk), CLOCK_PATH(src));
344
+
345
+ clk->period = src->period;
346
+ QLIST_INSERT_HEAD(&src->children, clk, sibling);
347
+ clk->source = src;
348
+ clock_propagate_period(clk, false);
349
+}
350
+
351
+static void clock_disconnect(Clock *clk)
352
+{
353
+ if (clk->source == NULL) {
354
+ return;
355
+ }
356
+
357
+ trace_clock_disconnect(CLOCK_PATH(clk));
358
+
359
+ clk->source = NULL;
360
+ QLIST_REMOVE(clk, sibling);
361
+}
362
+
363
+static void clock_initfn(Object *obj)
364
+{
365
+ Clock *clk = CLOCK(obj);
366
+
367
+ QLIST_INIT(&clk->children);
368
+}
369
+
370
+static void clock_finalizefn(Object *obj)
371
+{
372
+ Clock *clk = CLOCK(obj);
373
+ Clock *child, *next;
374
+
375
+ /* clear our list of children */
376
+ QLIST_FOREACH_SAFE(child, &clk->children, sibling, next) {
377
+ clock_disconnect(child);
378
+ }
379
+
380
+ /* remove us from source's children list */
381
+ clock_disconnect(clk);
382
+
383
+ g_free(clk->canonical_path);
384
+}
385
+
386
+static const TypeInfo clock_info = {
387
+ .name = TYPE_CLOCK,
388
+ .parent = TYPE_OBJECT,
389
+ .instance_size = sizeof(Clock),
390
+ .instance_init = clock_initfn,
391
+ .instance_finalize = clock_finalizefn,
392
+};
393
+
394
+static void clock_register_types(void)
395
+{
396
+ type_register_static(&clock_info);
397
+}
398
+
399
+type_init(clock_register_types)
400
diff --git a/hw/core/trace-events b/hw/core/trace-events
401
index XXXXXXX..XXXXXXX 100644
402
--- a/hw/core/trace-events
403
+++ b/hw/core/trace-events
404
@@ -XXX,XX +XXX,XX @@ resettable_phase_exit_begin(void *obj, const char *objtype, unsigned count, int
405
resettable_phase_exit_exec(void *obj, const char *objtype, int has_method) "obj=%p(%s) method=%d"
406
resettable_phase_exit_end(void *obj, const char *objtype, unsigned count) "obj=%p(%s) count=%d"
407
resettable_transitional_function(void *obj, const char *objtype) "obj=%p(%s)"
408
+
409
+# clock.c
410
+clock_set_source(const char *clk, const char *src) "'%s', src='%s'"
411
+clock_disconnect(const char *clk) "'%s'"
412
+clock_set(const char *clk, uint64_t old, uint64_t new) "'%s', ns=%"PRIu64"->%"PRIu64
413
+clock_propagate(const char *clk) "'%s'"
414
+clock_update(const char *clk, const char *src, uint64_t val, int cb) "'%s', src='%s', ns=%"PRIu64", cb=%d"
415
--
416
2.20.1
417
418
diff view generated by jsdifflib
1
From: Joel Stanley <joel@jms.id.au>
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
2
3
This adds the base for a machine model of the BBC micro:bit:
3
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
4
5
https://en.wikipedia.org/wiki/Micro_Bit
6
7
This is a system with a nRF51 SoC containing the main processor, with
8
various peripherals on board.
9
10
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
11
Signed-off-by: Joel Stanley <joel@jms.id.au>
12
Message-id: 20180831220920.27113-4-joel@jms.id.au
13
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
4
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
5
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
8
Message-id: 20200406135251.157596-3-damien.hedde@greensocs.com
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
10
---
16
hw/arm/Makefile.objs | 2 +-
11
hw/core/Makefile.objs | 1 +
17
hw/arm/microbit.c | 67 ++++++++++++++++++++++++++++++++++++++++++++
12
include/hw/clock.h | 9 +++++++++
18
2 files changed, 68 insertions(+), 1 deletion(-)
13
hw/core/clock-vmstate.c | 25 +++++++++++++++++++++++++
19
create mode 100644 hw/arm/microbit.c
14
3 files changed, 35 insertions(+)
15
create mode 100644 hw/core/clock-vmstate.c
20
16
21
diff --git a/hw/arm/Makefile.objs b/hw/arm/Makefile.objs
17
diff --git a/hw/core/Makefile.objs b/hw/core/Makefile.objs
22
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX 100644
23
--- a/hw/arm/Makefile.objs
19
--- a/hw/core/Makefile.objs
24
+++ b/hw/arm/Makefile.objs
20
+++ b/hw/core/Makefile.objs
25
@@ -XXX,XX +XXX,XX @@ obj-$(CONFIG_IOTKIT) += iotkit.o
21
@@ -XXX,XX +XXX,XX @@ common-obj-$(CONFIG_SOFTMMU) += null-machine.o
26
obj-$(CONFIG_FSL_IMX7) += fsl-imx7.o mcimx7d-sabre.o
22
common-obj-$(CONFIG_SOFTMMU) += loader.o
27
obj-$(CONFIG_ARM_SMMUV3) += smmu-common.o smmuv3.o
23
common-obj-$(CONFIG_SOFTMMU) += machine-hmp-cmds.o
28
obj-$(CONFIG_FSL_IMX6UL) += fsl-imx6ul.o mcimx6ul-evk.o
24
common-obj-$(CONFIG_SOFTMMU) += numa.o
29
-obj-$(CONFIG_NRF51_SOC) += nrf51_soc.o
25
+common-obj-$(CONFIG_SOFTMMU) += clock-vmstate.o
30
+obj-$(CONFIG_NRF51_SOC) += nrf51_soc.o microbit.o
26
obj-$(CONFIG_SOFTMMU) += machine-qmp-cmds.o
31
diff --git a/hw/arm/microbit.c b/hw/arm/microbit.c
27
28
common-obj-$(CONFIG_EMPTY_SLOT) += empty_slot.o
29
diff --git a/include/hw/clock.h b/include/hw/clock.h
30
index XXXXXXX..XXXXXXX 100644
31
--- a/include/hw/clock.h
32
+++ b/include/hw/clock.h
33
@@ -XXX,XX +XXX,XX @@ struct Clock {
34
QLIST_ENTRY(Clock) sibling;
35
};
36
37
+/*
38
+ * vmstate description entry to be added in device vmsd.
39
+ */
40
+extern const VMStateDescription vmstate_clock;
41
+#define VMSTATE_CLOCK(field, state) \
42
+ VMSTATE_CLOCK_V(field, state, 0)
43
+#define VMSTATE_CLOCK_V(field, state, version) \
44
+ VMSTATE_STRUCT_POINTER_V(field, state, version, vmstate_clock, Clock)
45
+
46
/**
47
* clock_setup_canonical_path:
48
* @clk: clock
49
diff --git a/hw/core/clock-vmstate.c b/hw/core/clock-vmstate.c
32
new file mode 100644
50
new file mode 100644
33
index XXXXXXX..XXXXXXX
51
index XXXXXXX..XXXXXXX
34
--- /dev/null
52
--- /dev/null
35
+++ b/hw/arm/microbit.c
53
+++ b/hw/core/clock-vmstate.c
36
@@ -XXX,XX +XXX,XX @@
54
@@ -XXX,XX +XXX,XX @@
37
+/*
55
+/*
38
+ * BBC micro:bit machine
56
+ * Clock migration structure
39
+ * http://tech.microbit.org/hardware/
40
+ *
57
+ *
41
+ * Copyright 2018 Joel Stanley <joel@jms.id.au>
58
+ * Copyright GreenSocs 2019-2020
42
+ *
59
+ *
43
+ * This code is licensed under the GPL version 2 or later. See
60
+ * Authors:
44
+ * the COPYING file in the top-level directory.
61
+ * Damien Hedde
62
+ *
63
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
64
+ * See the COPYING file in the top-level directory.
45
+ */
65
+ */
46
+
66
+
47
+#include "qemu/osdep.h"
67
+#include "qemu/osdep.h"
48
+#include "qapi/error.h"
68
+#include "migration/vmstate.h"
49
+#include "hw/boards.h"
69
+#include "hw/clock.h"
50
+#include "hw/arm/arm.h"
51
+#include "exec/address-spaces.h"
52
+
70
+
53
+#include "hw/arm/nrf51_soc.h"
71
+const VMStateDescription vmstate_clock = {
54
+
72
+ .name = "clock",
55
+typedef struct {
73
+ .version_id = 0,
56
+ MachineState parent;
74
+ .minimum_version_id = 0,
57
+
75
+ .fields = (VMStateField[]) {
58
+ NRF51State nrf51;
76
+ VMSTATE_UINT64(period, Clock),
59
+} MicrobitMachineState;
77
+ VMSTATE_END_OF_LIST()
60
+
78
+ }
61
+#define TYPE_MICROBIT_MACHINE MACHINE_TYPE_NAME("microbit")
62
+
63
+#define MICROBIT_MACHINE(obj) \
64
+ OBJECT_CHECK(MicrobitMachineState, obj, TYPE_MICROBIT_MACHINE)
65
+
66
+static void microbit_init(MachineState *machine)
67
+{
68
+ MicrobitMachineState *s = MICROBIT_MACHINE(machine);
69
+ MemoryRegion *system_memory = get_system_memory();
70
+ Object *soc = OBJECT(&s->nrf51);
71
+
72
+ sysbus_init_child_obj(OBJECT(machine), "nrf51", soc, sizeof(s->nrf51),
73
+ TYPE_NRF51_SOC);
74
+ object_property_set_link(soc, OBJECT(system_memory), "memory",
75
+ &error_fatal);
76
+ object_property_set_bool(soc, true, "realized", &error_fatal);
77
+
78
+ armv7m_load_kernel(ARM_CPU(first_cpu), machine->kernel_filename,
79
+ NRF51_SOC(soc)->flash_size);
80
+}
81
+
82
+static void microbit_machine_class_init(ObjectClass *oc, void *data)
83
+{
84
+ MachineClass *mc = MACHINE_CLASS(oc);
85
+
86
+ mc->desc = "BBC micro:bit";
87
+ mc->init = microbit_init;
88
+ mc->max_cpus = 1;
89
+}
90
+
91
+static const TypeInfo microbit_info = {
92
+ .name = TYPE_MICROBIT_MACHINE,
93
+ .parent = TYPE_MACHINE,
94
+ .instance_size = sizeof(MicrobitMachineState),
95
+ .class_init = microbit_machine_class_init,
96
+};
79
+};
97
+
98
+static void microbit_machine_init(void)
99
+{
100
+ type_register_static(&microbit_info);
101
+}
102
+
103
+type_init(microbit_machine_init);
104
--
80
--
105
2.19.0
81
2.20.1
106
82
107
83
diff view generated by jsdifflib
New patch
1
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
3
Add functions to easily handle clocks with devices.
4
Clock inputs and outputs should be used to handle clock propagation
5
between devices.
6
The API is very similar the GPIO API.
7
8
This is based on the original work of Frederic Konrad.
9
10
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
11
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
12
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
13
Message-id: 20200406135251.157596-4-damien.hedde@greensocs.com
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
16
hw/core/Makefile.objs | 2 +-
17
tests/Makefile.include | 1 +
18
include/hw/qdev-clock.h | 104 +++++++++++++++++++++++++
19
include/hw/qdev-core.h | 12 +++
20
hw/core/qdev-clock.c | 168 ++++++++++++++++++++++++++++++++++++++++
21
hw/core/qdev.c | 12 +++
22
6 files changed, 298 insertions(+), 1 deletion(-)
23
create mode 100644 include/hw/qdev-clock.h
24
create mode 100644 hw/core/qdev-clock.c
25
26
diff --git a/hw/core/Makefile.objs b/hw/core/Makefile.objs
27
index XXXXXXX..XXXXXXX 100644
28
--- a/hw/core/Makefile.objs
29
+++ b/hw/core/Makefile.objs
30
@@ -XXX,XX +XXX,XX @@ common-obj-y += hotplug.o
31
common-obj-y += vmstate-if.o
32
# irq.o needed for qdev GPIO handling:
33
common-obj-y += irq.o
34
-common-obj-y += clock.o
35
+common-obj-y += clock.o qdev-clock.o
36
37
common-obj-$(CONFIG_SOFTMMU) += reset.o
38
common-obj-$(CONFIG_SOFTMMU) += qdev-fw.o
39
diff --git a/tests/Makefile.include b/tests/Makefile.include
40
index XXXXXXX..XXXXXXX 100644
41
--- a/tests/Makefile.include
42
+++ b/tests/Makefile.include
43
@@ -XXX,XX +XXX,XX @@ tests/test-qdev-global-props$(EXESUF): tests/test-qdev-global-props.o \
44
    hw/core/fw-path-provider.o \
45
    hw/core/reset.o \
46
    hw/core/vmstate-if.o \
47
+    hw/core/clock.o hw/core/qdev-clock.o \
48
    $(test-qapi-obj-y)
49
tests/test-vmstate$(EXESUF): tests/test-vmstate.o \
50
    migration/vmstate.o migration/vmstate-types.o migration/qemu-file.o \
51
diff --git a/include/hw/qdev-clock.h b/include/hw/qdev-clock.h
52
new file mode 100644
53
index XXXXXXX..XXXXXXX
54
--- /dev/null
55
+++ b/include/hw/qdev-clock.h
56
@@ -XXX,XX +XXX,XX @@
57
+/*
58
+ * Device's clock input and output
59
+ *
60
+ * Copyright GreenSocs 2016-2020
61
+ *
62
+ * Authors:
63
+ * Frederic Konrad
64
+ * Damien Hedde
65
+ *
66
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
67
+ * See the COPYING file in the top-level directory.
68
+ */
69
+
70
+#ifndef QDEV_CLOCK_H
71
+#define QDEV_CLOCK_H
72
+
73
+#include "hw/clock.h"
74
+
75
+/**
76
+ * qdev_init_clock_in:
77
+ * @dev: the device to add an input clock to
78
+ * @name: the name of the clock (can't be NULL).
79
+ * @callback: optional callback to be called on update or NULL.
80
+ * @opaque: argument for the callback
81
+ * @returns: a pointer to the newly added clock
82
+ *
83
+ * Add an input clock to device @dev as a clock named @name.
84
+ * This adds a child<> property.
85
+ * The callback will be called with @opaque as opaque parameter.
86
+ */
87
+Clock *qdev_init_clock_in(DeviceState *dev, const char *name,
88
+ ClockCallback *callback, void *opaque);
89
+
90
+/**
91
+ * qdev_init_clock_out:
92
+ * @dev: the device to add an output clock to
93
+ * @name: the name of the clock (can't be NULL).
94
+ * @returns: a pointer to the newly added clock
95
+ *
96
+ * Add an output clock to device @dev as a clock named @name.
97
+ * This adds a child<> property.
98
+ */
99
+Clock *qdev_init_clock_out(DeviceState *dev, const char *name);
100
+
101
+/**
102
+ * qdev_get_clock_in:
103
+ * @dev: the device which has the clock
104
+ * @name: the name of the clock (can't be NULL).
105
+ * @returns: a pointer to the clock
106
+ *
107
+ * Get the input clock @name from @dev or NULL if does not exist.
108
+ */
109
+Clock *qdev_get_clock_in(DeviceState *dev, const char *name);
110
+
111
+/**
112
+ * qdev_get_clock_out:
113
+ * @dev: the device which has the clock
114
+ * @name: the name of the clock (can't be NULL).
115
+ * @returns: a pointer to the clock
116
+ *
117
+ * Get the output clock @name from @dev or NULL if does not exist.
118
+ */
119
+Clock *qdev_get_clock_out(DeviceState *dev, const char *name);
120
+
121
+/**
122
+ * qdev_connect_clock_in:
123
+ * @dev: a device
124
+ * @name: the name of an input clock in @dev
125
+ * @source: the source clock (an output clock of another device for example)
126
+ *
127
+ * Set the source clock of input clock @name of device @dev to @source.
128
+ * @source period update will be propagated to @name clock.
129
+ */
130
+static inline void qdev_connect_clock_in(DeviceState *dev, const char *name,
131
+ Clock *source)
132
+{
133
+ clock_set_source(qdev_get_clock_in(dev, name), source);
134
+}
135
+
136
+/**
137
+ * qdev_alias_clock:
138
+ * @dev: the device which has the clock
139
+ * @name: the name of the clock in @dev (can't be NULL)
140
+ * @alias_dev: the device to add the clock
141
+ * @alias_name: the name of the clock in @container
142
+ * @returns: a pointer to the clock
143
+ *
144
+ * Add a clock @alias_name in @alias_dev which is an alias of the clock @name
145
+ * in @dev. The direction _in_ or _out_ will the same as the original.
146
+ * An alias clock must not be modified or used by @alias_dev and should
147
+ * typically be only only for device composition purpose.
148
+ */
149
+Clock *qdev_alias_clock(DeviceState *dev, const char *name,
150
+ DeviceState *alias_dev, const char *alias_name);
151
+
152
+/**
153
+ * qdev_finalize_clocklist:
154
+ * @dev: the device being finalized
155
+ *
156
+ * Clear the clocklist from @dev. Only used internally in qdev.
157
+ */
158
+void qdev_finalize_clocklist(DeviceState *dev);
159
+
160
+#endif /* QDEV_CLOCK_H */
161
diff --git a/include/hw/qdev-core.h b/include/hw/qdev-core.h
162
index XXXXXXX..XXXXXXX 100644
163
--- a/include/hw/qdev-core.h
164
+++ b/include/hw/qdev-core.h
165
@@ -XXX,XX +XXX,XX @@ struct NamedGPIOList {
166
QLIST_ENTRY(NamedGPIOList) node;
167
};
168
169
+typedef struct Clock Clock;
170
+typedef struct NamedClockList NamedClockList;
171
+
172
+struct NamedClockList {
173
+ char *name;
174
+ Clock *clock;
175
+ bool output;
176
+ bool alias;
177
+ QLIST_ENTRY(NamedClockList) node;
178
+};
179
+
180
/**
181
* DeviceState:
182
* @realized: Indicates whether the device has been fully constructed.
183
@@ -XXX,XX +XXX,XX @@ struct DeviceState {
184
bool allow_unplug_during_migration;
185
BusState *parent_bus;
186
QLIST_HEAD(, NamedGPIOList) gpios;
187
+ QLIST_HEAD(, NamedClockList) clocks;
188
QLIST_HEAD(, BusState) child_bus;
189
int num_child_bus;
190
int instance_id_alias;
191
diff --git a/hw/core/qdev-clock.c b/hw/core/qdev-clock.c
192
new file mode 100644
193
index XXXXXXX..XXXXXXX
194
--- /dev/null
195
+++ b/hw/core/qdev-clock.c
196
@@ -XXX,XX +XXX,XX @@
197
+/*
198
+ * Device's clock input and output
199
+ *
200
+ * Copyright GreenSocs 2016-2020
201
+ *
202
+ * Authors:
203
+ * Frederic Konrad
204
+ * Damien Hedde
205
+ *
206
+ * This work is licensed under the terms of the GNU GPL, version 2 or later.
207
+ * See the COPYING file in the top-level directory.
208
+ */
209
+
210
+#include "qemu/osdep.h"
211
+#include "hw/qdev-clock.h"
212
+#include "hw/qdev-core.h"
213
+#include "qapi/error.h"
214
+
215
+/*
216
+ * qdev_init_clocklist:
217
+ * Add a new clock in a device
218
+ */
219
+static NamedClockList *qdev_init_clocklist(DeviceState *dev, const char *name,
220
+ bool output, Clock *clk)
221
+{
222
+ NamedClockList *ncl;
223
+
224
+ /*
225
+ * Clock must be added before realize() so that we can compute the
226
+ * clock's canonical path during device_realize().
227
+ */
228
+ assert(!dev->realized);
229
+
230
+ /*
231
+ * The ncl structure is freed by qdev_finalize_clocklist() which will
232
+ * be called during @dev's device_finalize().
233
+ */
234
+ ncl = g_new0(NamedClockList, 1);
235
+ ncl->name = g_strdup(name);
236
+ ncl->output = output;
237
+ ncl->alias = (clk != NULL);
238
+
239
+ /*
240
+ * Trying to create a clock whose name clashes with some other
241
+ * clock or property is a bug in the caller and we will abort().
242
+ */
243
+ if (clk == NULL) {
244
+ clk = CLOCK(object_new(TYPE_CLOCK));
245
+ object_property_add_child(OBJECT(dev), name, OBJECT(clk), &error_abort);
246
+ if (output) {
247
+ /*
248
+ * Remove object_new()'s initial reference.
249
+ * Note that for inputs, the reference created by object_new()
250
+ * will be deleted in qdev_finalize_clocklist().
251
+ */
252
+ object_unref(OBJECT(clk));
253
+ }
254
+ } else {
255
+ object_property_add_link(OBJECT(dev), name,
256
+ object_get_typename(OBJECT(clk)),
257
+ (Object **) &ncl->clock,
258
+ NULL, OBJ_PROP_LINK_STRONG, &error_abort);
259
+ }
260
+
261
+ ncl->clock = clk;
262
+
263
+ QLIST_INSERT_HEAD(&dev->clocks, ncl, node);
264
+ return ncl;
265
+}
266
+
267
+void qdev_finalize_clocklist(DeviceState *dev)
268
+{
269
+ /* called by @dev's device_finalize() */
270
+ NamedClockList *ncl, *ncl_next;
271
+
272
+ QLIST_FOREACH_SAFE(ncl, &dev->clocks, node, ncl_next) {
273
+ QLIST_REMOVE(ncl, node);
274
+ if (!ncl->output && !ncl->alias) {
275
+ /*
276
+ * We kept a reference on the input clock to ensure it lives up to
277
+ * this point so we can safely remove the callback.
278
+ * It avoids having a callback to a deleted object if ncl->clock
279
+ * is still referenced somewhere else (eg: by a clock output).
280
+ */
281
+ clock_clear_callback(ncl->clock);
282
+ object_unref(OBJECT(ncl->clock));
283
+ }
284
+ g_free(ncl->name);
285
+ g_free(ncl);
286
+ }
287
+}
288
+
289
+Clock *qdev_init_clock_out(DeviceState *dev, const char *name)
290
+{
291
+ NamedClockList *ncl;
292
+
293
+ assert(name);
294
+
295
+ ncl = qdev_init_clocklist(dev, name, true, NULL);
296
+
297
+ return ncl->clock;
298
+}
299
+
300
+Clock *qdev_init_clock_in(DeviceState *dev, const char *name,
301
+ ClockCallback *callback, void *opaque)
302
+{
303
+ NamedClockList *ncl;
304
+
305
+ assert(name);
306
+
307
+ ncl = qdev_init_clocklist(dev, name, false, NULL);
308
+
309
+ if (callback) {
310
+ clock_set_callback(ncl->clock, callback, opaque);
311
+ }
312
+ return ncl->clock;
313
+}
314
+
315
+static NamedClockList *qdev_get_clocklist(DeviceState *dev, const char *name)
316
+{
317
+ NamedClockList *ncl;
318
+
319
+ QLIST_FOREACH(ncl, &dev->clocks, node) {
320
+ if (strcmp(name, ncl->name) == 0) {
321
+ return ncl;
322
+ }
323
+ }
324
+
325
+ return NULL;
326
+}
327
+
328
+Clock *qdev_get_clock_in(DeviceState *dev, const char *name)
329
+{
330
+ NamedClockList *ncl;
331
+
332
+ assert(name);
333
+
334
+ ncl = qdev_get_clocklist(dev, name);
335
+ assert(!ncl->output);
336
+
337
+ return ncl->clock;
338
+}
339
+
340
+Clock *qdev_get_clock_out(DeviceState *dev, const char *name)
341
+{
342
+ NamedClockList *ncl;
343
+
344
+ assert(name);
345
+
346
+ ncl = qdev_get_clocklist(dev, name);
347
+ assert(ncl->output);
348
+
349
+ return ncl->clock;
350
+}
351
+
352
+Clock *qdev_alias_clock(DeviceState *dev, const char *name,
353
+ DeviceState *alias_dev, const char *alias_name)
354
+{
355
+ NamedClockList *ncl;
356
+
357
+ assert(name && alias_name);
358
+
359
+ ncl = qdev_get_clocklist(dev, name);
360
+
361
+ qdev_init_clocklist(alias_dev, alias_name, ncl->output, ncl->clock);
362
+
363
+ return ncl->clock;
364
+}
365
diff --git a/hw/core/qdev.c b/hw/core/qdev.c
366
index XXXXXXX..XXXXXXX 100644
367
--- a/hw/core/qdev.c
368
+++ b/hw/core/qdev.c
369
@@ -XXX,XX +XXX,XX @@
370
#include "hw/qdev-properties.h"
371
#include "hw/boards.h"
372
#include "hw/sysbus.h"
373
+#include "hw/qdev-clock.h"
374
#include "migration/vmstate.h"
375
#include "trace.h"
376
377
@@ -XXX,XX +XXX,XX @@ static void device_set_realized(Object *obj, bool value, Error **errp)
378
DeviceClass *dc = DEVICE_GET_CLASS(dev);
379
HotplugHandler *hotplug_ctrl;
380
BusState *bus;
381
+ NamedClockList *ncl;
382
Error *local_err = NULL;
383
bool unattached_parent = false;
384
static int unattached_count;
385
@@ -XXX,XX +XXX,XX @@ static void device_set_realized(Object *obj, bool value, Error **errp)
386
*/
387
g_free(dev->canonical_path);
388
dev->canonical_path = object_get_canonical_path(OBJECT(dev));
389
+ QLIST_FOREACH(ncl, &dev->clocks, node) {
390
+ if (ncl->alias) {
391
+ continue;
392
+ } else {
393
+ clock_setup_canonical_path(ncl->clock);
394
+ }
395
+ }
396
397
if (qdev_get_vmsd(dev)) {
398
if (vmstate_register_with_alias_id(VMSTATE_IF(dev),
399
@@ -XXX,XX +XXX,XX @@ static void device_initfn(Object *obj)
400
dev->allow_unplug_during_migration = false;
401
402
QLIST_INIT(&dev->gpios);
403
+ QLIST_INIT(&dev->clocks);
404
}
405
406
static void device_post_init(Object *obj)
407
@@ -XXX,XX +XXX,XX @@ static void device_finalize(Object *obj)
408
*/
409
}
410
411
+ qdev_finalize_clocklist(dev);
412
+
413
/* Only send event if the device had been completely realized */
414
if (dev->pending_deleted_event) {
415
g_assert(dev->canonical_path);
416
--
417
2.20.1
418
419
diff view generated by jsdifflib
New patch
1
From: Damien Hedde <damien.hedde@greensocs.com>
1
2
3
Introduce a function and macro helpers to setup several clocks
4
in a device from a static array description.
5
6
An element of the array describes the clock (name and direction) as
7
well as the related callback and an optional offset to store the
8
created object pointer in the device state structure.
9
10
The array must be terminated by a special element QDEV_CLOCK_END.
11
12
This is based on the original work of Frederic Konrad.
13
14
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
15
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
16
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
17
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
18
Message-id: 20200406135251.157596-5-damien.hedde@greensocs.com
19
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
20
---
21
include/hw/qdev-clock.h | 55 +++++++++++++++++++++++++++++++++++++++++
22
hw/core/qdev-clock.c | 17 +++++++++++++
23
2 files changed, 72 insertions(+)
24
25
diff --git a/include/hw/qdev-clock.h b/include/hw/qdev-clock.h
26
index XXXXXXX..XXXXXXX 100644
27
--- a/include/hw/qdev-clock.h
28
+++ b/include/hw/qdev-clock.h
29
@@ -XXX,XX +XXX,XX @@ Clock *qdev_alias_clock(DeviceState *dev, const char *name,
30
*/
31
void qdev_finalize_clocklist(DeviceState *dev);
32
33
+/**
34
+ * ClockPortInitElem:
35
+ * @name: name of the clock (can't be NULL)
36
+ * @output: indicates whether the clock is input or output
37
+ * @callback: for inputs, optional callback to be called on clock's update
38
+ * with device as opaque
39
+ * @offset: optional offset to store the ClockIn or ClockOut pointer in device
40
+ * state structure (0 means unused)
41
+ */
42
+struct ClockPortInitElem {
43
+ const char *name;
44
+ bool is_output;
45
+ ClockCallback *callback;
46
+ size_t offset;
47
+};
48
+
49
+#define clock_offset_value(devstate, field) \
50
+ (offsetof(devstate, field) + \
51
+ type_check(Clock *, typeof_field(devstate, field)))
52
+
53
+#define QDEV_CLOCK(out_not_in, devstate, field, cb) { \
54
+ .name = (stringify(field)), \
55
+ .is_output = out_not_in, \
56
+ .callback = cb, \
57
+ .offset = clock_offset_value(devstate, field), \
58
+}
59
+
60
+/**
61
+ * QDEV_CLOCK_(IN|OUT):
62
+ * @devstate: structure type. @dev argument of qdev_init_clocks below must be
63
+ * a pointer to that same type.
64
+ * @field: a field in @_devstate (must be Clock*)
65
+ * @callback: (for input only) callback (or NULL) to be called with the device
66
+ * state as argument
67
+ *
68
+ * The name of the clock will be derived from @field
69
+ */
70
+#define QDEV_CLOCK_IN(devstate, field, callback) \
71
+ QDEV_CLOCK(false, devstate, field, callback)
72
+
73
+#define QDEV_CLOCK_OUT(devstate, field) \
74
+ QDEV_CLOCK(true, devstate, field, NULL)
75
+
76
+#define QDEV_CLOCK_END { .name = NULL }
77
+
78
+typedef struct ClockPortInitElem ClockPortInitArray[];
79
+
80
+/**
81
+ * qdev_init_clocks:
82
+ * @dev: the device to add clocks to
83
+ * @clocks: a QDEV_CLOCK_END-terminated array which contains the
84
+ * clocks information.
85
+ */
86
+void qdev_init_clocks(DeviceState *dev, const ClockPortInitArray clocks);
87
+
88
#endif /* QDEV_CLOCK_H */
89
diff --git a/hw/core/qdev-clock.c b/hw/core/qdev-clock.c
90
index XXXXXXX..XXXXXXX 100644
91
--- a/hw/core/qdev-clock.c
92
+++ b/hw/core/qdev-clock.c
93
@@ -XXX,XX +XXX,XX @@ Clock *qdev_init_clock_in(DeviceState *dev, const char *name,
94
return ncl->clock;
95
}
96
97
+void qdev_init_clocks(DeviceState *dev, const ClockPortInitArray clocks)
98
+{
99
+ const struct ClockPortInitElem *elem;
100
+
101
+ for (elem = &clocks[0]; elem->name != NULL; elem++) {
102
+ Clock **clkp;
103
+ /* offset cannot be inside the DeviceState part */
104
+ assert(elem->offset > sizeof(DeviceState));
105
+ clkp = (Clock **)(((void *) dev) + elem->offset);
106
+ if (elem->is_output) {
107
+ *clkp = qdev_init_clock_out(dev, elem->name);
108
+ } else {
109
+ *clkp = qdev_init_clock_in(dev, elem->name, elem->callback, dev);
110
+ }
111
+ }
112
+}
113
+
114
static NamedClockList *qdev_get_clocklist(DeviceState *dev, const char *name)
115
{
116
NamedClockList *ncl;
117
--
118
2.20.1
119
120
diff view generated by jsdifflib
New patch
1
Add the documentation about the clock inputs and outputs in devices.
1
2
3
This is based on the original work of Frederic Konrad.
4
5
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
8
Message-id: 20200406135251.157596-6-damien.hedde@greensocs.com
9
[PMM: Editing pass for minor grammar, style and Sphinx
10
formatting fixes]
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
14
docs/devel/clocks.rst | 391 ++++++++++++++++++++++++++++++++++++++++++
15
docs/devel/index.rst | 1 +
16
2 files changed, 392 insertions(+)
17
create mode 100644 docs/devel/clocks.rst
18
19
diff --git a/docs/devel/clocks.rst b/docs/devel/clocks.rst
20
new file mode 100644
21
index XXXXXXX..XXXXXXX
22
--- /dev/null
23
+++ b/docs/devel/clocks.rst
24
@@ -XXX,XX +XXX,XX @@
25
+Modelling a clock tree in QEMU
26
+==============================
27
+
28
+What are clocks?
29
+----------------
30
+
31
+Clocks are QOM objects developed for the purpose of modelling the
32
+distribution of clocks in QEMU.
33
+
34
+They allow us to model the clock distribution of a platform and detect
35
+configuration errors in the clock tree such as badly configured PLL, clock
36
+source selection or disabled clock.
37
+
38
+The object is *Clock* and its QOM name is ``clock`` (in C code, the macro
39
+``TYPE_CLOCK``).
40
+
41
+Clocks are typically used with devices where they are used to model inputs
42
+and outputs. They are created in a similar way to GPIOs. Inputs and outputs
43
+of different devices can be connected together.
44
+
45
+In these cases a Clock object is a child of a Device object, but this
46
+is not a requirement. Clocks can be independent of devices. For
47
+example it is possible to create a clock outside of any device to
48
+model the main clock source of a machine.
49
+
50
+Here is an example of clocks::
51
+
52
+ +---------+ +----------------------+ +--------------+
53
+ | Clock 1 | | Device B | | Device C |
54
+ | | | +-------+ +-------+ | | +-------+ |
55
+ | |>>-+-->>|Clock 2| |Clock 3|>>--->>|Clock 6| |
56
+ +---------+ | | | (in) | | (out) | | | | (in) | |
57
+ | | +-------+ +-------+ | | +-------+ |
58
+ | | +-------+ | +--------------+
59
+ | | |Clock 4|>>
60
+ | | | (out) | | +--------------+
61
+ | | +-------+ | | Device D |
62
+ | | +-------+ | | +-------+ |
63
+ | | |Clock 5|>>--->>|Clock 7| |
64
+ | | | (out) | | | | (in) | |
65
+ | | +-------+ | | +-------+ |
66
+ | +----------------------+ | |
67
+ | | +-------+ |
68
+ +----------------------------->>|Clock 8| |
69
+ | | (in) | |
70
+ | +-------+ |
71
+ +--------------+
72
+
73
+Clocks are defined in the ``include/hw/clock.h`` header and device
74
+related functions are defined in the ``include/hw/qdev-clock.h``
75
+header.
76
+
77
+The clock state
78
+---------------
79
+
80
+The state of a clock is its period; it is stored as an integer
81
+representing it in units of 2 :sup:`-32` ns. The special value of 0 is used to
82
+represent the clock being inactive or gated. The clocks do not model
83
+the signal itself (pin toggling) or other properties such as the duty
84
+cycle.
85
+
86
+All clocks contain this state: outputs as well as inputs. This allows
87
+the current period of a clock to be fetched at any time. When a clock
88
+is updated, the value is immediately propagated to all connected
89
+clocks in the tree.
90
+
91
+To ease interaction with clocks, helpers with a unit suffix are defined for
92
+every clock state setter or getter. The suffixes are:
93
+
94
+- ``_ns`` for handling periods in nanoseconds
95
+- ``_hz`` for handling frequencies in hertz
96
+
97
+The 0 period value is converted to 0 in hertz and vice versa. 0 always means
98
+that the clock is disabled.
99
+
100
+Adding a new clock
101
+------------------
102
+
103
+Adding clocks to a device must be done during the init method of the Device
104
+instance.
105
+
106
+To add an input clock to a device, the function ``qdev_init_clock_in()``
107
+must be used. It takes the name, a callback and an opaque parameter
108
+for the callback (this will be explained in a following section).
109
+Output is simpler; only the name is required. Typically::
110
+
111
+ qdev_init_clock_in(DEVICE(dev), "clk_in", clk_in_callback, dev);
112
+ qdev_init_clock_out(DEVICE(dev), "clk_out");
113
+
114
+Both functions return the created Clock pointer, which should be saved in the
115
+device's state structure for further use.
116
+
117
+These objects will be automatically deleted by the QOM reference mechanism.
118
+
119
+Note that it is possible to create a static array describing clock inputs and
120
+outputs. The function ``qdev_init_clocks()`` must be called with the array as
121
+parameter to initialize the clocks: it has the same behaviour as calling the
122
+``qdev_init_clock_in/out()`` for each clock in the array. To ease the array
123
+construction, some macros are defined in ``include/hw/qdev-clock.h``.
124
+As an example, the following creates 2 clocks to a device: one input and one
125
+output.
126
+
127
+.. code-block:: c
128
+
129
+ /* device structure containing pointers to the clock objects */
130
+ typedef struct MyDeviceState {
131
+ DeviceState parent_obj;
132
+ Clock *clk_in;
133
+ Clock *clk_out;
134
+ } MyDeviceState;
135
+
136
+ /*
137
+ * callback for the input clock (see "Callback on input clock
138
+ * change" section below for more information).
139
+ */
140
+ static void clk_in_callback(void *opaque);
141
+
142
+ /*
143
+ * static array describing clocks:
144
+ * + a clock input named "clk_in", whose pointer is stored in
145
+ * the clk_in field of a MyDeviceState structure with callback
146
+ * clk_in_callback.
147
+ * + a clock output named "clk_out" whose pointer is stored in
148
+ * the clk_out field of a MyDeviceState structure.
149
+ */
150
+ static const ClockPortInitArray mydev_clocks = {
151
+ QDEV_CLOCK_IN(MyDeviceState, clk_in, clk_in_callback),
152
+ QDEV_CLOCK_OUT(MyDeviceState, clk_out),
153
+ QDEV_CLOCK_END
154
+ };
155
+
156
+ /* device initialization function */
157
+ static void mydev_init(Object *obj)
158
+ {
159
+ /* cast to MyDeviceState */
160
+ MyDeviceState *mydev = MYDEVICE(obj);
161
+ /* create and fill the pointer fields in the MyDeviceState */
162
+ qdev_init_clocks(mydev, mydev_clocks);
163
+ [...]
164
+ }
165
+
166
+An alternative way to create a clock is to simply call
167
+``object_new(TYPE_CLOCK)``. In that case the clock will neither be an
168
+input nor an output of a device. After the whole QOM hierarchy of the
169
+clock has been set ``clock_setup_canonical_path()`` should be called.
170
+
171
+At creation, the period of the clock is 0: the clock is disabled. You can
172
+change it using ``clock_set_ns()`` or ``clock_set_hz()``.
173
+
174
+Note that if you are creating a clock with a fixed period which will never
175
+change (for example the main clock source of a board), then you'll have
176
+nothing else to do. This value will be propagated to other clocks when
177
+connecting the clocks together and devices will fetch the right value during
178
+the first reset.
179
+
180
+Retrieving clocks from a device
181
+-------------------------------
182
+
183
+``qdev_get_clock_in()`` and ``dev_get_clock_out()`` are available to
184
+get the clock inputs or outputs of a device. For example:
185
+
186
+.. code-block:: c
187
+
188
+ Clock *clk = qdev_get_clock_in(DEVICE(mydev), "clk_in");
189
+
190
+or:
191
+
192
+.. code-block:: c
193
+
194
+ Clock *clk = qdev_get_clock_out(DEVICE(mydev), "clk_out");
195
+
196
+Connecting two clocks together
197
+------------------------------
198
+
199
+To connect two clocks together, use the ``clock_set_source()`` function.
200
+Given two clocks ``clk1``, and ``clk2``, ``clock_set_source(clk2, clk1);``
201
+configures ``clk2`` to follow the ``clk1`` period changes. Every time ``clk1``
202
+is updated, ``clk2`` will be updated too.
203
+
204
+When connecting clock between devices, prefer using the
205
+``qdev_connect_clock_in()`` function to set the source of an input
206
+device clock. For example, to connect the input clock ``clk2`` of
207
+``devB`` to the output clock ``clk1`` of ``devA``, do:
208
+
209
+.. code-block:: c
210
+
211
+ qdev_connect_clock_in(devB, "clk2", qdev_get_clock_out(devA, "clk1"))
212
+
213
+We used ``qdev_get_clock_out()`` above, but any clock can drive an
214
+input clock, even another input clock. The following diagram shows
215
+some examples of connections. Note also that a clock can drive several
216
+other clocks.
217
+
218
+::
219
+
220
+ +------------+ +--------------------------------------------------+
221
+ | Device A | | Device B |
222
+ | | | +---------------------+ |
223
+ | | | | Device C | |
224
+ | +-------+ | | +-------+ | +-------+ +-------+ | +-------+ |
225
+ | |Clock 1|>>-->>|Clock 2|>>+-->>|Clock 3| |Clock 5|>>>>|Clock 6|>>
226
+ | | (out) | | | | (in) | | | | (in) | | (out) | | | (out) | |
227
+ | +-------+ | | +-------+ | | +-------+ +-------+ | +-------+ |
228
+ +------------+ | | +---------------------+ |
229
+ | | |
230
+ | | +--------------+ |
231
+ | | | Device D | |
232
+ | | | +-------+ | |
233
+ | +-->>|Clock 4| | |
234
+ | | | (in) | | |
235
+ | | +-------+ | |
236
+ | +--------------+ |
237
+ +--------------------------------------------------+
238
+
239
+In the above example, when *Clock 1* is updated by *Device A*, three
240
+clocks get the new clock period value: *Clock 2*, *Clock 3* and *Clock 4*.
241
+
242
+It is not possible to disconnect a clock or to change the clock connection
243
+after it is connected.
244
+
245
+Unconnected input clocks
246
+------------------------
247
+
248
+A newly created input clock is disabled (period of 0). This means the
249
+clock will be considered as disabled until the period is updated. If
250
+the clock remains unconnected it will always keep its initial value
251
+of 0. If this is not the desired behaviour, ``clock_set()``,
252
+``clock_set_ns()`` or ``clock_set_hz()`` should be called on the Clock
253
+object during device instance init. For example:
254
+
255
+.. code-block:: c
256
+
257
+ clk = qdev_init_clock_in(DEVICE(dev), "clk-in", clk_in_callback,
258
+ dev);
259
+ /* set initial value to 10ns / 100MHz */
260
+ clock_set_ns(clk, 10);
261
+
262
+Fetching clock frequency/period
263
+-------------------------------
264
+
265
+To get the current state of a clock, use the functions ``clock_get()``,
266
+``clock_get_ns()`` or ``clock_get_hz()``.
267
+
268
+It is also possible to register a callback on clock frequency changes.
269
+Here is an example:
270
+
271
+.. code-block:: c
272
+
273
+ void clock_callback(void *opaque) {
274
+ MyDeviceState *s = (MyDeviceState *) opaque;
275
+ /*
276
+ * 'opaque' is the argument passed to qdev_init_clock_in();
277
+ * usually this will be the device state pointer.
278
+ */
279
+
280
+ /* do something with the new period */
281
+ fprintf(stdout, "device new period is %" PRIu64 "ns\n",
282
+ clock_get_ns(dev->my_clk_input));
283
+ }
284
+
285
+Changing a clock period
286
+-----------------------
287
+
288
+A device can change its outputs using the ``clock_update()``,
289
+``clock_update_ns()`` or ``clock_update_hz()`` function. It will trigger
290
+updates on every connected input.
291
+
292
+For example, let's say that we have an output clock *clkout* and we
293
+have a pointer to it in the device state because we did the following
294
+in init phase:
295
+
296
+.. code-block:: c
297
+
298
+ dev->clkout = qdev_init_clock_out(DEVICE(dev), "clkout");
299
+
300
+Then at any time (apart from the cases listed below), it is possible to
301
+change the clock value by doing:
302
+
303
+.. code-block:: c
304
+
305
+ clock_update_hz(dev->clkout, 1000 * 1000 * 1000); /* 1GHz */
306
+
307
+Because updating a clock may trigger any side effects through
308
+connected clocks and their callbacks, this operation must be done
309
+while holding the qemu io lock.
310
+
311
+For the same reason, one can update clocks only when it is allowed to have
312
+side effects on other objects. In consequence, it is forbidden:
313
+
314
+* during migration,
315
+* and in the enter phase of reset.
316
+
317
+Note that calling ``clock_update[_ns|_hz]()`` is equivalent to calling
318
+``clock_set[_ns|_hz]()`` (with the same arguments) then
319
+``clock_propagate()`` on the clock. Thus, setting the clock value can
320
+be separated from triggering the side-effects. This is often required
321
+to factorize code to handle reset and migration in devices.
322
+
323
+Aliasing clocks
324
+---------------
325
+
326
+Sometimes, one needs to forward, or inherit, a clock from another
327
+device. Typically, when doing device composition, a device might
328
+expose a sub-device's clock without interfering with it. The function
329
+``qdev_alias_clock()`` can be used to achieve this behaviour. Note
330
+that it is possible to expose the clock under a different name.
331
+``qdev_alias_clock()`` works for both input and output clocks.
332
+
333
+For example, if device B is a child of device A,
334
+``device_a_instance_init()`` may do something like this:
335
+
336
+.. code-block:: c
337
+
338
+ void device_a_instance_init(Object *obj)
339
+ {
340
+ AState *A = DEVICE_A(obj);
341
+ BState *B;
342
+ /* create object B as child of A */
343
+ [...]
344
+ qdev_alias_clock(B, "clk", A, "b_clk");
345
+ /*
346
+ * Now A has a clock "b_clk" which is an alias to
347
+ * the clock "clk" of its child B.
348
+ */
349
+ }
350
+
351
+This function does not return any clock object. The new clock has the
352
+same direction (input or output) as the original one. This function
353
+only adds a link to the existing clock. In the above example, object B
354
+remains the only object allowed to use the clock and device A must not
355
+try to change the clock period or set a callback to the clock. This
356
+diagram describes the example with an input clock::
357
+
358
+ +--------------------------+
359
+ | Device A |
360
+ | +--------------+ |
361
+ | | Device B | |
362
+ | | +-------+ | |
363
+ >>"b_clk">>>| "clk" | | |
364
+ | (in) | | (in) | | |
365
+ | | +-------+ | |
366
+ | +--------------+ |
367
+ +--------------------------+
368
+
369
+Migration
370
+---------
371
+
372
+Clock state is not migrated automatically. Every device must handle its
373
+clock migration. Alias clocks must not be migrated.
374
+
375
+To ensure clock states are restored correctly during migration, there
376
+are two solutions.
377
+
378
+Clock states can be migrated by adding an entry into the device
379
+vmstate description. You should use the ``VMSTATE_CLOCK`` macro for this.
380
+This is typically used to migrate an input clock state. For example:
381
+
382
+.. code-block:: c
383
+
384
+ MyDeviceState {
385
+ DeviceState parent_obj;
386
+ [...] /* some fields */
387
+ Clock *clk;
388
+ };
389
+
390
+ VMStateDescription my_device_vmstate = {
391
+ .name = "my_device",
392
+ .fields = (VMStateField[]) {
393
+ [...], /* other migrated fields */
394
+ VMSTATE_CLOCK(clk, MyDeviceState),
395
+ VMSTATE_END_OF_LIST()
396
+ }
397
+ };
398
+
399
+The second solution is to restore the clock state using information already
400
+at our disposal. This can be used to restore output clock states using the
401
+device state. The functions ``clock_set[_ns|_hz]()`` can be used during the
402
+``post_load()`` migration callback.
403
+
404
+When adding clock support to an existing device, if you care about
405
+migration compatibility you will need to be careful, as simply adding
406
+a ``VMSTATE_CLOCK()`` line will break compatibility. Instead, you can
407
+put the ``VMSTATE_CLOCK()`` line into a vmstate subsection with a
408
+suitable ``needed`` function, and use ``clock_set()`` in a
409
+``pre_load()`` function to set the default value that will be used if
410
+the source virtual machine in the migration does not send the clock
411
+state.
412
+
413
+Care should be taken not to use ``clock_update[_ns|_hz]()`` or
414
+``clock_propagate()`` during the whole migration procedure because it
415
+will trigger side effects to other devices in an unknown state.
416
diff --git a/docs/devel/index.rst b/docs/devel/index.rst
417
index XXXXXXX..XXXXXXX 100644
418
--- a/docs/devel/index.rst
419
+++ b/docs/devel/index.rst
420
@@ -XXX,XX +XXX,XX @@ Contents:
421
bitops
422
reset
423
s390-dasd-ipl
424
+ clocks
425
--
426
2.20.1
427
428
diff view generated by jsdifflib
New patch
1
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
3
Add some clocks to zynq_slcr
4
+ the main input clock (ps_clk)
5
+ the reference clock outputs for each uart (uart0 & 1)
6
7
This commit also transitional the slcr to multi-phase reset as it is
8
required to initialize the clocks correctly.
9
10
The clock frequencies are computed using the internal pll & uart configuration
11
registers and the input ps_clk frequency.
12
13
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
14
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
15
Acked-by: Alistair Francis <alistair.francis@wdc.com>
16
Message-id: 20200406135251.157596-7-damien.hedde@greensocs.com
17
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
18
---
19
hw/misc/zynq_slcr.c | 172 ++++++++++++++++++++++++++++++++++++++++++--
20
1 file changed, 168 insertions(+), 4 deletions(-)
21
22
diff --git a/hw/misc/zynq_slcr.c b/hw/misc/zynq_slcr.c
23
index XXXXXXX..XXXXXXX 100644
24
--- a/hw/misc/zynq_slcr.c
25
+++ b/hw/misc/zynq_slcr.c
26
@@ -XXX,XX +XXX,XX @@
27
#include "qemu/log.h"
28
#include "qemu/module.h"
29
#include "hw/registerfields.h"
30
+#include "hw/qdev-clock.h"
31
32
#ifndef ZYNQ_SLCR_ERR_DEBUG
33
#define ZYNQ_SLCR_ERR_DEBUG 0
34
@@ -XXX,XX +XXX,XX @@ REG32(LOCKSTA, 0x00c)
35
REG32(ARM_PLL_CTRL, 0x100)
36
REG32(DDR_PLL_CTRL, 0x104)
37
REG32(IO_PLL_CTRL, 0x108)
38
+/* fields for [ARM|DDR|IO]_PLL_CTRL registers */
39
+ FIELD(xxx_PLL_CTRL, PLL_RESET, 0, 1)
40
+ FIELD(xxx_PLL_CTRL, PLL_PWRDWN, 1, 1)
41
+ FIELD(xxx_PLL_CTRL, PLL_BYPASS_QUAL, 3, 1)
42
+ FIELD(xxx_PLL_CTRL, PLL_BYPASS_FORCE, 4, 1)
43
+ FIELD(xxx_PLL_CTRL, PLL_FPDIV, 12, 7)
44
REG32(PLL_STATUS, 0x10c)
45
REG32(ARM_PLL_CFG, 0x110)
46
REG32(DDR_PLL_CFG, 0x114)
47
@@ -XXX,XX +XXX,XX @@ REG32(SMC_CLK_CTRL, 0x148)
48
REG32(LQSPI_CLK_CTRL, 0x14c)
49
REG32(SDIO_CLK_CTRL, 0x150)
50
REG32(UART_CLK_CTRL, 0x154)
51
+ FIELD(UART_CLK_CTRL, CLKACT0, 0, 1)
52
+ FIELD(UART_CLK_CTRL, CLKACT1, 1, 1)
53
+ FIELD(UART_CLK_CTRL, SRCSEL, 4, 2)
54
+ FIELD(UART_CLK_CTRL, DIVISOR, 8, 6)
55
REG32(SPI_CLK_CTRL, 0x158)
56
REG32(CAN_CLK_CTRL, 0x15c)
57
REG32(CAN_MIOCLK_CTRL, 0x160)
58
@@ -XXX,XX +XXX,XX @@ typedef struct ZynqSLCRState {
59
MemoryRegion iomem;
60
61
uint32_t regs[ZYNQ_SLCR_NUM_REGS];
62
+
63
+ Clock *ps_clk;
64
+ Clock *uart0_ref_clk;
65
+ Clock *uart1_ref_clk;
66
} ZynqSLCRState;
67
68
-static void zynq_slcr_reset(DeviceState *d)
69
+/*
70
+ * return the output frequency of ARM/DDR/IO pll
71
+ * using input frequency and PLL_CTRL register
72
+ */
73
+static uint64_t zynq_slcr_compute_pll(uint64_t input, uint32_t ctrl_reg)
74
{
75
- ZynqSLCRState *s = ZYNQ_SLCR(d);
76
+ uint32_t mult = ((ctrl_reg & R_xxx_PLL_CTRL_PLL_FPDIV_MASK) >>
77
+ R_xxx_PLL_CTRL_PLL_FPDIV_SHIFT);
78
+
79
+ /* first, check if pll is bypassed */
80
+ if (ctrl_reg & R_xxx_PLL_CTRL_PLL_BYPASS_FORCE_MASK) {
81
+ return input;
82
+ }
83
+
84
+ /* is pll disabled ? */
85
+ if (ctrl_reg & (R_xxx_PLL_CTRL_PLL_RESET_MASK |
86
+ R_xxx_PLL_CTRL_PLL_PWRDWN_MASK)) {
87
+ return 0;
88
+ }
89
+
90
+ /* frequency multiplier -> period division */
91
+ return input / mult;
92
+}
93
+
94
+/*
95
+ * return the output period of a clock given:
96
+ * + the periods in an array corresponding to input mux selector
97
+ * + the register xxx_CLK_CTRL value
98
+ * + enable bit index in ctrl register
99
+ *
100
+ * This function makes the assumption that the ctrl_reg value is organized as
101
+ * follows:
102
+ * + bits[13:8] clock frequency divisor
103
+ * + bits[5:4] clock mux selector (index in array)
104
+ * + bits[index] clock enable
105
+ */
106
+static uint64_t zynq_slcr_compute_clock(const uint64_t periods[],
107
+ uint32_t ctrl_reg,
108
+ unsigned index)
109
+{
110
+ uint32_t srcsel = extract32(ctrl_reg, 4, 2); /* bits [5:4] */
111
+ uint32_t divisor = extract32(ctrl_reg, 8, 6); /* bits [13:8] */
112
+
113
+ /* first, check if clock is disabled */
114
+ if (((ctrl_reg >> index) & 1u) == 0) {
115
+ return 0;
116
+ }
117
+
118
+ /*
119
+ * according to the Zynq technical ref. manual UG585 v1.12.2 in
120
+ * Clocks chapter, section 25.10.1 page 705:
121
+ * "The 6-bit divider provides a divide range of 1 to 63"
122
+ * We follow here what is implemented in linux kernel and consider
123
+ * the 0 value as a bypass (no division).
124
+ */
125
+ /* frequency divisor -> period multiplication */
126
+ return periods[srcsel] * (divisor ? divisor : 1u);
127
+}
128
+
129
+/*
130
+ * macro helper around zynq_slcr_compute_clock to avoid repeating
131
+ * the register name.
132
+ */
133
+#define ZYNQ_COMPUTE_CLK(state, plls, reg, enable_field) \
134
+ zynq_slcr_compute_clock((plls), (state)->regs[reg], \
135
+ reg ## _ ## enable_field ## _SHIFT)
136
+
137
+/**
138
+ * Compute and set the ouputs clocks periods.
139
+ * But do not propagate them further. Connected clocks
140
+ * will not receive any updates (See zynq_slcr_compute_clocks())
141
+ */
142
+static void zynq_slcr_compute_clocks(ZynqSLCRState *s)
143
+{
144
+ uint64_t ps_clk = clock_get(s->ps_clk);
145
+
146
+ /* consider outputs clocks are disabled while in reset */
147
+ if (device_is_in_reset(DEVICE(s))) {
148
+ ps_clk = 0;
149
+ }
150
+
151
+ uint64_t io_pll = zynq_slcr_compute_pll(ps_clk, s->regs[R_IO_PLL_CTRL]);
152
+ uint64_t arm_pll = zynq_slcr_compute_pll(ps_clk, s->regs[R_ARM_PLL_CTRL]);
153
+ uint64_t ddr_pll = zynq_slcr_compute_pll(ps_clk, s->regs[R_DDR_PLL_CTRL]);
154
+
155
+ uint64_t uart_mux[4] = {io_pll, io_pll, arm_pll, ddr_pll};
156
+
157
+ /* compute uartX reference clocks */
158
+ clock_set(s->uart0_ref_clk,
159
+ ZYNQ_COMPUTE_CLK(s, uart_mux, R_UART_CLK_CTRL, CLKACT0));
160
+ clock_set(s->uart1_ref_clk,
161
+ ZYNQ_COMPUTE_CLK(s, uart_mux, R_UART_CLK_CTRL, CLKACT1));
162
+}
163
+
164
+/**
165
+ * Propagate the outputs clocks.
166
+ * zynq_slcr_compute_clocks() should have been called before
167
+ * to configure them.
168
+ */
169
+static void zynq_slcr_propagate_clocks(ZynqSLCRState *s)
170
+{
171
+ clock_propagate(s->uart0_ref_clk);
172
+ clock_propagate(s->uart1_ref_clk);
173
+}
174
+
175
+static void zynq_slcr_ps_clk_callback(void *opaque)
176
+{
177
+ ZynqSLCRState *s = (ZynqSLCRState *) opaque;
178
+ zynq_slcr_compute_clocks(s);
179
+ zynq_slcr_propagate_clocks(s);
180
+}
181
+
182
+static void zynq_slcr_reset_init(Object *obj, ResetType type)
183
+{
184
+ ZynqSLCRState *s = ZYNQ_SLCR(obj);
185
int i;
186
187
DB_PRINT("RESET\n");
188
@@ -XXX,XX +XXX,XX @@ static void zynq_slcr_reset(DeviceState *d)
189
s->regs[R_DDRIOB + 12] = 0x00000021;
190
}
191
192
+static void zynq_slcr_reset_hold(Object *obj)
193
+{
194
+ ZynqSLCRState *s = ZYNQ_SLCR(obj);
195
+
196
+ /* will disable all output clocks */
197
+ zynq_slcr_compute_clocks(s);
198
+ zynq_slcr_propagate_clocks(s);
199
+}
200
+
201
+static void zynq_slcr_reset_exit(Object *obj)
202
+{
203
+ ZynqSLCRState *s = ZYNQ_SLCR(obj);
204
+
205
+ /* will compute output clocks according to ps_clk and registers */
206
+ zynq_slcr_compute_clocks(s);
207
+ zynq_slcr_propagate_clocks(s);
208
+}
209
210
static bool zynq_slcr_check_offset(hwaddr offset, bool rnw)
211
{
212
@@ -XXX,XX +XXX,XX @@ static void zynq_slcr_write(void *opaque, hwaddr offset,
213
qemu_system_reset_request(SHUTDOWN_CAUSE_GUEST_RESET);
214
}
215
break;
216
+ case R_IO_PLL_CTRL:
217
+ case R_ARM_PLL_CTRL:
218
+ case R_DDR_PLL_CTRL:
219
+ case R_UART_CLK_CTRL:
220
+ zynq_slcr_compute_clocks(s);
221
+ zynq_slcr_propagate_clocks(s);
222
+ break;
223
}
224
}
225
226
@@ -XXX,XX +XXX,XX @@ static const MemoryRegionOps slcr_ops = {
227
.endianness = DEVICE_NATIVE_ENDIAN,
228
};
229
230
+static const ClockPortInitArray zynq_slcr_clocks = {
231
+ QDEV_CLOCK_IN(ZynqSLCRState, ps_clk, zynq_slcr_ps_clk_callback),
232
+ QDEV_CLOCK_OUT(ZynqSLCRState, uart0_ref_clk),
233
+ QDEV_CLOCK_OUT(ZynqSLCRState, uart1_ref_clk),
234
+ QDEV_CLOCK_END
235
+};
236
+
237
static void zynq_slcr_init(Object *obj)
238
{
239
ZynqSLCRState *s = ZYNQ_SLCR(obj);
240
@@ -XXX,XX +XXX,XX @@ static void zynq_slcr_init(Object *obj)
241
memory_region_init_io(&s->iomem, obj, &slcr_ops, s, "slcr",
242
ZYNQ_SLCR_MMIO_SIZE);
243
sysbus_init_mmio(SYS_BUS_DEVICE(obj), &s->iomem);
244
+
245
+ qdev_init_clocks(DEVICE(obj), zynq_slcr_clocks);
246
}
247
248
static const VMStateDescription vmstate_zynq_slcr = {
249
.name = "zynq_slcr",
250
- .version_id = 2,
251
+ .version_id = 3,
252
.minimum_version_id = 2,
253
.fields = (VMStateField[]) {
254
VMSTATE_UINT32_ARRAY(regs, ZynqSLCRState, ZYNQ_SLCR_NUM_REGS),
255
+ VMSTATE_CLOCK_V(ps_clk, ZynqSLCRState, 3),
256
VMSTATE_END_OF_LIST()
257
}
258
};
259
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_zynq_slcr = {
260
static void zynq_slcr_class_init(ObjectClass *klass, void *data)
261
{
262
DeviceClass *dc = DEVICE_CLASS(klass);
263
+ ResettableClass *rc = RESETTABLE_CLASS(klass);
264
265
dc->vmsd = &vmstate_zynq_slcr;
266
- dc->reset = zynq_slcr_reset;
267
+ rc->phases.enter = zynq_slcr_reset_init;
268
+ rc->phases.hold = zynq_slcr_reset_hold;
269
+ rc->phases.exit = zynq_slcr_reset_exit;
270
}
271
272
static const TypeInfo zynq_slcr_info = {
273
--
274
2.20.1
275
276
diff view generated by jsdifflib
New patch
1
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
3
Switch the cadence uart to multi-phase reset and add the
4
reference clock input.
5
6
The input clock frequency is added to the migration structure.
7
8
The reference clock controls the baudrate generation. If it disabled,
9
any input characters and events are ignored.
10
11
If this clock remains unconnected, the uart behaves as before
12
(it default to a 50MHz ref clock).
13
14
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
15
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
16
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
17
Message-id: 20200406135251.157596-8-damien.hedde@greensocs.com
18
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
19
---
20
include/hw/char/cadence_uart.h | 1 +
21
hw/char/cadence_uart.c | 73 +++++++++++++++++++++++++++++-----
22
hw/char/trace-events | 3 ++
23
3 files changed, 67 insertions(+), 10 deletions(-)
24
25
diff --git a/include/hw/char/cadence_uart.h b/include/hw/char/cadence_uart.h
26
index XXXXXXX..XXXXXXX 100644
27
--- a/include/hw/char/cadence_uart.h
28
+++ b/include/hw/char/cadence_uart.h
29
@@ -XXX,XX +XXX,XX @@ typedef struct {
30
CharBackend chr;
31
qemu_irq irq;
32
QEMUTimer *fifo_trigger_handle;
33
+ Clock *refclk;
34
} CadenceUARTState;
35
36
static inline DeviceState *cadence_uart_create(hwaddr addr,
37
diff --git a/hw/char/cadence_uart.c b/hw/char/cadence_uart.c
38
index XXXXXXX..XXXXXXX 100644
39
--- a/hw/char/cadence_uart.c
40
+++ b/hw/char/cadence_uart.c
41
@@ -XXX,XX +XXX,XX @@
42
#include "qemu/module.h"
43
#include "hw/char/cadence_uart.h"
44
#include "hw/irq.h"
45
+#include "hw/qdev-clock.h"
46
+#include "trace.h"
47
48
#ifdef CADENCE_UART_ERR_DEBUG
49
#define DB_PRINT(...) do { \
50
@@ -XXX,XX +XXX,XX @@
51
#define LOCAL_LOOPBACK (0x2 << UART_MR_CHMODE_SH)
52
#define REMOTE_LOOPBACK (0x3 << UART_MR_CHMODE_SH)
53
54
-#define UART_INPUT_CLK 50000000
55
+#define UART_DEFAULT_REF_CLK (50 * 1000 * 1000)
56
57
#define R_CR (0x00/4)
58
#define R_MR (0x04/4)
59
@@ -XXX,XX +XXX,XX @@ static void uart_send_breaks(CadenceUARTState *s)
60
static void uart_parameters_setup(CadenceUARTState *s)
61
{
62
QEMUSerialSetParams ssp;
63
- unsigned int baud_rate, packet_size;
64
+ unsigned int baud_rate, packet_size, input_clk;
65
+ input_clk = clock_get_hz(s->refclk);
66
67
- baud_rate = (s->r[R_MR] & UART_MR_CLKS) ?
68
- UART_INPUT_CLK / 8 : UART_INPUT_CLK;
69
+ baud_rate = (s->r[R_MR] & UART_MR_CLKS) ? input_clk / 8 : input_clk;
70
+ baud_rate /= (s->r[R_BRGR] * (s->r[R_BDIV] + 1));
71
+ trace_cadence_uart_baudrate(baud_rate);
72
+
73
+ ssp.speed = baud_rate;
74
75
- ssp.speed = baud_rate / (s->r[R_BRGR] * (s->r[R_BDIV] + 1));
76
packet_size = 1;
77
78
switch (s->r[R_MR] & UART_MR_PAR) {
79
@@ -XXX,XX +XXX,XX @@ static void uart_parameters_setup(CadenceUARTState *s)
80
}
81
82
packet_size += ssp.data_bits + ssp.stop_bits;
83
+ if (ssp.speed == 0) {
84
+ /*
85
+ * Avoid division-by-zero below.
86
+ * TODO: find something better
87
+ */
88
+ ssp.speed = 1;
89
+ }
90
s->char_tx_time = (NANOSECONDS_PER_SECOND / ssp.speed) * packet_size;
91
qemu_chr_fe_ioctl(&s->chr, CHR_IOCTL_SERIAL_SET_PARAMS, &ssp);
92
}
93
@@ -XXX,XX +XXX,XX @@ static void uart_receive(void *opaque, const uint8_t *buf, int size)
94
CadenceUARTState *s = opaque;
95
uint32_t ch_mode = s->r[R_MR] & UART_MR_CHMODE;
96
97
+ /* ignore characters when unclocked or in reset */
98
+ if (!clock_is_enabled(s->refclk) || device_is_in_reset(DEVICE(s))) {
99
+ return;
100
+ }
101
+
102
if (ch_mode == NORMAL_MODE || ch_mode == ECHO_MODE) {
103
uart_write_rx_fifo(opaque, buf, size);
104
}
105
@@ -XXX,XX +XXX,XX @@ static void uart_event(void *opaque, QEMUChrEvent event)
106
CadenceUARTState *s = opaque;
107
uint8_t buf = '\0';
108
109
+ /* ignore characters when unclocked or in reset */
110
+ if (!clock_is_enabled(s->refclk) || device_is_in_reset(DEVICE(s))) {
111
+ return;
112
+ }
113
+
114
if (event == CHR_EVENT_BREAK) {
115
uart_write_rx_fifo(opaque, &buf, 1);
116
}
117
@@ -XXX,XX +XXX,XX @@ static const MemoryRegionOps uart_ops = {
118
.endianness = DEVICE_NATIVE_ENDIAN,
119
};
120
121
-static void cadence_uart_reset(DeviceState *dev)
122
+static void cadence_uart_reset_init(Object *obj, ResetType type)
123
{
124
- CadenceUARTState *s = CADENCE_UART(dev);
125
+ CadenceUARTState *s = CADENCE_UART(obj);
126
127
s->r[R_CR] = 0x00000128;
128
s->r[R_IMR] = 0;
129
@@ -XXX,XX +XXX,XX @@ static void cadence_uart_reset(DeviceState *dev)
130
s->r[R_BRGR] = 0x0000028B;
131
s->r[R_BDIV] = 0x0000000F;
132
s->r[R_TTRIG] = 0x00000020;
133
+}
134
+
135
+static void cadence_uart_reset_hold(Object *obj)
136
+{
137
+ CadenceUARTState *s = CADENCE_UART(obj);
138
139
uart_rx_reset(s);
140
uart_tx_reset(s);
141
@@ -XXX,XX +XXX,XX @@ static void cadence_uart_realize(DeviceState *dev, Error **errp)
142
uart_event, NULL, s, NULL, true);
143
}
144
145
+static void cadence_uart_refclk_update(void *opaque)
146
+{
147
+ CadenceUARTState *s = opaque;
148
+
149
+ /* recompute uart's speed on clock change */
150
+ uart_parameters_setup(s);
151
+}
152
+
153
static void cadence_uart_init(Object *obj)
154
{
155
SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
156
@@ -XXX,XX +XXX,XX @@ static void cadence_uart_init(Object *obj)
157
sysbus_init_mmio(sbd, &s->iomem);
158
sysbus_init_irq(sbd, &s->irq);
159
160
+ s->refclk = qdev_init_clock_in(DEVICE(obj), "refclk",
161
+ cadence_uart_refclk_update, s);
162
+ /* initialize the frequency in case the clock remains unconnected */
163
+ clock_set_hz(s->refclk, UART_DEFAULT_REF_CLK);
164
+
165
s->char_tx_time = (NANOSECONDS_PER_SECOND / 9600) * 10;
166
}
167
168
+static int cadence_uart_pre_load(void *opaque)
169
+{
170
+ CadenceUARTState *s = opaque;
171
+
172
+ /* the frequency will be overriden if the refclk field is present */
173
+ clock_set_hz(s->refclk, UART_DEFAULT_REF_CLK);
174
+ return 0;
175
+}
176
+
177
static int cadence_uart_post_load(void *opaque, int version_id)
178
{
179
CadenceUARTState *s = opaque;
180
@@ -XXX,XX +XXX,XX @@ static int cadence_uart_post_load(void *opaque, int version_id)
181
182
static const VMStateDescription vmstate_cadence_uart = {
183
.name = "cadence_uart",
184
- .version_id = 2,
185
+ .version_id = 3,
186
.minimum_version_id = 2,
187
+ .pre_load = cadence_uart_pre_load,
188
.post_load = cadence_uart_post_load,
189
.fields = (VMStateField[]) {
190
VMSTATE_UINT32_ARRAY(r, CadenceUARTState, CADENCE_UART_R_MAX),
191
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_cadence_uart = {
192
VMSTATE_UINT32(tx_count, CadenceUARTState),
193
VMSTATE_UINT32(rx_wpos, CadenceUARTState),
194
VMSTATE_TIMER_PTR(fifo_trigger_handle, CadenceUARTState),
195
+ VMSTATE_CLOCK_V(refclk, CadenceUARTState, 3),
196
VMSTATE_END_OF_LIST()
197
- }
198
+ },
199
};
200
201
static Property cadence_uart_properties[] = {
202
@@ -XXX,XX +XXX,XX @@ static Property cadence_uart_properties[] = {
203
static void cadence_uart_class_init(ObjectClass *klass, void *data)
204
{
205
DeviceClass *dc = DEVICE_CLASS(klass);
206
+ ResettableClass *rc = RESETTABLE_CLASS(klass);
207
208
dc->realize = cadence_uart_realize;
209
dc->vmsd = &vmstate_cadence_uart;
210
- dc->reset = cadence_uart_reset;
211
+ rc->phases.enter = cadence_uart_reset_init;
212
+ rc->phases.hold = cadence_uart_reset_hold;
213
device_class_set_props(dc, cadence_uart_properties);
214
}
215
216
diff --git a/hw/char/trace-events b/hw/char/trace-events
217
index XXXXXXX..XXXXXXX 100644
218
--- a/hw/char/trace-events
219
+++ b/hw/char/trace-events
220
@@ -XXX,XX +XXX,XX @@ exynos_uart_wo_read(uint32_t channel, const char *name, uint32_t reg) "UART%d: T
221
exynos_uart_rxsize(uint32_t channel, uint32_t size) "UART%d: Rx FIFO size: %d"
222
exynos_uart_channel_error(uint32_t channel) "Wrong UART channel number: %d"
223
exynos_uart_rx_timeout(uint32_t channel, uint32_t stat, uint32_t intsp) "UART%d: Rx timeout stat=0x%x intsp=0x%x"
224
+
225
+# hw/char/cadence_uart.c
226
+cadence_uart_baudrate(unsigned baudrate) "baudrate %u"
227
--
228
2.20.1
229
230
diff view generated by jsdifflib
1
From: Cédric Le Goater <clg@kaod.org>
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
2
3
The code looks better, it removes duplicated lines and it will ease
3
Add the connection between the slcr's output clocks and the uarts inputs.
4
the introduction of common properties for the Aspeed machines.
5
4
6
Signed-off-by: Cédric Le Goater <clg@kaod.org>
5
Also add the main board clock 'ps_clk', which is hard-coded to 33.33MHz
7
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
6
(the default frequency). This clock is used to feed the slcr's input
8
Message-id: 20180921161939.822-4-clg@kaod.org
7
clock.
8
9
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
10
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
11
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
12
Message-id: 20200406135251.157596-9-damien.hedde@greensocs.com
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
14
---
11
include/hw/arm/aspeed.h | 46 +++++++++
15
hw/arm/xilinx_zynq.c | 57 +++++++++++++++++++++++++++++++++++++-------
12
hw/arm/aspeed.c | 212 +++++++++++++---------------------------
16
1 file changed, 49 insertions(+), 8 deletions(-)
13
2 files changed, 116 insertions(+), 142 deletions(-)
14
create mode 100644 include/hw/arm/aspeed.h
15
17
16
diff --git a/include/hw/arm/aspeed.h b/include/hw/arm/aspeed.h
18
diff --git a/hw/arm/xilinx_zynq.c b/hw/arm/xilinx_zynq.c
17
new file mode 100644
19
index XXXXXXX..XXXXXXX 100644
18
index XXXXXXX..XXXXXXX
20
--- a/hw/arm/xilinx_zynq.c
19
--- /dev/null
21
+++ b/hw/arm/xilinx_zynq.c
20
+++ b/include/hw/arm/aspeed.h
21
@@ -XXX,XX +XXX,XX @@
22
@@ -XXX,XX +XXX,XX @@
22
+/*
23
#include "hw/char/cadence_uart.h"
23
+ * Aspeed Machines
24
#include "hw/net/cadence_gem.h"
24
+ *
25
#include "hw/cpu/a9mpcore.h"
25
+ * Copyright 2018 IBM Corp.
26
+#include "hw/qdev-clock.h"
26
+ *
27
+#include "sysemu/reset.h"
27
+ * This code is licensed under the GPL version 2 or later. See
28
+ * the COPYING file in the top-level directory.
29
+ */
30
+#ifndef ARM_ASPEED_H
31
+#define ARM_ASPEED_H
32
+
28
+
33
+#include "hw/boards.h"
29
+#define TYPE_ZYNQ_MACHINE MACHINE_TYPE_NAME("xilinx-zynq-a9")
30
+#define ZYNQ_MACHINE(obj) \
31
+ OBJECT_CHECK(ZynqMachineState, (obj), TYPE_ZYNQ_MACHINE)
34
+
32
+
35
+typedef struct AspeedBoardState AspeedBoardState;
33
+/* board base frequency: 33.333333 MHz */
34
+#define PS_CLK_FREQUENCY (100 * 1000 * 1000 / 3)
35
36
#define NUM_SPI_FLASHES 4
37
#define NUM_QSPI_FLASHES 2
38
@@ -XXX,XX +XXX,XX @@ static const int dma_irqs[8] = {
39
0xe3401000 + ARMV7_IMM16(extract32((val), 16, 16)), /* movt r1 ... */ \
40
0xe5801000 + (addr)
41
42
+typedef struct ZynqMachineState {
43
+ MachineState parent;
44
+ Clock *ps_clk;
45
+} ZynqMachineState;
36
+
46
+
37
+typedef struct AspeedBoardConfig {
47
static void zynq_write_board_setup(ARMCPU *cpu,
38
+ const char *name;
48
const struct arm_boot_info *info)
39
+ const char *desc;
49
{
40
+ const char *soc_name;
50
@@ -XXX,XX +XXX,XX @@ static inline void zynq_init_spi_flashes(uint32_t base_addr, qemu_irq irq,
41
+ uint32_t hw_strap1;
51
42
+ const char *fmc_model;
52
static void zynq_init(MachineState *machine)
43
+ const char *spi_model;
53
{
44
+ uint32_t num_cs;
54
+ ZynqMachineState *zynq_machine = ZYNQ_MACHINE(machine);
45
+ void (*i2c_init)(AspeedBoardState *bmc);
55
ARMCPU *cpu;
46
+} AspeedBoardConfig;
56
MemoryRegion *address_space_mem = get_system_memory();
57
MemoryRegion *ocm_ram = g_new(MemoryRegion, 1);
58
- DeviceState *dev;
59
+ DeviceState *dev, *slcr;
60
SysBusDevice *busdev;
61
qemu_irq pic[64];
62
int n;
63
@@ -XXX,XX +XXX,XX @@ static void zynq_init(MachineState *machine)
64
1, 0x0066, 0x0022, 0x0000, 0x0000, 0x0555, 0x2aa,
65
0);
66
67
- dev = qdev_create(NULL, "xilinx,zynq_slcr");
68
- qdev_init_nofail(dev);
69
- sysbus_mmio_map(SYS_BUS_DEVICE(dev), 0, 0xF8000000);
70
+ /* Create slcr, keep a pointer to connect clocks */
71
+ slcr = qdev_create(NULL, "xilinx,zynq_slcr");
72
+ qdev_init_nofail(slcr);
73
+ sysbus_mmio_map(SYS_BUS_DEVICE(slcr), 0, 0xF8000000);
47
+
74
+
48
+#define TYPE_ASPEED_MACHINE MACHINE_TYPE_NAME("aspeed")
75
+ /* Create the main clock source, and feed slcr with it */
49
+#define ASPEED_MACHINE(obj) \
76
+ zynq_machine->ps_clk = CLOCK(object_new(TYPE_CLOCK));
50
+ OBJECT_CHECK(AspeedMachine, (obj), TYPE_ASPEED_MACHINE)
77
+ object_property_add_child(OBJECT(zynq_machine), "ps_clk",
51
+
78
+ OBJECT(zynq_machine->ps_clk), &error_abort);
52
+typedef struct AspeedMachine {
79
+ object_unref(OBJECT(zynq_machine->ps_clk));
53
+ MachineState parent_obj;
80
+ clock_set_hz(zynq_machine->ps_clk, PS_CLK_FREQUENCY);
54
+} AspeedMachine;
81
+ qdev_connect_clock_in(slcr, "ps_clk", zynq_machine->ps_clk);
55
+
82
56
+#define ASPEED_MACHINE_CLASS(klass) \
83
dev = qdev_create(NULL, TYPE_A9MPCORE_PRIV);
57
+ OBJECT_CLASS_CHECK(AspeedMachineClass, (klass), TYPE_ASPEED_MACHINE)
84
qdev_prop_set_uint32(dev, "num-cpu", 1);
58
+#define ASPEED_MACHINE_GET_CLASS(obj) \
85
@@ -XXX,XX +XXX,XX @@ static void zynq_init(MachineState *machine)
59
+ OBJECT_GET_CLASS(AspeedMachineClass, (obj), TYPE_ASPEED_MACHINE)
86
sysbus_create_simple(TYPE_CHIPIDEA, 0xE0002000, pic[53 - IRQ_OFFSET]);
60
+
87
sysbus_create_simple(TYPE_CHIPIDEA, 0xE0003000, pic[76 - IRQ_OFFSET]);
61
+typedef struct AspeedMachineClass {
88
62
+ MachineClass parent_obj;
89
- cadence_uart_create(0xE0000000, pic[59 - IRQ_OFFSET], serial_hd(0));
63
+ const AspeedBoardConfig *board;
90
- cadence_uart_create(0xE0001000, pic[82 - IRQ_OFFSET], serial_hd(1));
64
+} AspeedMachineClass;
91
+ dev = cadence_uart_create(0xE0000000, pic[59 - IRQ_OFFSET], serial_hd(0));
65
+
92
+ qdev_connect_clock_in(dev, "refclk",
66
+
93
+ qdev_get_clock_out(slcr, "uart0_ref_clk"));
67
+#endif
94
+ dev = cadence_uart_create(0xE0001000, pic[82 - IRQ_OFFSET], serial_hd(1));
68
diff --git a/hw/arm/aspeed.c b/hw/arm/aspeed.c
95
+ qdev_connect_clock_in(dev, "refclk",
69
index XXXXXXX..XXXXXXX 100644
96
+ qdev_get_clock_out(slcr, "uart1_ref_clk"));
70
--- a/hw/arm/aspeed.c
97
71
+++ b/hw/arm/aspeed.c
98
sysbus_create_varargs("cadence_ttc", 0xF8001000,
72
@@ -XXX,XX +XXX,XX @@
99
pic[42-IRQ_OFFSET], pic[43-IRQ_OFFSET], pic[44-IRQ_OFFSET], NULL);
73
#include "cpu.h"
100
@@ -XXX,XX +XXX,XX @@ static void zynq_init(MachineState *machine)
74
#include "exec/address-spaces.h"
101
arm_load_kernel(ARM_CPU(first_cpu), machine, &zynq_binfo);
75
#include "hw/arm/arm.h"
76
+#include "hw/arm/aspeed.h"
77
#include "hw/arm/aspeed_soc.h"
78
#include "hw/boards.h"
79
#include "hw/i2c/smbus.h"
80
@@ -XXX,XX +XXX,XX @@ typedef struct AspeedBoardState {
81
MemoryRegion max_ram;
82
} AspeedBoardState;
83
84
-typedef struct AspeedBoardConfig {
85
- const char *soc_name;
86
- uint32_t hw_strap1;
87
- const char *fmc_model;
88
- const char *spi_model;
89
- uint32_t num_cs;
90
- void (*i2c_init)(AspeedBoardState *bmc);
91
-} AspeedBoardConfig;
92
-
93
-enum {
94
- PALMETTO_BMC,
95
- AST2500_EVB,
96
- ROMULUS_BMC,
97
- WITHERSPOON_BMC,
98
-};
99
-
100
/* Palmetto hardware value: 0x120CE416 */
101
#define PALMETTO_BMC_HW_STRAP1 ( \
102
SCU_AST2400_HW_STRAP_DRAM_SIZE(DRAM_SIZE_256MB) | \
103
@@ -XXX,XX +XXX,XX @@ enum {
104
/* Witherspoon hardware value: 0xF10AD216 (but use romulus definition) */
105
#define WITHERSPOON_BMC_HW_STRAP1 ROMULUS_BMC_HW_STRAP1
106
107
-static void palmetto_bmc_i2c_init(AspeedBoardState *bmc);
108
-static void ast2500_evb_i2c_init(AspeedBoardState *bmc);
109
-static void romulus_bmc_i2c_init(AspeedBoardState *bmc);
110
-static void witherspoon_bmc_i2c_init(AspeedBoardState *bmc);
111
-
112
-static const AspeedBoardConfig aspeed_boards[] = {
113
- [PALMETTO_BMC] = {
114
- .soc_name = "ast2400-a1",
115
- .hw_strap1 = PALMETTO_BMC_HW_STRAP1,
116
- .fmc_model = "n25q256a",
117
- .spi_model = "mx25l25635e",
118
- .num_cs = 1,
119
- .i2c_init = palmetto_bmc_i2c_init,
120
- },
121
- [AST2500_EVB] = {
122
- .soc_name = "ast2500-a1",
123
- .hw_strap1 = AST2500_EVB_HW_STRAP1,
124
- .fmc_model = "w25q256",
125
- .spi_model = "mx25l25635e",
126
- .num_cs = 1,
127
- .i2c_init = ast2500_evb_i2c_init,
128
- },
129
- [ROMULUS_BMC] = {
130
- .soc_name = "ast2500-a1",
131
- .hw_strap1 = ROMULUS_BMC_HW_STRAP1,
132
- .fmc_model = "n25q256a",
133
- .spi_model = "mx66l1g45g",
134
- .num_cs = 2,
135
- .i2c_init = romulus_bmc_i2c_init,
136
- },
137
- [WITHERSPOON_BMC] = {
138
- .soc_name = "ast2500-a1",
139
- .hw_strap1 = WITHERSPOON_BMC_HW_STRAP1,
140
- .fmc_model = "mx25l25635e",
141
- .spi_model = "mx66l1g45g",
142
- .num_cs = 2,
143
- .i2c_init = witherspoon_bmc_i2c_init,
144
- },
145
-};
146
-
147
/*
148
* The max ram region is for firmwares that scan the address space
149
* with load/store to guess how much RAM the SoC has.
150
@@ -XXX,XX +XXX,XX @@ static void palmetto_bmc_i2c_init(AspeedBoardState *bmc)
151
object_property_set_int(OBJECT(dev), 110000, "temperature3", &error_abort);
152
}
102
}
153
103
154
-static void palmetto_bmc_init(MachineState *machine)
104
-static void zynq_machine_init(MachineClass *mc)
155
-{
105
+static void zynq_machine_class_init(ObjectClass *oc, void *data)
156
- aspeed_board_init(machine, &aspeed_boards[PALMETTO_BMC]);
157
-}
158
-
159
-static void palmetto_bmc_class_init(ObjectClass *oc, void *data)
160
-{
161
- MachineClass *mc = MACHINE_CLASS(oc);
162
-
163
- mc->desc = "OpenPOWER Palmetto BMC (ARM926EJ-S)";
164
- mc->init = palmetto_bmc_init;
165
- mc->max_cpus = 1;
166
- mc->no_sdcard = 1;
167
- mc->no_floppy = 1;
168
- mc->no_cdrom = 1;
169
- mc->no_parallel = 1;
170
-}
171
-
172
-static const TypeInfo palmetto_bmc_type = {
173
- .name = MACHINE_TYPE_NAME("palmetto-bmc"),
174
- .parent = TYPE_MACHINE,
175
- .class_init = palmetto_bmc_class_init,
176
-};
177
-
178
static void ast2500_evb_i2c_init(AspeedBoardState *bmc)
179
{
106
{
180
AspeedSoCState *soc = &bmc->soc;
107
+ MachineClass *mc = MACHINE_CLASS(oc);
181
@@ -XXX,XX +XXX,XX @@ static void ast2500_evb_i2c_init(AspeedBoardState *bmc)
108
mc->desc = "Xilinx Zynq Platform Baseboard for Cortex-A9";
182
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), "ds1338", 0x32);
109
mc->init = zynq_init;
110
mc->max_cpus = 1;
111
@@ -XXX,XX +XXX,XX @@ static void zynq_machine_init(MachineClass *mc)
112
mc->default_ram_id = "zynq.ext_ram";
183
}
113
}
184
114
185
-static void ast2500_evb_init(MachineState *machine)
115
-DEFINE_MACHINE("xilinx-zynq-a9", zynq_machine_init)
186
-{
116
+static const TypeInfo zynq_machine_type = {
187
- aspeed_board_init(machine, &aspeed_boards[AST2500_EVB]);
117
+ .name = TYPE_ZYNQ_MACHINE,
188
-}
118
+ .parent = TYPE_MACHINE,
189
-
119
+ .class_init = zynq_machine_class_init,
190
-static void ast2500_evb_class_init(ObjectClass *oc, void *data)
120
+ .instance_size = sizeof(ZynqMachineState),
191
-{
192
- MachineClass *mc = MACHINE_CLASS(oc);
193
-
194
- mc->desc = "Aspeed AST2500 EVB (ARM1176)";
195
- mc->init = ast2500_evb_init;
196
- mc->max_cpus = 1;
197
- mc->no_sdcard = 1;
198
- mc->no_floppy = 1;
199
- mc->no_cdrom = 1;
200
- mc->no_parallel = 1;
201
-}
202
-
203
-static const TypeInfo ast2500_evb_type = {
204
- .name = MACHINE_TYPE_NAME("ast2500-evb"),
205
- .parent = TYPE_MACHINE,
206
- .class_init = ast2500_evb_class_init,
207
-};
208
-
209
static void romulus_bmc_i2c_init(AspeedBoardState *bmc)
210
{
211
AspeedSoCState *soc = &bmc->soc;
212
@@ -XXX,XX +XXX,XX @@ static void romulus_bmc_i2c_init(AspeedBoardState *bmc)
213
i2c_create_slave(aspeed_i2c_get_bus(DEVICE(&soc->i2c), 11), "ds1338", 0x32);
214
}
215
216
-static void romulus_bmc_init(MachineState *machine)
217
-{
218
- aspeed_board_init(machine, &aspeed_boards[ROMULUS_BMC]);
219
-}
220
-
221
-static void romulus_bmc_class_init(ObjectClass *oc, void *data)
222
-{
223
- MachineClass *mc = MACHINE_CLASS(oc);
224
-
225
- mc->desc = "OpenPOWER Romulus BMC (ARM1176)";
226
- mc->init = romulus_bmc_init;
227
- mc->max_cpus = 1;
228
- mc->no_sdcard = 1;
229
- mc->no_floppy = 1;
230
- mc->no_cdrom = 1;
231
- mc->no_parallel = 1;
232
-}
233
-
234
-static const TypeInfo romulus_bmc_type = {
235
- .name = MACHINE_TYPE_NAME("romulus-bmc"),
236
- .parent = TYPE_MACHINE,
237
- .class_init = romulus_bmc_class_init,
238
-};
239
-
240
static void witherspoon_bmc_i2c_init(AspeedBoardState *bmc)
241
{
242
AspeedSoCState *soc = &bmc->soc;
243
@@ -XXX,XX +XXX,XX @@ static void witherspoon_bmc_i2c_init(AspeedBoardState *bmc)
244
0x60);
245
}
246
247
-static void witherspoon_bmc_init(MachineState *machine)
248
+static void aspeed_machine_init(MachineState *machine)
249
{
250
- aspeed_board_init(machine, &aspeed_boards[WITHERSPOON_BMC]);
251
+ AspeedMachineClass *amc = ASPEED_MACHINE_GET_CLASS(machine);
252
+
253
+ aspeed_board_init(machine, amc->board);
254
}
255
256
-static void witherspoon_bmc_class_init(ObjectClass *oc, void *data)
257
+static void aspeed_machine_class_init(ObjectClass *oc, void *data)
258
{
259
MachineClass *mc = MACHINE_CLASS(oc);
260
+ AspeedMachineClass *amc = ASPEED_MACHINE_CLASS(oc);
261
+ const AspeedBoardConfig *board = data;
262
263
- mc->desc = "OpenPOWER Witherspoon BMC (ARM1176)";
264
- mc->init = witherspoon_bmc_init;
265
+ mc->desc = board->desc;
266
+ mc->init = aspeed_machine_init;
267
mc->max_cpus = 1;
268
mc->no_sdcard = 1;
269
mc->no_floppy = 1;
270
mc->no_cdrom = 1;
271
mc->no_parallel = 1;
272
+ amc->board = board;
273
}
274
275
-static const TypeInfo witherspoon_bmc_type = {
276
- .name = MACHINE_TYPE_NAME("witherspoon-bmc"),
277
+static const TypeInfo aspeed_machine_type = {
278
+ .name = TYPE_ASPEED_MACHINE,
279
.parent = TYPE_MACHINE,
280
- .class_init = witherspoon_bmc_class_init,
281
+ .instance_size = sizeof(AspeedMachine),
282
+ .class_size = sizeof(AspeedMachineClass),
283
+ .abstract = true,
284
};
285
286
-static void aspeed_machine_init(void)
287
+static const AspeedBoardConfig aspeed_boards[] = {
288
+ {
289
+ .name = MACHINE_TYPE_NAME("palmetto-bmc"),
290
+ .desc = "OpenPOWER Palmetto BMC (ARM926EJ-S)",
291
+ .soc_name = "ast2400-a1",
292
+ .hw_strap1 = PALMETTO_BMC_HW_STRAP1,
293
+ .fmc_model = "n25q256a",
294
+ .spi_model = "mx25l25635e",
295
+ .num_cs = 1,
296
+ .i2c_init = palmetto_bmc_i2c_init,
297
+ }, {
298
+ .name = MACHINE_TYPE_NAME("ast2500-evb"),
299
+ .desc = "Aspeed AST2500 EVB (ARM1176)",
300
+ .soc_name = "ast2500-a1",
301
+ .hw_strap1 = AST2500_EVB_HW_STRAP1,
302
+ .fmc_model = "w25q256",
303
+ .spi_model = "mx25l25635e",
304
+ .num_cs = 1,
305
+ .i2c_init = ast2500_evb_i2c_init,
306
+ }, {
307
+ .name = MACHINE_TYPE_NAME("romulus-bmc"),
308
+ .desc = "OpenPOWER Romulus BMC (ARM1176)",
309
+ .soc_name = "ast2500-a1",
310
+ .hw_strap1 = ROMULUS_BMC_HW_STRAP1,
311
+ .fmc_model = "n25q256a",
312
+ .spi_model = "mx66l1g45g",
313
+ .num_cs = 2,
314
+ .i2c_init = romulus_bmc_i2c_init,
315
+ }, {
316
+ .name = MACHINE_TYPE_NAME("witherspoon-bmc"),
317
+ .desc = "OpenPOWER Witherspoon BMC (ARM1176)",
318
+ .soc_name = "ast2500-a1",
319
+ .hw_strap1 = WITHERSPOON_BMC_HW_STRAP1,
320
+ .fmc_model = "mx25l25635e",
321
+ .spi_model = "mx66l1g45g",
322
+ .num_cs = 2,
323
+ .i2c_init = witherspoon_bmc_i2c_init,
324
+ },
325
+};
121
+};
326
+
122
+
327
+static void aspeed_machine_types(void)
123
+static void zynq_machine_register_types(void)
328
{
124
+{
329
- type_register_static(&palmetto_bmc_type);
125
+ type_register_static(&zynq_machine_type);
330
- type_register_static(&ast2500_evb_type);
126
+}
331
- type_register_static(&romulus_bmc_type);
332
- type_register_static(&witherspoon_bmc_type);
333
+ int i;
334
+
127
+
335
+ type_register_static(&aspeed_machine_type);
128
+type_init(zynq_machine_register_types)
336
+ for (i = 0; i < ARRAY_SIZE(aspeed_boards); ++i) {
337
+ TypeInfo ti = {
338
+ .name = aspeed_boards[i].name,
339
+ .parent = TYPE_ASPEED_MACHINE,
340
+ .class_init = aspeed_machine_class_init,
341
+ .class_data = (void *)&aspeed_boards[i],
342
+ };
343
+ type_register(&ti);
344
+ }
345
}
346
347
-type_init(aspeed_machine_init)
348
+type_init(aspeed_machine_types)
349
--
129
--
350
2.19.0
130
2.20.1
351
131
352
132
diff view generated by jsdifflib
1
The only difference between our implementation of the pcnet ioport
1
From: Damien Hedde <damien.hedde@greensocs.com>
2
accessors and the mmio accessors is that the former check BCR_DWIO to
3
see what access widths are permitted for addresses in the aprom range
4
(0x0..0xf). In fact our failure to do this in the mmio accessors
5
is a bug (one which was fixed for the ioport accessors in
6
commit 7ba79741970 in 2011).
7
2
8
The data sheet for the Am79C970A does not describe the DWIO
3
This prints the clocks attached to a DeviceState when using
9
bit as only applying for I/O space mapped I/O resources and
4
"info qtree" monitor command. For every clock, it displays the
10
not memory mapped I/O resources, and our MMIO accessors already
5
direction, the name and if the clock is forwarded. For input clock,
11
honour DWIO for accesses in the 0x10..0x1f range (since the
6
it displays also the frequency.
12
pcnet_ioport_{read,write}{w,l} functions check it).
13
7
14
The data sheet for the later but compatible Am79C976 is clearer:
8
This is based on the original work of Frederic Konrad.
15
it states specifically "DWIO mode applies to both I/O- and
16
memory-mapped acceses." This seems to be reasonable evidence
17
in favour of interpretating the Am79C970A spec as being the same.
18
9
19
(NB: Linux's pcnet driver only supports I/O accesses, so the
10
Here follows a sample of `info qtree` output on xilinx_zynq machine
20
MMIO access part of this device is probably untested anyway.)
11
after linux boot with only one uart clocked:
12
> bus: main-system-bus
13
> type System
14
> [...]
15
> dev: cadence_uart, id ""
16
> gpio-out "sysbus-irq" 1
17
> clock-in "refclk" freq_hz=0.000000e+00
18
> chardev = ""
19
> mmio 00000000e0001000/0000000000001000
20
> dev: cadence_uart, id ""
21
> gpio-out "sysbus-irq" 1
22
> clock-in "refclk" freq_hz=1.375661e+07
23
> chardev = "serial0"
24
> mmio 00000000e0000000/0000000000001000
25
> [...]
26
> dev: xilinx,zynq_slcr, id ""
27
> clock-out "uart1_ref_clk" freq_hz=0.000000e+00
28
> clock-out "uart0_ref_clk" freq_hz=1.375661e+07
29
> clock-in "ps_clk" freq_hz=3.333333e+07
30
> mmio 00000000f8000000/0000000000001000
21
31
22
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
32
Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
33
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
34
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
35
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
36
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
37
Message-id: 20200406135251.157596-10-damien.hedde@greensocs.com
23
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
38
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
24
---
39
---
25
hw/net/pcnet-pci.c | 67 ++-------------------------------------------
40
qdev-monitor.c | 9 +++++++++
26
hw/net/trace-events | 2 --
41
1 file changed, 9 insertions(+)
27
2 files changed, 2 insertions(+), 67 deletions(-)
28
42
29
diff --git a/hw/net/pcnet-pci.c b/hw/net/pcnet-pci.c
43
diff --git a/qdev-monitor.c b/qdev-monitor.c
30
index XXXXXXX..XXXXXXX 100644
44
index XXXXXXX..XXXXXXX 100644
31
--- a/hw/net/pcnet-pci.c
45
--- a/qdev-monitor.c
32
+++ b/hw/net/pcnet-pci.c
46
+++ b/qdev-monitor.c
33
@@ -XXX,XX +XXX,XX @@ static const MemoryRegionOps pcnet_io_ops = {
47
@@ -XXX,XX +XXX,XX @@
34
.endianness = DEVICE_LITTLE_ENDIAN,
48
#include "migration/misc.h"
35
};
49
#include "migration/migration.h"
36
50
#include "qemu/cutils.h"
37
-/*
51
+#include "hw/clock.h"
38
- * TODO: should MMIO accesses to the addresses corresponding to the
52
39
- * APROM also honour the BCR_DWIO() setting? If so, then these functions
53
/*
40
- * and pcnet_ioport_write/pcnet_ioport_read could be merged.
54
* Aliases were a bad idea from the start. Let's keep them
41
- * If not, then should pcnet_ioport_{read,write}{w,l} really check
55
@@ -XXX,XX +XXX,XX @@ static void qdev_print(Monitor *mon, DeviceState *dev, int indent)
42
- * BCR_DWIO() for MMIO writes ?
56
ObjectClass *class;
43
- */
57
BusState *child;
44
-static void pcnet_mmio_write(void *opaque, hwaddr addr, uint64_t value,
58
NamedGPIOList *ngl;
45
- unsigned size)
59
+ NamedClockList *ncl;
46
-{
60
47
- PCNetState *d = opaque;
61
qdev_printf("dev: %s, id \"%s\"\n", object_get_typename(OBJECT(dev)),
48
-
62
dev->id ? dev->id : "");
49
- trace_pcnet_mmio_write(opaque, addr, size, val);
63
@@ -XXX,XX +XXX,XX @@ static void qdev_print(Monitor *mon, DeviceState *dev, int indent)
50
-
64
ngl->num_out);
51
- if (addr < 0x10) {
65
}
52
- if (size == 1) {
66
}
53
- pcnet_aprom_writeb(d, addr, data);
67
+ QLIST_FOREACH(ncl, &dev->clocks, node) {
54
- } else if ((addr & 1) == 0 && size == 2) {
68
+ qdev_printf("clock-%s%s \"%s\" freq_hz=%e\n",
55
- pcnet_aprom_writeb(d, addr, data & 0xff);
69
+ ncl->output ? "out" : "in",
56
- pcnet_aprom_writeb(d, addr + 1, data >> 8);
70
+ ncl->alias ? " (alias)" : "",
57
- } else if ((addr & 3) == 0 && size == 4) {
71
+ ncl->name,
58
- pcnet_aprom_writeb(d, addr, data & 0xff);
72
+ CLOCK_PERIOD_TO_HZ(1.0 * clock_get(ncl->clock)));
59
- pcnet_aprom_writeb(d, addr + 1, (data >> 8) & 0xff);
73
+ }
60
- pcnet_aprom_writeb(d, addr + 2, (data >> 16) & 0xff);
74
class = object_get_class(OBJECT(dev));
61
- pcnet_aprom_writeb(d, addr + 3, data >> 24);
75
do {
62
- }
76
qdev_print_props(mon, dev, DEVICE_CLASS(class)->props_, indent);
63
- } else {
64
- if (size == 2) {
65
- pcnet_ioport_writew(d, addr, data);
66
- } else if (size == 4) {
67
- pcnet_ioport_writel(d, addr, data);
68
- }
69
- }
70
-}
71
-
72
-static uint64_t pcnet_mmio_read(void *opque, hwaddr addr, unsigned size)
73
-{
74
- PCNetState *d = opaque;
75
-
76
- trace_pcnet_ioport_read(opaque, addr, size);
77
-
78
- if (addr < 0x10) {
79
- if (size == 1) {
80
- return pcnet_aprom_readb(d, addr);
81
- } else if ((addr & 1) == 0 && size == 2) {
82
- return pcnet_aprom_readb(d, addr) |
83
- (pcnet_aprom_readb(d, addr + 1) << 8);
84
- } else if ((addr & 3) == 0 && size == 4) {
85
- return pcnet_aprom_readb(d, addr) |
86
- (pcnet_aprom_readb(d, addr + 1) << 8) |
87
- (pcnet_aprom_readb(d, addr + 2) << 16) |
88
- (pcnet_aprom_readb(d, addr + 3) << 24);
89
- }
90
- } else {
91
- if (size == 2) {
92
- return pcnet_ioport_readw(d, addr);
93
- } else if (size == 4) {
94
- return pcnet_ioport_readl(d, addr);
95
- }
96
- }
97
- return ((uint64_t)1 << (size * 8)) - 1;
98
-}
99
-
100
static const VMStateDescription vmstate_pci_pcnet = {
101
.name = "pcnet",
102
.version_id = 3,
103
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_pci_pcnet = {
104
/* PCI interface */
105
106
static const MemoryRegionOps pcnet_mmio_ops = {
107
- .read = pcnet_mmio_read,
108
- .write = pcnet_mmio_write,
109
+ .read = pcnet_ioport_read,
110
+ .write = pcnet_ioport_write,
111
.valid.min_access_size = 1,
112
.valid.max_access_size = 4,
113
.impl.min_access_size = 1,
114
diff --git a/hw/net/trace-events b/hw/net/trace-events
115
index XXXXXXX..XXXXXXX 100644
116
--- a/hw/net/trace-events
117
+++ b/hw/net/trace-events
118
@@ -XXX,XX +XXX,XX @@ pcnet_aprom_writeb(void *opaque, uint32_t addr, uint32_t val) "opaque=%p addr=0x
119
pcnet_aprom_readb(void *opaque, uint32_t addr, uint32_t val) "opaque=%p addr=0x%08x val=0x%02x"
120
pcnet_ioport_read(void *opaque, uint64_t addr, unsigned size) "opaque=%p addr=0x%"PRIx64" size=%d"
121
pcnet_ioport_write(void *opaque, uint64_t addr, uint64_t data, unsigned size) "opaque=%p addr=0x%"PRIx64" data=0x%"PRIx64" size=%d"
122
-pcnet_mmio_write(void *opaque, uint64_t addr, uint32_t val, unsigned size) "opaque=%p addr=0x%"PRIx64" val=0x%x size=%d"
123
-pcnet_mmio_read(void *opaque, uint64_t addr, unsigned size) "opaque=%p addr=0x%"PRIx64" size=%d"
124
125
# hw/net/net_rx_pkt.c
126
net_rx_pkt_parsed(bool ip4, bool ip6, bool udp, bool tcp, size_t l3o, size_t l4o, size_t l5o) "RX packet parsed: ip4: %d, ip6: %d, udp: %d, tcp: %d, l3 offset: %zu, l4 offset: %zu, l5 offset: %zu"
127
--
77
--
128
2.19.0
78
2.20.1
129
79
130
80
diff view generated by jsdifflib
1
Add 'break' statements missing from a switch in the APB dual-timer
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
write function. Spotted by Coverity as CID 1395626 and 1395633.
3
2
4
Reported-by: Paolo Bonzini <pbonzini@redhat.com>
3
Setup the ADMA with 128bit bus-width. This matters when
4
FIXED BURST mode is used.
5
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
9
Message-id: 20200417153800.27399-2-edgar.iglesias@gmail.com
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Message-id: 20180924123122.14549-1-peter.maydell@linaro.org
8
---
11
---
9
hw/timer/cmsdk-apb-dualtimer.c | 2 ++
12
hw/arm/xlnx-versal.c | 2 ++
10
1 file changed, 2 insertions(+)
13
1 file changed, 2 insertions(+)
11
14
12
diff --git a/hw/timer/cmsdk-apb-dualtimer.c b/hw/timer/cmsdk-apb-dualtimer.c
15
diff --git a/hw/arm/xlnx-versal.c b/hw/arm/xlnx-versal.c
13
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/timer/cmsdk-apb-dualtimer.c
17
--- a/hw/arm/xlnx-versal.c
15
+++ b/hw/timer/cmsdk-apb-dualtimer.c
18
+++ b/hw/arm/xlnx-versal.c
16
@@ -XXX,XX +XXX,XX @@ static void cmsdk_apb_dualtimer_write(void *opaque, hwaddr offset,
19
@@ -XXX,XX +XXX,XX @@ static void versal_create_admas(Versal *s, qemu_irq *pic)
17
case A_TIMERITCR:
20
18
s->timeritcr = value & R_TIMERITCR_VALID_MASK;
21
dev = qdev_create(NULL, "xlnx.zdma");
19
cmsdk_apb_dualtimer_update(s);
22
s->lpd.iou.adma[i] = SYS_BUS_DEVICE(dev);
20
+ break;
23
+ object_property_set_int(OBJECT(s->lpd.iou.adma[i]), 128, "bus-width",
21
case A_TIMERITOP:
24
+ &error_abort);
22
s->timeritop = value & R_TIMERITOP_VALID_MASK;
25
object_property_add_child(OBJECT(s), name, OBJECT(dev), &error_fatal);
23
cmsdk_apb_dualtimer_update(s);
26
qdev_init_nofail(dev);
24
+ break;
27
25
default:
26
bad_offset:
27
qemu_log_mask(LOG_GUEST_ERROR,
28
--
28
--
29
2.19.0
29
2.20.1
30
30
31
31
diff view generated by jsdifflib
1
The GIC_BASE_IRQ macro is a leftover from when we shared code
1
From: Ramon Fried <rfried.dev@gmail.com>
2
between the GICv2 and the v7M NVIC. Since the NVIC is now
3
split off, GIC_BASE_IRQ is always 0, and we can just delete it.
4
2
3
Wraparound of TX descriptor cyclic buffer only updated
4
the low 32 bits of the descriptor.
5
Fix that by checking if we're working with 64bit descriptors.
6
7
Signed-off-by: Ramon Fried <rfried.dev@gmail.com>
8
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Message-id: 20200417171736.441607-1-rfried.dev@gmail.com
5
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
6
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180824161819.11085-1-peter.maydell@linaro.org
9
---
11
---
10
hw/intc/gic_internal.h | 2 --
12
hw/net/cadence_gem.c | 9 ++++++++-
11
hw/intc/arm_gic.c | 31 ++++++++++++++-----------------
13
1 file changed, 8 insertions(+), 1 deletion(-)
12
hw/intc/arm_gic_common.c | 1 -
13
3 files changed, 14 insertions(+), 20 deletions(-)
14
14
15
diff --git a/hw/intc/gic_internal.h b/hw/intc/gic_internal.h
15
diff --git a/hw/net/cadence_gem.c b/hw/net/cadence_gem.c
16
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
17
--- a/hw/intc/gic_internal.h
17
--- a/hw/net/cadence_gem.c
18
+++ b/hw/intc/gic_internal.h
18
+++ b/hw/net/cadence_gem.c
19
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ static void gem_transmit(CadenceGEMState *s)
20
20
/* read next descriptor */
21
#define ALL_CPU_MASK ((unsigned)(((1 << GIC_NCPU) - 1)))
21
if (tx_desc_get_wrap(desc)) {
22
22
tx_desc_set_last(desc);
23
-#define GIC_BASE_IRQ 0
23
- packet_desc_addr = s->regs[GEM_TXQBASE];
24
-
24
+
25
#define GIC_DIST_SET_ENABLED(irq, cm) (s->irq_state[irq].enabled |= (cm))
25
+ if (s->regs[GEM_DMACFG] & GEM_DMACFG_ADDR_64B) {
26
#define GIC_DIST_CLEAR_ENABLED(irq, cm) (s->irq_state[irq].enabled &= ~(cm))
26
+ packet_desc_addr = s->regs[GEM_TBQPH];
27
#define GIC_DIST_TEST_ENABLED(irq, cm) ((s->irq_state[irq].enabled & (cm)) != 0)
27
+ packet_desc_addr <<= 32;
28
diff --git a/hw/intc/arm_gic.c b/hw/intc/arm_gic.c
28
+ } else {
29
index XXXXXXX..XXXXXXX 100644
29
+ packet_desc_addr = 0;
30
--- a/hw/intc/arm_gic.c
30
+ }
31
+++ b/hw/intc/arm_gic.c
31
+ packet_desc_addr |= s->regs[GEM_TXQBASE];
32
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
32
} else {
33
res = 0;
33
packet_desc_addr += 4 * gem_get_desc_len(s, false);
34
if (!(s->security_extn && !attrs.secure) && gic_has_groups(s)) {
35
/* Every byte offset holds 8 group status bits */
36
- irq = (offset - 0x080) * 8 + GIC_BASE_IRQ;
37
+ irq = (offset - 0x080) * 8;
38
if (irq >= s->num_irq) {
39
goto bad_reg;
40
}
41
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
42
irq = (offset - 0x100) * 8;
43
else
44
irq = (offset - 0x180) * 8;
45
- irq += GIC_BASE_IRQ;
46
if (irq >= s->num_irq)
47
goto bad_reg;
48
res = 0;
49
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
50
irq = (offset - 0x200) * 8;
51
else
52
irq = (offset - 0x280) * 8;
53
- irq += GIC_BASE_IRQ;
54
if (irq >= s->num_irq)
55
goto bad_reg;
56
res = 0;
57
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
58
goto bad_reg;
59
}
60
61
- irq += GIC_BASE_IRQ;
62
if (irq >= s->num_irq)
63
goto bad_reg;
64
res = 0;
65
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
66
}
67
} else if (offset < 0x800) {
68
/* Interrupt Priority. */
69
- irq = (offset - 0x400) + GIC_BASE_IRQ;
70
+ irq = (offset - 0x400);
71
if (irq >= s->num_irq)
72
goto bad_reg;
73
res = gic_dist_get_priority(s, cpu, irq, attrs);
74
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
75
/* For uniprocessor GICs these RAZ/WI */
76
res = 0;
77
} else {
78
- irq = (offset - 0x800) + GIC_BASE_IRQ;
79
+ irq = (offset - 0x800);
80
if (irq >= s->num_irq) {
81
goto bad_reg;
82
}
34
}
83
@@ -XXX,XX +XXX,XX @@ static uint32_t gic_dist_readb(void *opaque, hwaddr offset, MemTxAttrs attrs)
84
}
85
} else if (offset < 0xf00) {
86
/* Interrupt Configuration. */
87
- irq = (offset - 0xc00) * 4 + GIC_BASE_IRQ;
88
+ irq = (offset - 0xc00) * 4;
89
if (irq >= s->num_irq)
90
goto bad_reg;
91
res = 0;
92
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
93
*/
94
if (!(s->security_extn && !attrs.secure) && gic_has_groups(s)) {
95
/* Every byte offset holds 8 group status bits */
96
- irq = (offset - 0x80) * 8 + GIC_BASE_IRQ;
97
+ irq = (offset - 0x80) * 8;
98
if (irq >= s->num_irq) {
99
goto bad_reg;
100
}
101
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
102
}
103
} else if (offset < 0x180) {
104
/* Interrupt Set Enable. */
105
- irq = (offset - 0x100) * 8 + GIC_BASE_IRQ;
106
+ irq = (offset - 0x100) * 8;
107
if (irq >= s->num_irq)
108
goto bad_reg;
109
if (irq < GIC_NR_SGIS) {
110
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
111
}
112
} else if (offset < 0x200) {
113
/* Interrupt Clear Enable. */
114
- irq = (offset - 0x180) * 8 + GIC_BASE_IRQ;
115
+ irq = (offset - 0x180) * 8;
116
if (irq >= s->num_irq)
117
goto bad_reg;
118
if (irq < GIC_NR_SGIS) {
119
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
120
}
121
} else if (offset < 0x280) {
122
/* Interrupt Set Pending. */
123
- irq = (offset - 0x200) * 8 + GIC_BASE_IRQ;
124
+ irq = (offset - 0x200) * 8;
125
if (irq >= s->num_irq)
126
goto bad_reg;
127
if (irq < GIC_NR_SGIS) {
128
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
129
}
130
} else if (offset < 0x300) {
131
/* Interrupt Clear Pending. */
132
- irq = (offset - 0x280) * 8 + GIC_BASE_IRQ;
133
+ irq = (offset - 0x280) * 8;
134
if (irq >= s->num_irq)
135
goto bad_reg;
136
if (irq < GIC_NR_SGIS) {
137
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
138
goto bad_reg;
139
}
140
141
- irq = (offset - 0x300) * 8 + GIC_BASE_IRQ;
142
+ irq = (offset - 0x300) * 8;
143
if (irq >= s->num_irq) {
144
goto bad_reg;
145
}
146
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
147
goto bad_reg;
148
}
149
150
- irq = (offset - 0x380) * 8 + GIC_BASE_IRQ;
151
+ irq = (offset - 0x380) * 8;
152
if (irq >= s->num_irq) {
153
goto bad_reg;
154
}
155
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
156
}
157
} else if (offset < 0x800) {
158
/* Interrupt Priority. */
159
- irq = (offset - 0x400) + GIC_BASE_IRQ;
160
+ irq = (offset - 0x400);
161
if (irq >= s->num_irq)
162
goto bad_reg;
163
gic_dist_set_priority(s, cpu, irq, value, attrs);
164
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
165
* annoying exception of the 11MPCore's GIC.
166
*/
167
if (s->num_cpu != 1 || s->revision == REV_11MPCORE) {
168
- irq = (offset - 0x800) + GIC_BASE_IRQ;
169
+ irq = (offset - 0x800);
170
if (irq >= s->num_irq) {
171
goto bad_reg;
172
}
173
@@ -XXX,XX +XXX,XX @@ static void gic_dist_writeb(void *opaque, hwaddr offset,
174
}
175
} else if (offset < 0xf00) {
176
/* Interrupt Configuration. */
177
- irq = (offset - 0xc00) * 4 + GIC_BASE_IRQ;
178
+ irq = (offset - 0xc00) * 4;
179
if (irq >= s->num_irq)
180
goto bad_reg;
181
if (irq < GIC_NR_SGIS)
182
diff --git a/hw/intc/arm_gic_common.c b/hw/intc/arm_gic_common.c
183
index XXXXXXX..XXXXXXX 100644
184
--- a/hw/intc/arm_gic_common.c
185
+++ b/hw/intc/arm_gic_common.c
186
@@ -XXX,XX +XXX,XX @@ static void arm_gic_common_realize(DeviceState *dev, Error **errp)
187
s->num_cpu, GIC_NCPU);
188
return;
189
}
190
- s->num_irq += GIC_BASE_IRQ;
191
if (s->num_irq > GIC_MAXIRQ) {
192
error_setg(errp,
193
"requested %u interrupt lines exceeds GIC maximum %d",
194
--
35
--
195
2.19.0
36
2.20.1
196
37
197
38
diff view generated by jsdifflib
1
From: Cédric Le Goater <clg@kaod.org>
1
From: Ramon Fried <rfried.dev@gmail.com>
2
2
3
aspeed i2c interrupts should be cleared by software only, and the bus
3
The RX ring descriptors control field is used for setting
4
interrupt should be lowered when all interrupts have been cleared.
4
SOF and EOF (start of frame and end of frame).
5
The SOF and EOF weren't cleared from the previous descriptors,
6
causing inconsistencies in ring buffer.
7
Fix that by clearing the control field of every descriptors we're
8
processing.
5
9
6
Signed-off-by: Cédric Le Goater <clg@kaod.org>
10
Signed-off-by: Ramon Fried <rfried.dev@gmail.com>
7
Message-id: 20180914063506.20815-2-clg@kaod.org
11
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
[PMM: drop TODO comment describing an issue which is
13
Message-id: 20200418085145.489726-1-rfried.dev@gmail.com
10
fixed later in the patch series, and clean up commit message]
11
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
12
---
15
---
13
hw/i2c/aspeed_i2c.c | 16 ++++++++++++----
16
hw/net/cadence_gem.c | 7 +++++++
14
1 file changed, 12 insertions(+), 4 deletions(-)
17
1 file changed, 7 insertions(+)
15
18
16
diff --git a/hw/i2c/aspeed_i2c.c b/hw/i2c/aspeed_i2c.c
19
diff --git a/hw/net/cadence_gem.c b/hw/net/cadence_gem.c
17
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
18
--- a/hw/i2c/aspeed_i2c.c
21
--- a/hw/net/cadence_gem.c
19
+++ b/hw/i2c/aspeed_i2c.c
22
+++ b/hw/net/cadence_gem.c
20
@@ -XXX,XX +XXX,XX @@
23
@@ -XXX,XX +XXX,XX @@ static inline void rx_desc_set_sof(uint32_t *desc)
21
#define I2CD_AC_TIMING_REG2 0x08 /* Clock and AC Timing Control #1 */
24
desc[1] |= DESC_1_RX_SOF;
22
#define I2CD_INTR_CTRL_REG 0x0c /* I2CD Interrupt Control */
25
}
23
#define I2CD_INTR_STS_REG 0x10 /* I2CD Interrupt Status */
26
27
+static inline void rx_desc_clear_control(uint32_t *desc)
28
+{
29
+ desc[1] = 0;
30
+}
24
+
31
+
25
+#define I2CD_INTR_SLAVE_ADDR_MATCH (0x1 << 31) /* 0: addr1 1: addr2 */
32
static inline void rx_desc_set_eof(uint32_t *desc)
26
+#define I2CD_INTR_SLAVE_ADDR_RX_PENDING (0x1 << 30)
33
{
27
+/* bits[19-16] Reserved */
34
desc[1] |= DESC_1_RX_EOF;
35
@@ -XXX,XX +XXX,XX @@ static ssize_t gem_receive(NetClientState *nc, const uint8_t *buf, size_t size)
36
rxbuf_ptr += MIN(bytes_to_copy, rxbufsize);
37
bytes_to_copy -= MIN(bytes_to_copy, rxbufsize);
38
39
+ rx_desc_clear_control(s->rx_desc[q]);
28
+
40
+
29
+/* All bits below are cleared by writing 1 */
41
/* Update the descriptor. */
30
+#define I2CD_INTR_SLAVE_INACTIVE_TIMEOUT (0x1 << 15)
42
if (first_desc) {
31
#define I2CD_INTR_SDA_DL_TIMEOUT (0x1 << 14)
43
rx_desc_set_sof(s->rx_desc[q]);
32
#define I2CD_INTR_BUS_RECOVER_DONE (0x1 << 13)
33
#define I2CD_INTR_SMBUS_ALERT (0x1 << 12) /* Bus [0-3] only */
34
@@ -XXX,XX +XXX,XX @@
35
#define I2CD_INTR_SMBUS_DEV_ALERT_ADDR (0x1 << 10) /* Removed */
36
#define I2CD_INTR_SMBUS_DEF_ADDR (0x1 << 9) /* Removed */
37
#define I2CD_INTR_GCALL_ADDR (0x1 << 8) /* Removed */
38
-#define I2CD_INTR_SLAVE_MATCH (0x1 << 7) /* use RX_DONE */
39
+#define I2CD_INTR_SLAVE_ADDR_RX_MATCH (0x1 << 7) /* use RX_DONE */
40
#define I2CD_INTR_SCL_TIMEOUT (0x1 << 6)
41
#define I2CD_INTR_ABNORMAL (0x1 << 5)
42
#define I2CD_INTR_NORMAL_STOP (0x1 << 4)
43
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_handle_cmd(AspeedI2CBus *bus, uint64_t value)
44
{
45
bus->cmd &= ~0xFFFF;
46
bus->cmd |= value & 0xFFFF;
47
- bus->intr_status = 0;
48
49
if (bus->cmd & I2CD_M_START_CMD) {
50
uint8_t state = aspeed_i2c_get_state(bus) & I2CD_MACTIVE ?
51
@@ -XXX,XX +XXX,XX @@ static void aspeed_i2c_bus_write(void *opaque, hwaddr offset,
52
break;
53
case I2CD_INTR_STS_REG:
54
bus->intr_status &= ~(value & 0x7FFF);
55
- bus->controller->intr_status &= ~(1 << bus->id);
56
- qemu_irq_lower(bus->controller->irq);
57
+ if (!bus->intr_status) {
58
+ bus->controller->intr_status &= ~(1 << bus->id);
59
+ qemu_irq_lower(bus->controller->irq);
60
+ }
61
break;
62
case I2CD_DEV_ADDR_REG:
63
qemu_log_mask(LOG_UNIMP, "%s: slave mode not implemented\n",
64
--
44
--
65
2.19.0
45
2.20.1
66
46
67
47
diff view generated by jsdifflib
1
From: Cédric Le Goater <clg@kaod.org>
1
From: Richard Henderson <richard.henderson@linaro.org>
2
2
3
Signed-off-by: Cédric Le Goater <clg@kaod.org>
3
These instructions are often used in glibc's string routines.
4
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
4
They were the final uses of the 32-bit at a time neon helpers.
5
Message-id: 20180921161939.822-6-clg@kaod.org
5
6
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
7
Message-id: 20200418162808.4680-1-richard.henderson@linaro.org
6
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
---
10
---
9
hw/ssi/aspeed_smc.c | 8 ++++----
11
target/arm/helper.h | 27 ++--
10
1 file changed, 4 insertions(+), 4 deletions(-)
12
target/arm/translate.h | 5 +
13
target/arm/neon_helper.c | 24 ----
14
target/arm/translate-a64.c | 64 +++-------
15
target/arm/translate.c | 256 +++++++++++++++++++++++++++++++------
16
target/arm/vec_helper.c | 25 ++++
17
6 files changed, 278 insertions(+), 123 deletions(-)
11
18
12
diff --git a/hw/ssi/aspeed_smc.c b/hw/ssi/aspeed_smc.c
19
diff --git a/target/arm/helper.h b/target/arm/helper.h
13
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
14
--- a/hw/ssi/aspeed_smc.c
21
--- a/target/arm/helper.h
15
+++ b/hw/ssi/aspeed_smc.c
22
+++ b/target/arm/helper.h
16
@@ -XXX,XX +XXX,XX @@ static uint64_t aspeed_smc_flash_default_read(void *opaque, hwaddr addr,
23
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_2(neon_hsub_u16, i32, i32, i32)
17
static void aspeed_smc_flash_default_write(void *opaque, hwaddr addr,
24
DEF_HELPER_2(neon_hsub_s32, s32, s32, s32)
18
uint64_t data, unsigned size)
25
DEF_HELPER_2(neon_hsub_u32, i32, i32, i32)
26
27
-DEF_HELPER_2(neon_cgt_u8, i32, i32, i32)
28
-DEF_HELPER_2(neon_cgt_s8, i32, i32, i32)
29
-DEF_HELPER_2(neon_cgt_u16, i32, i32, i32)
30
-DEF_HELPER_2(neon_cgt_s16, i32, i32, i32)
31
-DEF_HELPER_2(neon_cgt_u32, i32, i32, i32)
32
-DEF_HELPER_2(neon_cgt_s32, i32, i32, i32)
33
-DEF_HELPER_2(neon_cge_u8, i32, i32, i32)
34
-DEF_HELPER_2(neon_cge_s8, i32, i32, i32)
35
-DEF_HELPER_2(neon_cge_u16, i32, i32, i32)
36
-DEF_HELPER_2(neon_cge_s16, i32, i32, i32)
37
-DEF_HELPER_2(neon_cge_u32, i32, i32, i32)
38
-DEF_HELPER_2(neon_cge_s32, i32, i32, i32)
39
-
40
DEF_HELPER_2(neon_pmin_u8, i32, i32, i32)
41
DEF_HELPER_2(neon_pmin_s8, i32, i32, i32)
42
DEF_HELPER_2(neon_pmin_u16, i32, i32, i32)
43
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_2(neon_mul_u16, i32, i32, i32)
44
DEF_HELPER_2(neon_tst_u8, i32, i32, i32)
45
DEF_HELPER_2(neon_tst_u16, i32, i32, i32)
46
DEF_HELPER_2(neon_tst_u32, i32, i32, i32)
47
-DEF_HELPER_2(neon_ceq_u8, i32, i32, i32)
48
-DEF_HELPER_2(neon_ceq_u16, i32, i32, i32)
49
-DEF_HELPER_2(neon_ceq_u32, i32, i32, i32)
50
51
DEF_HELPER_1(neon_clz_u8, i32, i32)
52
DEF_HELPER_1(neon_clz_u16, i32, i32)
53
@@ -XXX,XX +XXX,XX @@ DEF_HELPER_FLAGS_2(frint64_s, TCG_CALL_NO_RWG, f32, f32, ptr)
54
DEF_HELPER_FLAGS_2(frint32_d, TCG_CALL_NO_RWG, f64, f64, ptr)
55
DEF_HELPER_FLAGS_2(frint64_d, TCG_CALL_NO_RWG, f64, f64, ptr)
56
57
+DEF_HELPER_FLAGS_3(gvec_ceq0_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
58
+DEF_HELPER_FLAGS_3(gvec_ceq0_h, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
59
+DEF_HELPER_FLAGS_3(gvec_clt0_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
60
+DEF_HELPER_FLAGS_3(gvec_clt0_h, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
61
+DEF_HELPER_FLAGS_3(gvec_cle0_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
62
+DEF_HELPER_FLAGS_3(gvec_cle0_h, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
63
+DEF_HELPER_FLAGS_3(gvec_cgt0_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
64
+DEF_HELPER_FLAGS_3(gvec_cgt0_h, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
65
+DEF_HELPER_FLAGS_3(gvec_cge0_b, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
66
+DEF_HELPER_FLAGS_3(gvec_cge0_h, TCG_CALL_NO_RWG, void, ptr, ptr, i32)
67
+
68
DEF_HELPER_FLAGS_4(gvec_sshl_b, TCG_CALL_NO_RWG, void, ptr, ptr, ptr, i32)
69
DEF_HELPER_FLAGS_4(gvec_sshl_h, TCG_CALL_NO_RWG, void, ptr, ptr, ptr, i32)
70
DEF_HELPER_FLAGS_4(gvec_ushl_b, TCG_CALL_NO_RWG, void, ptr, ptr, ptr, i32)
71
diff --git a/target/arm/translate.h b/target/arm/translate.h
72
index XXXXXXX..XXXXXXX 100644
73
--- a/target/arm/translate.h
74
+++ b/target/arm/translate.h
75
@@ -XXX,XX +XXX,XX @@ static inline void gen_swstep_exception(DisasContext *s, int isv, int ex)
76
uint64_t vfp_expand_imm(int size, uint8_t imm8);
77
78
/* Vector operations shared between ARM and AArch64. */
79
+extern const GVecGen2 ceq0_op[4];
80
+extern const GVecGen2 clt0_op[4];
81
+extern const GVecGen2 cgt0_op[4];
82
+extern const GVecGen2 cle0_op[4];
83
+extern const GVecGen2 cge0_op[4];
84
extern const GVecGen3 mla_op[4];
85
extern const GVecGen3 mls_op[4];
86
extern const GVecGen3 cmtst_op[4];
87
diff --git a/target/arm/neon_helper.c b/target/arm/neon_helper.c
88
index XXXXXXX..XXXXXXX 100644
89
--- a/target/arm/neon_helper.c
90
+++ b/target/arm/neon_helper.c
91
@@ -XXX,XX +XXX,XX @@ uint32_t HELPER(neon_hsub_u32)(uint32_t src1, uint32_t src2)
92
return dest;
93
}
94
95
-#define NEON_FN(dest, src1, src2) dest = (src1 > src2) ? ~0 : 0
96
-NEON_VOP(cgt_s8, neon_s8, 4)
97
-NEON_VOP(cgt_u8, neon_u8, 4)
98
-NEON_VOP(cgt_s16, neon_s16, 2)
99
-NEON_VOP(cgt_u16, neon_u16, 2)
100
-NEON_VOP(cgt_s32, neon_s32, 1)
101
-NEON_VOP(cgt_u32, neon_u32, 1)
102
-#undef NEON_FN
103
-
104
-#define NEON_FN(dest, src1, src2) dest = (src1 >= src2) ? ~0 : 0
105
-NEON_VOP(cge_s8, neon_s8, 4)
106
-NEON_VOP(cge_u8, neon_u8, 4)
107
-NEON_VOP(cge_s16, neon_s16, 2)
108
-NEON_VOP(cge_u16, neon_u16, 2)
109
-NEON_VOP(cge_s32, neon_s32, 1)
110
-NEON_VOP(cge_u32, neon_u32, 1)
111
-#undef NEON_FN
112
-
113
#define NEON_FN(dest, src1, src2) dest = (src1 < src2) ? src1 : src2
114
NEON_POP(pmin_s8, neon_s8, 4)
115
NEON_POP(pmin_u8, neon_u8, 4)
116
@@ -XXX,XX +XXX,XX @@ NEON_VOP(tst_u16, neon_u16, 2)
117
NEON_VOP(tst_u32, neon_u32, 1)
118
#undef NEON_FN
119
120
-#define NEON_FN(dest, src1, src2) dest = (src1 == src2) ? -1 : 0
121
-NEON_VOP(ceq_u8, neon_u8, 4)
122
-NEON_VOP(ceq_u16, neon_u16, 2)
123
-NEON_VOP(ceq_u32, neon_u32, 1)
124
-#undef NEON_FN
125
-
126
/* Count Leading Sign/Zero Bits. */
127
static inline int do_clz8(uint8_t x)
19
{
128
{
20
- qemu_log_mask(LOG_GUEST_ERROR, "%s: To 0x%" HWADDR_PRIx " of size %u: 0x%"
129
diff --git a/target/arm/translate-a64.c b/target/arm/translate-a64.c
21
- PRIx64 "\n", __func__, addr, size, data);
130
index XXXXXXX..XXXXXXX 100644
22
+ qemu_log_mask(LOG_GUEST_ERROR, "%s: To 0x%" HWADDR_PRIx " of size %u: 0x%"
131
--- a/target/arm/translate-a64.c
23
+ PRIx64 "\n", __func__, addr, size, data);
132
+++ b/target/arm/translate-a64.c
133
@@ -XXX,XX +XXX,XX @@ static void gen_gvec_fn4(DisasContext *s, bool is_q, int rd, int rn, int rm,
134
is_q ? 16 : 8, vec_full_reg_size(s));
24
}
135
}
25
136
26
static const MemoryRegionOps aspeed_smc_flash_default_ops = {
137
+/* Expand a 2-operand AdvSIMD vector operation using an op descriptor. */
27
@@ -XXX,XX +XXX,XX @@ static void aspeed_smc_flash_setup(AspeedSMCFlash *fl, uint32_t addr)
138
+static void gen_gvec_op2(DisasContext *s, bool is_q, int rd,
28
*/
139
+ int rn, const GVecGen2 *gvec_op)
29
if (aspeed_smc_flash_mode(fl) == CTRL_FREADMODE) {
140
+{
30
for (i = 0; i < aspeed_smc_flash_dummies(fl); i++) {
141
+ tcg_gen_gvec_2(vec_full_reg_offset(s, rd), vec_full_reg_offset(s, rn),
31
- ssi_transfer(fl->controller->spi, 0xFF);
142
+ is_q ? 16 : 8, vec_full_reg_size(s), gvec_op);
32
+ ssi_transfer(fl->controller->spi, 0xFF);
143
+}
144
+
145
/* Expand a 2-operand + immediate AdvSIMD vector operation using
146
* an op descriptor.
147
*/
148
@@ -XXX,XX +XXX,XX @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
149
return;
33
}
150
}
151
break;
152
+ case 0x8: /* CMGT, CMGE */
153
+ gen_gvec_op2(s, is_q, rd, rn, u ? &cge0_op[size] : &cgt0_op[size]);
154
+ return;
155
+ case 0x9: /* CMEQ, CMLE */
156
+ gen_gvec_op2(s, is_q, rd, rn, u ? &cle0_op[size] : &ceq0_op[size]);
157
+ return;
158
+ case 0xa: /* CMLT */
159
+ gen_gvec_op2(s, is_q, rd, rn, &clt0_op[size]);
160
+ return;
161
case 0xb:
162
if (u) { /* ABS, NEG */
163
gen_gvec_fn2(s, is_q, rd, rn, tcg_gen_gvec_neg, size);
164
@@ -XXX,XX +XXX,XX @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
165
for (pass = 0; pass < (is_q ? 4 : 2); pass++) {
166
TCGv_i32 tcg_op = tcg_temp_new_i32();
167
TCGv_i32 tcg_res = tcg_temp_new_i32();
168
- TCGCond cond;
169
170
read_vec_element_i32(s, tcg_op, rn, pass, MO_32);
171
172
if (size == 2) {
173
/* Special cases for 32 bit elements */
174
switch (opcode) {
175
- case 0xa: /* CMLT */
176
- /* 32 bit integer comparison against zero, result is
177
- * test ? (2^32 - 1) : 0. We implement via setcond(test)
178
- * and inverting.
179
- */
180
- cond = TCG_COND_LT;
181
- do_cmop:
182
- tcg_gen_setcondi_i32(cond, tcg_res, tcg_op, 0);
183
- tcg_gen_neg_i32(tcg_res, tcg_res);
184
- break;
185
- case 0x8: /* CMGT, CMGE */
186
- cond = u ? TCG_COND_GE : TCG_COND_GT;
187
- goto do_cmop;
188
- case 0x9: /* CMEQ, CMLE */
189
- cond = u ? TCG_COND_LE : TCG_COND_EQ;
190
- goto do_cmop;
191
case 0x4: /* CLS */
192
if (u) {
193
tcg_gen_clzi_i32(tcg_res, tcg_op, 32);
194
@@ -XXX,XX +XXX,XX @@ static void disas_simd_two_reg_misc(DisasContext *s, uint32_t insn)
195
genfn(tcg_res, cpu_env, tcg_op);
196
break;
197
}
198
- case 0x8: /* CMGT, CMGE */
199
- case 0x9: /* CMEQ, CMLE */
200
- case 0xa: /* CMLT */
201
- {
202
- static NeonGenTwoOpFn * const fns[3][2] = {
203
- { gen_helper_neon_cgt_s8, gen_helper_neon_cgt_s16 },
204
- { gen_helper_neon_cge_s8, gen_helper_neon_cge_s16 },
205
- { gen_helper_neon_ceq_u8, gen_helper_neon_ceq_u16 },
206
- };
207
- NeonGenTwoOpFn *genfn;
208
- int comp;
209
- bool reverse;
210
- TCGv_i32 tcg_zero = tcg_const_i32(0);
211
-
212
- /* comp = index into [CMGT, CMGE, CMEQ, CMLE, CMLT] */
213
- comp = (opcode - 0x8) * 2 + u;
214
- /* ...but LE, LT are implemented as reverse GE, GT */
215
- reverse = (comp > 2);
216
- if (reverse) {
217
- comp = 4 - comp;
218
- }
219
- genfn = fns[comp][size];
220
- if (reverse) {
221
- genfn(tcg_res, tcg_zero, tcg_op);
222
- } else {
223
- genfn(tcg_res, tcg_op, tcg_zero);
224
- }
225
- tcg_temp_free_i32(tcg_zero);
226
- break;
227
- }
228
case 0x4: /* CLS, CLZ */
229
if (u) {
230
if (size == 0) {
231
diff --git a/target/arm/translate.c b/target/arm/translate.c
232
index XXXXXXX..XXXXXXX 100644
233
--- a/target/arm/translate.c
234
+++ b/target/arm/translate.c
235
@@ -XXX,XX +XXX,XX @@ static int do_v81_helper(DisasContext *s, gen_helper_gvec_3_ptr *fn,
236
return 1;
237
}
238
239
+static void gen_ceq0_i32(TCGv_i32 d, TCGv_i32 a)
240
+{
241
+ tcg_gen_setcondi_i32(TCG_COND_EQ, d, a, 0);
242
+ tcg_gen_neg_i32(d, d);
243
+}
244
+
245
+static void gen_ceq0_i64(TCGv_i64 d, TCGv_i64 a)
246
+{
247
+ tcg_gen_setcondi_i64(TCG_COND_EQ, d, a, 0);
248
+ tcg_gen_neg_i64(d, d);
249
+}
250
+
251
+static void gen_ceq0_vec(unsigned vece, TCGv_vec d, TCGv_vec a)
252
+{
253
+ TCGv_vec zero = tcg_const_zeros_vec_matching(d);
254
+ tcg_gen_cmp_vec(TCG_COND_EQ, vece, d, a, zero);
255
+ tcg_temp_free_vec(zero);
256
+}
257
+
258
+static const TCGOpcode vecop_list_cmp[] = {
259
+ INDEX_op_cmp_vec, 0
260
+};
261
+
262
+const GVecGen2 ceq0_op[4] = {
263
+ { .fno = gen_helper_gvec_ceq0_b,
264
+ .fniv = gen_ceq0_vec,
265
+ .opt_opc = vecop_list_cmp,
266
+ .vece = MO_8 },
267
+ { .fno = gen_helper_gvec_ceq0_h,
268
+ .fniv = gen_ceq0_vec,
269
+ .opt_opc = vecop_list_cmp,
270
+ .vece = MO_16 },
271
+ { .fni4 = gen_ceq0_i32,
272
+ .fniv = gen_ceq0_vec,
273
+ .opt_opc = vecop_list_cmp,
274
+ .vece = MO_32 },
275
+ { .fni8 = gen_ceq0_i64,
276
+ .fniv = gen_ceq0_vec,
277
+ .opt_opc = vecop_list_cmp,
278
+ .prefer_i64 = TCG_TARGET_REG_BITS == 64,
279
+ .vece = MO_64 },
280
+};
281
+
282
+static void gen_cle0_i32(TCGv_i32 d, TCGv_i32 a)
283
+{
284
+ tcg_gen_setcondi_i32(TCG_COND_LE, d, a, 0);
285
+ tcg_gen_neg_i32(d, d);
286
+}
287
+
288
+static void gen_cle0_i64(TCGv_i64 d, TCGv_i64 a)
289
+{
290
+ tcg_gen_setcondi_i64(TCG_COND_LE, d, a, 0);
291
+ tcg_gen_neg_i64(d, d);
292
+}
293
+
294
+static void gen_cle0_vec(unsigned vece, TCGv_vec d, TCGv_vec a)
295
+{
296
+ TCGv_vec zero = tcg_const_zeros_vec_matching(d);
297
+ tcg_gen_cmp_vec(TCG_COND_LE, vece, d, a, zero);
298
+ tcg_temp_free_vec(zero);
299
+}
300
+
301
+const GVecGen2 cle0_op[4] = {
302
+ { .fno = gen_helper_gvec_cle0_b,
303
+ .fniv = gen_cle0_vec,
304
+ .opt_opc = vecop_list_cmp,
305
+ .vece = MO_8 },
306
+ { .fno = gen_helper_gvec_cle0_h,
307
+ .fniv = gen_cle0_vec,
308
+ .opt_opc = vecop_list_cmp,
309
+ .vece = MO_16 },
310
+ { .fni4 = gen_cle0_i32,
311
+ .fniv = gen_cle0_vec,
312
+ .opt_opc = vecop_list_cmp,
313
+ .vece = MO_32 },
314
+ { .fni8 = gen_cle0_i64,
315
+ .fniv = gen_cle0_vec,
316
+ .opt_opc = vecop_list_cmp,
317
+ .prefer_i64 = TCG_TARGET_REG_BITS == 64,
318
+ .vece = MO_64 },
319
+};
320
+
321
+static void gen_cge0_i32(TCGv_i32 d, TCGv_i32 a)
322
+{
323
+ tcg_gen_setcondi_i32(TCG_COND_GE, d, a, 0);
324
+ tcg_gen_neg_i32(d, d);
325
+}
326
+
327
+static void gen_cge0_i64(TCGv_i64 d, TCGv_i64 a)
328
+{
329
+ tcg_gen_setcondi_i64(TCG_COND_GE, d, a, 0);
330
+ tcg_gen_neg_i64(d, d);
331
+}
332
+
333
+static void gen_cge0_vec(unsigned vece, TCGv_vec d, TCGv_vec a)
334
+{
335
+ TCGv_vec zero = tcg_const_zeros_vec_matching(d);
336
+ tcg_gen_cmp_vec(TCG_COND_GE, vece, d, a, zero);
337
+ tcg_temp_free_vec(zero);
338
+}
339
+
340
+const GVecGen2 cge0_op[4] = {
341
+ { .fno = gen_helper_gvec_cge0_b,
342
+ .fniv = gen_cge0_vec,
343
+ .opt_opc = vecop_list_cmp,
344
+ .vece = MO_8 },
345
+ { .fno = gen_helper_gvec_cge0_h,
346
+ .fniv = gen_cge0_vec,
347
+ .opt_opc = vecop_list_cmp,
348
+ .vece = MO_16 },
349
+ { .fni4 = gen_cge0_i32,
350
+ .fniv = gen_cge0_vec,
351
+ .opt_opc = vecop_list_cmp,
352
+ .vece = MO_32 },
353
+ { .fni8 = gen_cge0_i64,
354
+ .fniv = gen_cge0_vec,
355
+ .opt_opc = vecop_list_cmp,
356
+ .prefer_i64 = TCG_TARGET_REG_BITS == 64,
357
+ .vece = MO_64 },
358
+};
359
+
360
+static void gen_clt0_i32(TCGv_i32 d, TCGv_i32 a)
361
+{
362
+ tcg_gen_setcondi_i32(TCG_COND_LT, d, a, 0);
363
+ tcg_gen_neg_i32(d, d);
364
+}
365
+
366
+static void gen_clt0_i64(TCGv_i64 d, TCGv_i64 a)
367
+{
368
+ tcg_gen_setcondi_i64(TCG_COND_LT, d, a, 0);
369
+ tcg_gen_neg_i64(d, d);
370
+}
371
+
372
+static void gen_clt0_vec(unsigned vece, TCGv_vec d, TCGv_vec a)
373
+{
374
+ TCGv_vec zero = tcg_const_zeros_vec_matching(d);
375
+ tcg_gen_cmp_vec(TCG_COND_LT, vece, d, a, zero);
376
+ tcg_temp_free_vec(zero);
377
+}
378
+
379
+const GVecGen2 clt0_op[4] = {
380
+ { .fno = gen_helper_gvec_clt0_b,
381
+ .fniv = gen_clt0_vec,
382
+ .opt_opc = vecop_list_cmp,
383
+ .vece = MO_8 },
384
+ { .fno = gen_helper_gvec_clt0_h,
385
+ .fniv = gen_clt0_vec,
386
+ .opt_opc = vecop_list_cmp,
387
+ .vece = MO_16 },
388
+ { .fni4 = gen_clt0_i32,
389
+ .fniv = gen_clt0_vec,
390
+ .opt_opc = vecop_list_cmp,
391
+ .vece = MO_32 },
392
+ { .fni8 = gen_clt0_i64,
393
+ .fniv = gen_clt0_vec,
394
+ .opt_opc = vecop_list_cmp,
395
+ .prefer_i64 = TCG_TARGET_REG_BITS == 64,
396
+ .vece = MO_64 },
397
+};
398
+
399
+static void gen_cgt0_i32(TCGv_i32 d, TCGv_i32 a)
400
+{
401
+ tcg_gen_setcondi_i32(TCG_COND_GT, d, a, 0);
402
+ tcg_gen_neg_i32(d, d);
403
+}
404
+
405
+static void gen_cgt0_i64(TCGv_i64 d, TCGv_i64 a)
406
+{
407
+ tcg_gen_setcondi_i64(TCG_COND_GT, d, a, 0);
408
+ tcg_gen_neg_i64(d, d);
409
+}
410
+
411
+static void gen_cgt0_vec(unsigned vece, TCGv_vec d, TCGv_vec a)
412
+{
413
+ TCGv_vec zero = tcg_const_zeros_vec_matching(d);
414
+ tcg_gen_cmp_vec(TCG_COND_GT, vece, d, a, zero);
415
+ tcg_temp_free_vec(zero);
416
+}
417
+
418
+const GVecGen2 cgt0_op[4] = {
419
+ { .fno = gen_helper_gvec_cgt0_b,
420
+ .fniv = gen_cgt0_vec,
421
+ .opt_opc = vecop_list_cmp,
422
+ .vece = MO_8 },
423
+ { .fno = gen_helper_gvec_cgt0_h,
424
+ .fniv = gen_cgt0_vec,
425
+ .opt_opc = vecop_list_cmp,
426
+ .vece = MO_16 },
427
+ { .fni4 = gen_cgt0_i32,
428
+ .fniv = gen_cgt0_vec,
429
+ .opt_opc = vecop_list_cmp,
430
+ .vece = MO_32 },
431
+ { .fni8 = gen_cgt0_i64,
432
+ .fniv = gen_cgt0_vec,
433
+ .opt_opc = vecop_list_cmp,
434
+ .prefer_i64 = TCG_TARGET_REG_BITS == 64,
435
+ .vece = MO_64 },
436
+};
437
+
438
static void gen_ssra8_i64(TCGv_i64 d, TCGv_i64 a, int64_t shift)
439
{
440
tcg_gen_vec_sar8i_i64(a, a, shift);
441
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
442
tcg_gen_gvec_abs(size, rd_ofs, rm_ofs, vec_size, vec_size);
443
break;
444
445
+ case NEON_2RM_VCEQ0:
446
+ tcg_gen_gvec_2(rd_ofs, rm_ofs, vec_size,
447
+ vec_size, &ceq0_op[size]);
448
+ break;
449
+ case NEON_2RM_VCGT0:
450
+ tcg_gen_gvec_2(rd_ofs, rm_ofs, vec_size,
451
+ vec_size, &cgt0_op[size]);
452
+ break;
453
+ case NEON_2RM_VCLE0:
454
+ tcg_gen_gvec_2(rd_ofs, rm_ofs, vec_size,
455
+ vec_size, &cle0_op[size]);
456
+ break;
457
+ case NEON_2RM_VCGE0:
458
+ tcg_gen_gvec_2(rd_ofs, rm_ofs, vec_size,
459
+ vec_size, &cge0_op[size]);
460
+ break;
461
+ case NEON_2RM_VCLT0:
462
+ tcg_gen_gvec_2(rd_ofs, rm_ofs, vec_size,
463
+ vec_size, &clt0_op[size]);
464
+ break;
465
+
466
default:
467
elementwise:
468
for (pass = 0; pass < (q ? 4 : 2); pass++) {
469
@@ -XXX,XX +XXX,XX @@ static int disas_neon_data_insn(DisasContext *s, uint32_t insn)
470
default: abort();
471
}
472
break;
473
- case NEON_2RM_VCGT0: case NEON_2RM_VCLE0:
474
- tmp2 = tcg_const_i32(0);
475
- switch(size) {
476
- case 0: gen_helper_neon_cgt_s8(tmp, tmp, tmp2); break;
477
- case 1: gen_helper_neon_cgt_s16(tmp, tmp, tmp2); break;
478
- case 2: gen_helper_neon_cgt_s32(tmp, tmp, tmp2); break;
479
- default: abort();
480
- }
481
- tcg_temp_free_i32(tmp2);
482
- if (op == NEON_2RM_VCLE0) {
483
- tcg_gen_not_i32(tmp, tmp);
484
- }
485
- break;
486
- case NEON_2RM_VCGE0: case NEON_2RM_VCLT0:
487
- tmp2 = tcg_const_i32(0);
488
- switch(size) {
489
- case 0: gen_helper_neon_cge_s8(tmp, tmp, tmp2); break;
490
- case 1: gen_helper_neon_cge_s16(tmp, tmp, tmp2); break;
491
- case 2: gen_helper_neon_cge_s32(tmp, tmp, tmp2); break;
492
- default: abort();
493
- }
494
- tcg_temp_free_i32(tmp2);
495
- if (op == NEON_2RM_VCLT0) {
496
- tcg_gen_not_i32(tmp, tmp);
497
- }
498
- break;
499
- case NEON_2RM_VCEQ0:
500
- tmp2 = tcg_const_i32(0);
501
- switch(size) {
502
- case 0: gen_helper_neon_ceq_u8(tmp, tmp, tmp2); break;
503
- case 1: gen_helper_neon_ceq_u16(tmp, tmp, tmp2); break;
504
- case 2: gen_helper_neon_ceq_u32(tmp, tmp, tmp2); break;
505
- default: abort();
506
- }
507
- tcg_temp_free_i32(tmp2);
508
- break;
509
case NEON_2RM_VCGT0_F:
510
{
511
TCGv_ptr fpstatus = get_fpstatus_ptr(1);
512
diff --git a/target/arm/vec_helper.c b/target/arm/vec_helper.c
513
index XXXXXXX..XXXXXXX 100644
514
--- a/target/arm/vec_helper.c
515
+++ b/target/arm/vec_helper.c
516
@@ -XXX,XX +XXX,XX @@ void HELPER(sve2_pmull_h)(void *vd, void *vn, void *vm, uint32_t desc)
34
}
517
}
35
}
518
}
36
@@ -XXX,XX +XXX,XX @@ static uint64_t aspeed_smc_flash_read(void *opaque, hwaddr addr, unsigned size)
519
#endif
37
}
520
+
38
521
+#define DO_CMP0(NAME, TYPE, OP) \
39
static void aspeed_smc_flash_write(void *opaque, hwaddr addr, uint64_t data,
522
+void HELPER(NAME)(void *vd, void *vn, uint32_t desc) \
40
- unsigned size)
523
+{ \
41
+ unsigned size)
524
+ intptr_t i, opr_sz = simd_oprsz(desc); \
42
{
525
+ for (i = 0; i < opr_sz; i += sizeof(TYPE)) { \
43
AspeedSMCFlash *fl = opaque;
526
+ TYPE nn = *(TYPE *)(vn + i); \
44
AspeedSMCState *s = fl->controller;
527
+ *(TYPE *)(vd + i) = -(nn OP 0); \
528
+ } \
529
+ clear_tail(vd, opr_sz, simd_maxsz(desc)); \
530
+}
531
+
532
+DO_CMP0(gvec_ceq0_b, int8_t, ==)
533
+DO_CMP0(gvec_clt0_b, int8_t, <)
534
+DO_CMP0(gvec_cle0_b, int8_t, <=)
535
+DO_CMP0(gvec_cgt0_b, int8_t, >)
536
+DO_CMP0(gvec_cge0_b, int8_t, >=)
537
+
538
+DO_CMP0(gvec_ceq0_h, int16_t, ==)
539
+DO_CMP0(gvec_clt0_h, int16_t, <)
540
+DO_CMP0(gvec_cle0_h, int16_t, <=)
541
+DO_CMP0(gvec_cgt0_h, int16_t, >)
542
+DO_CMP0(gvec_cge0_h, int16_t, >=)
543
+
544
+#undef DO_CMP0
45
--
545
--
46
2.19.0
546
2.20.1
47
547
48
548
diff view generated by jsdifflib
1
From: Cédric Le Goater <clg@kaod.org>
1
From: Jerome Forissier <jerome@forissier.org>
2
2
3
In file included from /home/thuth/devel/qemu/hw/timer/aspeed_timer.c:16:
3
The /secure-chosen node is currently used only by create_uart(), but
4
/home/thuth/devel/qemu/include/hw/misc/aspeed_scu.h:37:3: error:
4
this will change. Therefore move the creation of this node to
5
redefinition of typedef 'AspeedSCUState' is a C11 feature
5
create_fdt().
6
[-Werror,-Wtypedef-redefinition]
7
} AspeedSCUState;
8
^
9
/home/thuth/devel/qemu/include/hw/timer/aspeed_timer.h:27:31: note:
10
previous definition is here
11
typedef struct AspeedSCUState AspeedSCUState;
12
6
13
Reported-by: Thomas Huth <thuth@redhat.com>
7
Signed-off-by: Jerome Forissier <jerome@forissier.org>
14
Signed-off-by: Cédric Le Goater <clg@kaod.org>
8
Message-id: 20200420121807.8204-2-jerome@forissier.org
15
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
16
Message-id: 20180921161939.822-2-clg@kaod.org
17
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
9
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
18
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
19
---
11
---
20
include/hw/timer/aspeed_timer.h | 3 +--
12
hw/arm/virt.c | 5 ++++-
21
hw/timer/aspeed_timer.c | 1 -
13
1 file changed, 4 insertions(+), 1 deletion(-)
22
2 files changed, 1 insertion(+), 3 deletions(-)
23
14
24
diff --git a/include/hw/timer/aspeed_timer.h b/include/hw/timer/aspeed_timer.h
15
diff --git a/hw/arm/virt.c b/hw/arm/virt.c
25
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
26
--- a/include/hw/timer/aspeed_timer.h
17
--- a/hw/arm/virt.c
27
+++ b/include/hw/timer/aspeed_timer.h
18
+++ b/hw/arm/virt.c
28
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@ static void create_fdt(VirtMachineState *vms)
29
#define ASPEED_TIMER_H
20
/* /chosen must exist for load_dtb to fill in necessary properties later */
30
21
qemu_fdt_add_subnode(fdt, "/chosen");
31
#include "qemu/timer.h"
22
32
-
23
+ if (vms->secure) {
33
-typedef struct AspeedSCUState AspeedSCUState;
24
+ qemu_fdt_add_subnode(fdt, "/secure-chosen");
34
+#include "hw/misc/aspeed_scu.h"
25
+ }
35
26
+
36
#define ASPEED_TIMER(obj) \
27
/* Clock node, for the benefit of the UART. The kernel device tree
37
OBJECT_CHECK(AspeedTimerCtrlState, (obj), TYPE_ASPEED_TIMER);
28
* binding documentation claims the PL011 node clock properties are
38
diff --git a/hw/timer/aspeed_timer.c b/hw/timer/aspeed_timer.c
29
* optional but in practice if you omit them the kernel refuses to
39
index XXXXXXX..XXXXXXX 100644
30
@@ -XXX,XX +XXX,XX @@ static void create_uart(const VirtMachineState *vms, int uart,
40
--- a/hw/timer/aspeed_timer.c
31
qemu_fdt_setprop_string(vms->fdt, nodename, "status", "disabled");
41
+++ b/hw/timer/aspeed_timer.c
32
qemu_fdt_setprop_string(vms->fdt, nodename, "secure-status", "okay");
42
@@ -XXX,XX +XXX,XX @@
33
43
#include "qapi/error.h"
34
- qemu_fdt_add_subnode(vms->fdt, "/secure-chosen");
44
#include "hw/sysbus.h"
35
qemu_fdt_setprop_string(vms->fdt, "/secure-chosen", "stdout-path",
45
#include "hw/timer/aspeed_timer.h"
36
nodename);
46
-#include "hw/misc/aspeed_scu.h"
37
}
47
#include "qemu-common.h"
48
#include "qemu/bitops.h"
49
#include "qemu/timer.h"
50
--
38
--
51
2.19.0
39
2.20.1
52
40
53
41
diff view generated by jsdifflib
1
From: Eric Auger <eric.auger@redhat.com>
1
From: Jerome Forissier <jerome@forissier.org>
2
2
3
The event queue management is broken today. Event records
3
Generate random seeds to be used by the non-secure and/or secure OSes
4
are not properly written as EVT_SET_* macro was not updating
4
for ASLR. The seeds are 64-bit random values exported via the DT
5
the actual event record. Also the event queue interrupt
5
properties /chosen/kaslr-seed [1] and /secure-chosen/kaslr-seed, the
6
is not correctly triggered.
6
latter being used by OP-TEE [2].
7
7
8
Fixes: bb981004eaf4 ("hw/arm/smmuv3: Event queue recording helper")
8
[1] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=e5bc0c37c97e1
9
Signed-off-by: Eric Auger <eric.auger@redhat.com>
9
[2] https://github.com/OP-TEE/optee_os/commit/ef262691fe0e
10
Message-id: 20180921070138.10114-3-eric.auger@redhat.com
10
11
Signed-off-by: Jerome Forissier <jerome@forissier.org>
12
Message-id: 20200420121807.8204-3-jerome@forissier.org
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
13
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
15
---
14
hw/arm/smmuv3-internal.h | 26 +++++++++++++-------------
16
hw/arm/virt.c | 15 +++++++++++++++
15
hw/arm/smmuv3.c | 2 +-
17
1 file changed, 15 insertions(+)
16
2 files changed, 14 insertions(+), 14 deletions(-)
17
18
18
diff --git a/hw/arm/smmuv3-internal.h b/hw/arm/smmuv3-internal.h
19
diff --git a/hw/arm/virt.c b/hw/arm/virt.c
19
index XXXXXXX..XXXXXXX 100644
20
index XXXXXXX..XXXXXXX 100644
20
--- a/hw/arm/smmuv3-internal.h
21
--- a/hw/arm/virt.c
21
+++ b/hw/arm/smmuv3-internal.h
22
+++ b/hw/arm/virt.c
22
@@ -XXX,XX +XXX,XX @@ typedef struct SMMUEventInfo {
23
@@ -XXX,XX +XXX,XX @@
23
24
#include "hw/acpi/generic_event_device.h"
24
#define EVT_Q_OVERFLOW (1 << 31)
25
#include "hw/virtio/virtio-iommu.h"
25
26
#include "hw/char/pl011.h"
26
-#define EVT_SET_TYPE(x, v) deposit32((x)->word[0], 0 , 8 , v)
27
+#include "qemu/guest-random.h"
27
-#define EVT_SET_SSV(x, v) deposit32((x)->word[0], 11, 1 , v)
28
28
-#define EVT_SET_SSID(x, v) deposit32((x)->word[0], 12, 20, v)
29
#define DEFINE_VIRT_MACHINE_LATEST(major, minor, latest) \
29
-#define EVT_SET_SID(x, v) ((x)->word[1] = v)
30
static void virt_##major##_##minor##_class_init(ObjectClass *oc, \
30
-#define EVT_SET_STAG(x, v) deposit32((x)->word[2], 0 , 16, v)
31
@@ -XXX,XX +XXX,XX @@ static bool cpu_type_valid(const char *cpu)
31
-#define EVT_SET_STALL(x, v) deposit32((x)->word[2], 31, 1 , v)
32
return false;
32
-#define EVT_SET_PNU(x, v) deposit32((x)->word[3], 1 , 1 , v)
33
}
33
-#define EVT_SET_IND(x, v) deposit32((x)->word[3], 2 , 1 , v)
34
34
-#define EVT_SET_RNW(x, v) deposit32((x)->word[3], 3 , 1 , v)
35
+static void create_kaslr_seed(VirtMachineState *vms, const char *node)
35
-#define EVT_SET_S2(x, v) deposit32((x)->word[3], 7 , 1 , v)
36
+{
36
-#define EVT_SET_CLASS(x, v) deposit32((x)->word[3], 8 , 2 , v)
37
+ Error *err = NULL;
37
+#define EVT_SET_TYPE(x, v) ((x)->word[0] = deposit32((x)->word[0], 0 , 8 , v))
38
+ uint64_t seed;
38
+#define EVT_SET_SSV(x, v) ((x)->word[0] = deposit32((x)->word[0], 11, 1 , v))
39
+
39
+#define EVT_SET_SSID(x, v) ((x)->word[0] = deposit32((x)->word[0], 12, 20, v))
40
+ if (qemu_guest_getrandom(&seed, sizeof(seed), &err)) {
40
+#define EVT_SET_SID(x, v) ((x)->word[1] = v)
41
+ error_free(err);
41
+#define EVT_SET_STAG(x, v) ((x)->word[2] = deposit32((x)->word[2], 0 , 16, v))
42
+ return;
42
+#define EVT_SET_STALL(x, v) ((x)->word[2] = deposit32((x)->word[2], 31, 1 , v))
43
+ }
43
+#define EVT_SET_PNU(x, v) ((x)->word[3] = deposit32((x)->word[3], 1 , 1 , v))
44
+ qemu_fdt_setprop_u64(vms->fdt, node, "kaslr-seed", seed);
44
+#define EVT_SET_IND(x, v) ((x)->word[3] = deposit32((x)->word[3], 2 , 1 , v))
45
+}
45
+#define EVT_SET_RNW(x, v) ((x)->word[3] = deposit32((x)->word[3], 3 , 1 , v))
46
+
46
+#define EVT_SET_S2(x, v) ((x)->word[3] = deposit32((x)->word[3], 7 , 1 , v))
47
static void create_fdt(VirtMachineState *vms)
47
+#define EVT_SET_CLASS(x, v) ((x)->word[3] = deposit32((x)->word[3], 8 , 2 , v))
48
{
48
#define EVT_SET_ADDR(x, addr) \
49
MachineState *ms = MACHINE(vms);
49
do { \
50
@@ -XXX,XX +XXX,XX @@ static void create_fdt(VirtMachineState *vms)
50
(x)->word[5] = (uint32_t)(addr >> 32); \
51
51
@@ -XXX,XX +XXX,XX @@ typedef struct SMMUEventInfo {
52
/* /chosen must exist for load_dtb to fill in necessary properties later */
52
} while (0)
53
qemu_fdt_add_subnode(fdt, "/chosen");
53
#define EVT_SET_ADDR2(x, addr) \
54
+ create_kaslr_seed(vms, "/chosen");
54
do { \
55
55
- deposit32((x)->word[7], 3, 29, addr >> 16); \
56
if (vms->secure) {
56
- deposit32((x)->word[7], 0, 16, addr & 0xffff);\
57
qemu_fdt_add_subnode(fdt, "/secure-chosen");
57
+ (x)->word[7] = deposit32((x)->word[7], 3, 29, addr >> 16); \
58
+ create_kaslr_seed(vms, "/secure-chosen");
58
+ (x)->word[7] = deposit32((x)->word[7], 0, 16, addr & 0xffff);\
59
} while (0)
60
61
void smmuv3_record_event(SMMUv3State *s, SMMUEventInfo *event);
62
diff --git a/hw/arm/smmuv3.c b/hw/arm/smmuv3.c
63
index XXXXXXX..XXXXXXX 100644
64
--- a/hw/arm/smmuv3.c
65
+++ b/hw/arm/smmuv3.c
66
@@ -XXX,XX +XXX,XX @@ static MemTxResult smmuv3_write_eventq(SMMUv3State *s, Evt *evt)
67
return r;
68
}
59
}
69
60
70
- if (smmuv3_q_empty(q)) {
61
/* Clock node, for the benefit of the UART. The kernel device tree
71
+ if (!smmuv3_q_empty(q)) {
72
smmuv3_trigger_irq(s, SMMU_IRQ_EVTQ, 0);
73
}
74
return MEMTX_OK;
75
--
62
--
76
2.19.0
63
2.20.1
77
64
78
65
diff view generated by jsdifflib
1
From: Richard Henderson <richard.henderson@linaro.org>
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
2
3
Not only are the sve-related tb_flags fields unused when SVE is
3
Under KVM these registers are written by the hardware.
4
disabled, but not all of the cpu registers are initialized properly
4
Restrict the writefn handlers to TCG to avoid when building
5
for computing same. This can corrupt other fields by ORing in -1,
5
without TCG:
6
which might result in QEMU crashing.
7
6
8
This bug was not present in 3.0, but this patch is cc'd to
7
LINK aarch64-softmmu/qemu-system-aarch64
9
stable because adf92eab90e3f5f34c285 where the bug was
8
target/arm/helper.o: In function `do_ats_write':
10
introduced was marked for stable.
9
target/arm/helper.c:3524: undefined reference to `raise_exception'
11
10
12
Fixes: adf92eab90e3f5f34c285
11
Suggested-by: Richard Henderson <richard.henderson@linaro.org>
13
Cc: qemu-stable@nongnu.org (3.0.1)
12
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
14
Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
13
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
15
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
14
Message-id: 20200423073358.27155-2-philmd@redhat.com
16
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
17
---
16
---
18
target/arm/helper.c | 45 ++++++++++++++++++++++++---------------------
17
target/arm/helper.c | 17 +++++++++++++++++
19
1 file changed, 24 insertions(+), 21 deletions(-)
18
1 file changed, 17 insertions(+)
20
19
21
diff --git a/target/arm/helper.c b/target/arm/helper.c
20
diff --git a/target/arm/helper.c b/target/arm/helper.c
22
index XXXXXXX..XXXXXXX 100644
21
index XXXXXXX..XXXXXXX 100644
23
--- a/target/arm/helper.c
22
--- a/target/arm/helper.c
24
+++ b/target/arm/helper.c
23
+++ b/target/arm/helper.c
25
@@ -XXX,XX +XXX,XX @@ void cpu_get_tb_cpu_state(CPUARMState *env, target_ulong *pc,
24
@@ -XXX,XX +XXX,XX @@ static CPAccessResult ats_access(CPUARMState *env, const ARMCPRegInfo *ri,
26
uint32_t flags;
25
return CP_ACCESS_OK;
27
26
}
28
if (is_a64(env)) {
27
29
- int sve_el = sve_exception_el(env);
28
+#ifdef CONFIG_TCG
30
- uint32_t zcr_len;
29
static uint64_t do_ats_write(CPUARMState *env, uint64_t value,
31
-
30
MMUAccessType access_type, ARMMMUIdx mmu_idx)
32
*pc = env->pc;
31
{
33
flags = ARM_TBFLAG_AARCH64_STATE_MASK;
32
@@ -XXX,XX +XXX,XX @@ static uint64_t do_ats_write(CPUARMState *env, uint64_t value,
34
/* Get control bits for tagged addresses */
33
}
35
flags |= (arm_regime_tbi0(env, mmu_idx) << ARM_TBFLAG_TBI0_SHIFT);
34
return par64;
36
flags |= (arm_regime_tbi1(env, mmu_idx) << ARM_TBFLAG_TBI1_SHIFT);
35
}
37
- flags |= sve_el << ARM_TBFLAG_SVEEXC_EL_SHIFT;
36
+#endif /* CONFIG_TCG */
38
37
39
- /* If SVE is disabled, but FP is enabled,
38
static void ats_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
40
- then the effective len is 0. */
39
{
41
- if (sve_el != 0 && fp_el == 0) {
40
+#ifdef CONFIG_TCG
42
- zcr_len = 0;
41
MMUAccessType access_type = ri->opc2 & 1 ? MMU_DATA_STORE : MMU_DATA_LOAD;
43
- } else {
42
uint64_t par64;
44
- int current_el = arm_current_el(env);
43
ARMMMUIdx mmu_idx;
45
- ARMCPU *cpu = arm_env_get_cpu(env);
44
@@ -XXX,XX +XXX,XX @@ static void ats_write(CPUARMState *env, const ARMCPRegInfo *ri, uint64_t value)
46
+ if (arm_feature(env, ARM_FEATURE_SVE)) {
45
par64 = do_ats_write(env, value, access_type, mmu_idx);
47
+ int sve_el = sve_exception_el(env);
46
48
+ uint32_t zcr_len;
47
A32_BANKED_CURRENT_REG_SET(env, par, par64);
49
48
+#else
50
- zcr_len = cpu->sve_max_vq - 1;
49
+ /* Handled by hardware accelerator. */
51
- if (current_el <= 1) {
50
+ g_assert_not_reached();
52
- zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[1]);
51
+#endif /* CONFIG_TCG */
53
- }
52
}
54
- if (current_el < 2 && arm_feature(env, ARM_FEATURE_EL2)) {
53
55
- zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[2]);
54
static void ats1h_write(CPUARMState *env, const ARMCPRegInfo *ri,
56
- }
55
uint64_t value)
57
- if (current_el < 3 && arm_feature(env, ARM_FEATURE_EL3)) {
56
{
58
- zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[3]);
57
+#ifdef CONFIG_TCG
59
+ /* If SVE is disabled, but FP is enabled,
58
MMUAccessType access_type = ri->opc2 & 1 ? MMU_DATA_STORE : MMU_DATA_LOAD;
60
+ * then the effective len is 0.
59
uint64_t par64;
61
+ */
60
62
+ if (sve_el != 0 && fp_el == 0) {
61
par64 = do_ats_write(env, value, access_type, ARMMMUIdx_E2);
63
+ zcr_len = 0;
62
64
+ } else {
63
A32_BANKED_CURRENT_REG_SET(env, par, par64);
65
+ int current_el = arm_current_el(env);
64
+#else
66
+ ARMCPU *cpu = arm_env_get_cpu(env);
65
+ /* Handled by hardware accelerator. */
67
+
66
+ g_assert_not_reached();
68
+ zcr_len = cpu->sve_max_vq - 1;
67
+#endif /* CONFIG_TCG */
69
+ if (current_el <= 1) {
68
}
70
+ zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[1]);
69
71
+ }
70
static CPAccessResult at_s1e2_access(CPUARMState *env, const ARMCPRegInfo *ri,
72
+ if (current_el < 2 && arm_feature(env, ARM_FEATURE_EL2)) {
71
@@ -XXX,XX +XXX,XX @@ static CPAccessResult at_s1e2_access(CPUARMState *env, const ARMCPRegInfo *ri,
73
+ zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[2]);
72
static void ats_write64(CPUARMState *env, const ARMCPRegInfo *ri,
74
+ }
73
uint64_t value)
75
+ if (current_el < 3 && arm_feature(env, ARM_FEATURE_EL3)) {
74
{
76
+ zcr_len = MIN(zcr_len, 0xf & (uint32_t)env->vfp.zcr_el[3]);
75
+#ifdef CONFIG_TCG
77
+ }
76
MMUAccessType access_type = ri->opc2 & 1 ? MMU_DATA_STORE : MMU_DATA_LOAD;
78
}
77
ARMMMUIdx mmu_idx;
79
+ flags |= sve_el << ARM_TBFLAG_SVEEXC_EL_SHIFT;
78
int secure = arm_is_secure_below_el3(env);
80
+ flags |= zcr_len << ARM_TBFLAG_ZCR_LEN_SHIFT;
79
@@ -XXX,XX +XXX,XX @@ static void ats_write64(CPUARMState *env, const ARMCPRegInfo *ri,
81
}
80
}
82
- flags |= zcr_len << ARM_TBFLAG_ZCR_LEN_SHIFT;
81
83
} else {
82
env->cp15.par_el[1] = do_ats_write(env, value, access_type, mmu_idx);
84
*pc = env->regs[15];
83
+#else
85
flags = (env->thumb << ARM_TBFLAG_THUMB_SHIFT)
84
+ /* Handled by hardware accelerator. */
85
+ g_assert_not_reached();
86
+#endif /* CONFIG_TCG */
87
}
88
#endif
89
86
--
90
--
87
2.19.0
91
2.20.1
88
92
89
93
diff view generated by jsdifflib
1
From: Shannon Zhao <shannon.zhaosl@gmail.com>
1
From: Thomas Huth <thuth@redhat.com>
2
2
3
Like commit 16b4226(hw/acpi-build: Add a check for memory-less NUMA node
3
Make cpu_register() (renamed to arm_cpu_register()) available
4
), it also needs to check memory length for NUMA nodes on ARM.
4
from internals.h so we can register CPUs also from other files
5
in the future.
5
6
6
Signed-off-by: Shannon Zhao <shannon.zhaosl@gmail.com>
7
Signed-off-by: Thomas Huth <thuth@redhat.com>
7
Reviewed-by: Andrew Jones <drjones@redhat.com>
8
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
8
Message-id: 20180911112643.19296-1-shenglong.zsl@alibaba-inc.com
9
Reviewed-by: Eric Auger <eric.auger@redhat.com>
10
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
11
Message-id: 20200423073358.27155-3-philmd@redhat.com
12
Message-ID: <20190921150420.30743-2-thuth@redhat.com>
13
[PMD: Only take cpu_register() from Thomas's patch]
14
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
---
16
---
11
hw/arm/virt-acpi-build.c | 10 ++++++----
17
target/arm/cpu-qom.h | 9 ++++++++-
12
1 file changed, 6 insertions(+), 4 deletions(-)
18
target/arm/cpu.c | 10 ++--------
19
target/arm/cpu64.c | 8 +-------
20
3 files changed, 11 insertions(+), 16 deletions(-)
13
21
14
diff --git a/hw/arm/virt-acpi-build.c b/hw/arm/virt-acpi-build.c
22
diff --git a/target/arm/cpu-qom.h b/target/arm/cpu-qom.h
15
index XXXXXXX..XXXXXXX 100644
23
index XXXXXXX..XXXXXXX 100644
16
--- a/hw/arm/virt-acpi-build.c
24
--- a/target/arm/cpu-qom.h
17
+++ b/hw/arm/virt-acpi-build.c
25
+++ b/target/arm/cpu-qom.h
18
@@ -XXX,XX +XXX,XX @@ build_srat(GArray *table_data, BIOSLinker *linker, VirtMachineState *vms)
26
@@ -XXX,XX +XXX,XX @@ struct arm_boot_info;
19
27
20
mem_base = vms->memmap[VIRT_MEM].base;
28
#define TYPE_ARM_MAX_CPU "max-" TYPE_ARM_CPU
21
for (i = 0; i < nb_numa_nodes; ++i) {
29
22
- numamem = acpi_data_push(table_data, sizeof(*numamem));
30
-typedef struct ARMCPUInfo ARMCPUInfo;
23
- build_srat_memory(numamem, mem_base, numa_info[i].node_mem, i,
31
+typedef struct ARMCPUInfo {
24
- MEM_AFFINITY_ENABLED);
32
+ const char *name;
25
- mem_base += numa_info[i].node_mem;
33
+ void (*initfn)(Object *obj);
26
+ if (numa_info[i].node_mem > 0) {
34
+ void (*class_init)(ObjectClass *oc, void *data);
27
+ numamem = acpi_data_push(table_data, sizeof(*numamem));
35
+} ARMCPUInfo;
28
+ build_srat_memory(numamem, mem_base, numa_info[i].node_mem, i,
36
+
29
+ MEM_AFFINITY_ENABLED);
37
+void arm_cpu_register(const ARMCPUInfo *info);
30
+ mem_base += numa_info[i].node_mem;
38
+void aarch64_cpu_register(const ARMCPUInfo *info);
31
+ }
39
40
/**
41
* ARMCPUClass:
42
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
43
index XXXXXXX..XXXXXXX 100644
44
--- a/target/arm/cpu.c
45
+++ b/target/arm/cpu.c
46
@@ -XXX,XX +XXX,XX @@ static void arm_max_initfn(Object *obj)
47
48
#endif /* !defined(CONFIG_USER_ONLY) || !defined(TARGET_AARCH64) */
49
50
-struct ARMCPUInfo {
51
- const char *name;
52
- void (*initfn)(Object *obj);
53
- void (*class_init)(ObjectClass *oc, void *data);
54
-};
55
-
56
static const ARMCPUInfo arm_cpus[] = {
57
#if !defined(CONFIG_USER_ONLY) || !defined(TARGET_AARCH64)
58
{ .name = "arm926", .initfn = arm926_initfn },
59
@@ -XXX,XX +XXX,XX @@ static void cpu_register_class_init(ObjectClass *oc, void *data)
60
acc->info = data;
61
}
62
63
-static void cpu_register(const ARMCPUInfo *info)
64
+void arm_cpu_register(const ARMCPUInfo *info)
65
{
66
TypeInfo type_info = {
67
.parent = TYPE_ARM_CPU,
68
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_register_types(void)
69
type_register_static(&idau_interface_type_info);
70
71
while (info->name) {
72
- cpu_register(info);
73
+ arm_cpu_register(info);
74
info++;
32
}
75
}
33
76
34
build_header(linker, table_data, (void *)(table_data->data + srat_start),
77
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
78
index XXXXXXX..XXXXXXX 100644
79
--- a/target/arm/cpu64.c
80
+++ b/target/arm/cpu64.c
81
@@ -XXX,XX +XXX,XX @@ static void aarch64_max_initfn(Object *obj)
82
cpu_max_set_sve_max_vq, NULL, NULL, &error_fatal);
83
}
84
85
-struct ARMCPUInfo {
86
- const char *name;
87
- void (*initfn)(Object *obj);
88
- void (*class_init)(ObjectClass *oc, void *data);
89
-};
90
-
91
static const ARMCPUInfo aarch64_cpus[] = {
92
{ .name = "cortex-a57", .initfn = aarch64_a57_initfn },
93
{ .name = "cortex-a53", .initfn = aarch64_a53_initfn },
94
@@ -XXX,XX +XXX,XX @@ static void cpu_register_class_init(ObjectClass *oc, void *data)
95
acc->info = data;
96
}
97
98
-static void aarch64_cpu_register(const ARMCPUInfo *info)
99
+void aarch64_cpu_register(const ARMCPUInfo *info)
100
{
101
TypeInfo type_info = {
102
.parent = TYPE_AARCH64_CPU,
35
--
103
--
36
2.19.0
104
2.20.1
37
105
38
106
diff view generated by jsdifflib
1
Convert the pcnet-pci device away from using the old_mmio
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
MemoryRegionOps accessor functions.
3
2
4
This commit is a no-behaviour-change API conversion.
3
Suggested-by: Richard Henderson <richard.henderson@linaro.org>
5
(Since PCNET_PNPMMIO_SIZE is 0x20, the old "addr & 0x10"
6
check and the new "addr < 0x10" check are exact opposites;
7
the new code is phrased to be parallel with the
8
pcnet_io_read/write functions.)
9
10
I have left a TODO comment marker because the similarity
11
between the MMIO and IO accessor behaviour is suspicious
12
and they could be combined, but this will be left to a
13
different patch.
14
15
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
4
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
5
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
6
Message-id: 20200423073358.27155-4-philmd@redhat.com
16
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
17
---
8
---
18
hw/net/pcnet-pci.c | 133 ++++++++++++++++++--------------------------
9
target/arm/cpu.c | 8 +++-----
19
hw/net/trace-events | 8 +--
10
target/arm/cpu64.c | 8 +++-----
20
2 files changed, 57 insertions(+), 84 deletions(-)
11
2 files changed, 6 insertions(+), 10 deletions(-)
21
12
22
diff --git a/hw/net/pcnet-pci.c b/hw/net/pcnet-pci.c
13
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
23
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
24
--- a/hw/net/pcnet-pci.c
15
--- a/target/arm/cpu.c
25
+++ b/hw/net/pcnet-pci.c
16
+++ b/target/arm/cpu.c
26
@@ -XXX,XX +XXX,XX @@ static const MemoryRegionOps pcnet_io_ops = {
17
@@ -XXX,XX +XXX,XX @@ static const ARMCPUInfo arm_cpus[] = {
27
.endianness = DEVICE_LITTLE_ENDIAN,
18
{ .name = "any", .initfn = arm_max_initfn },
19
#endif
20
#endif
21
- { .name = NULL }
28
};
22
};
29
23
30
-static void pcnet_mmio_writeb(void *opaque, hwaddr addr, uint32_t val)
24
static Property arm_cpu_properties[] = {
31
+/*
25
@@ -XXX,XX +XXX,XX @@ static const TypeInfo idau_interface_type_info = {
32
+ * TODO: should MMIO accesses to the addresses corresponding to the
26
33
+ * APROM also honour the BCR_DWIO() setting? If so, then these functions
27
static void arm_cpu_register_types(void)
34
+ * and pcnet_ioport_write/pcnet_ioport_read could be merged.
35
+ * If not, then should pcnet_ioport_{read,write}{w,l} really check
36
+ * BCR_DWIO() for MMIO writes ?
37
+ */
38
+static void pcnet_mmio_write(void *opaque, hwaddr addr, uint64_t value,
39
+ unsigned size)
40
{
28
{
41
PCNetState *d = opaque;
29
- const ARMCPUInfo *info = arm_cpus;
42
30
+ size_t i;
43
- trace_pcnet_mmio_writeb(opaque, addr, val);
31
44
- if (!(addr & 0x10))
32
type_register_static(&arm_cpu_type_info);
45
- pcnet_aprom_writeb(d, addr & 0x0f, val);
33
type_register_static(&idau_interface_type_info);
46
-}
34
47
+ trace_pcnet_mmio_write(opaque, addr, size, val);
35
- while (info->name) {
48
36
- arm_cpu_register(info);
49
-static uint32_t pcnet_mmio_readb(void *opaque, hwaddr addr)
37
- info++;
50
-{
38
+ for (i = 0; i < ARRAY_SIZE(arm_cpus); ++i) {
51
- PCNetState *d = opaque;
39
+ arm_cpu_register(&arm_cpus[i]);
52
- uint32_t val = -1;
40
}
53
-
41
54
- if (!(addr & 0x10))
42
#ifdef CONFIG_KVM
55
- val = pcnet_aprom_readb(d, addr & 0x0f);
43
diff --git a/target/arm/cpu64.c b/target/arm/cpu64.c
56
- trace_pcnet_mmio_readb(opaque, addr, val);
44
index XXXXXXX..XXXXXXX 100644
57
- return val;
45
--- a/target/arm/cpu64.c
58
-}
46
+++ b/target/arm/cpu64.c
59
-
47
@@ -XXX,XX +XXX,XX @@ static const ARMCPUInfo aarch64_cpus[] = {
60
-static void pcnet_mmio_writew(void *opaque, hwaddr addr, uint32_t val)
48
{ .name = "cortex-a53", .initfn = aarch64_a53_initfn },
61
-{
49
{ .name = "cortex-a72", .initfn = aarch64_a72_initfn },
62
- PCNetState *d = opaque;
50
{ .name = "max", .initfn = aarch64_max_initfn },
63
-
51
- { .name = NULL }
64
- trace_pcnet_mmio_writew(opaque, addr, val);
52
};
65
- if (addr & 0x10)
53
66
- pcnet_ioport_writew(d, addr & 0x0f, val);
54
static bool aarch64_cpu_get_aarch64(Object *obj, Error **errp)
67
- else {
55
@@ -XXX,XX +XXX,XX @@ static const TypeInfo aarch64_cpu_type_info = {
68
- addr &= 0x0f;
56
69
- pcnet_aprom_writeb(d, addr, val & 0xff);
57
static void aarch64_cpu_register_types(void)
70
- pcnet_aprom_writeb(d, addr+1, (val & 0xff00) >> 8);
58
{
71
+ if (addr < 0x10) {
59
- const ARMCPUInfo *info = aarch64_cpus;
72
+ if (size == 1) {
60
+ size_t i;
73
+ pcnet_aprom_writeb(d, addr, data);
61
74
+ } else if ((addr & 1) == 0 && size == 2) {
62
type_register_static(&aarch64_cpu_type_info);
75
+ pcnet_aprom_writeb(d, addr, data & 0xff);
63
76
+ pcnet_aprom_writeb(d, addr + 1, data >> 8);
64
- while (info->name) {
77
+ } else if ((addr & 3) == 0 && size == 4) {
65
- aarch64_cpu_register(info);
78
+ pcnet_aprom_writeb(d, addr, data & 0xff);
66
- info++;
79
+ pcnet_aprom_writeb(d, addr + 1, (data >> 8) & 0xff);
67
+ for (i = 0; i < ARRAY_SIZE(aarch64_cpus); ++i) {
80
+ pcnet_aprom_writeb(d, addr + 2, (data >> 16) & 0xff);
68
+ aarch64_cpu_register(&aarch64_cpus[i]);
81
+ pcnet_aprom_writeb(d, addr + 3, data >> 24);
82
+ }
83
+ } else {
84
+ if (size == 2) {
85
+ pcnet_ioport_writew(d, addr, data);
86
+ } else if (size == 4) {
87
+ pcnet_ioport_writel(d, addr, data);
88
+ }
89
}
69
}
90
}
70
}
91
71
92
-static uint32_t pcnet_mmio_readw(void *opaque, hwaddr addr)
93
-{
94
- PCNetState *d = opaque;
95
- uint32_t val = -1;
96
-
97
- if (addr & 0x10)
98
- val = pcnet_ioport_readw(d, addr & 0x0f);
99
- else {
100
- addr &= 0x0f;
101
- val = pcnet_aprom_readb(d, addr+1);
102
- val <<= 8;
103
- val |= pcnet_aprom_readb(d, addr);
104
- }
105
- trace_pcnet_mmio_readw(opaque, addr, val);
106
- return val;
107
-}
108
-
109
-static void pcnet_mmio_writel(void *opaque, hwaddr addr, uint32_t val)
110
+static uint64_t pcnet_mmio_read(void *opque, hwaddr addr, unsigned size)
111
{
112
PCNetState *d = opaque;
113
114
- trace_pcnet_mmio_writel(opaque, addr, val);
115
- if (addr & 0x10)
116
- pcnet_ioport_writel(d, addr & 0x0f, val);
117
- else {
118
- addr &= 0x0f;
119
- pcnet_aprom_writeb(d, addr, val & 0xff);
120
- pcnet_aprom_writeb(d, addr+1, (val & 0xff00) >> 8);
121
- pcnet_aprom_writeb(d, addr+2, (val & 0xff0000) >> 16);
122
- pcnet_aprom_writeb(d, addr+3, (val & 0xff000000) >> 24);
123
- }
124
-}
125
+ trace_pcnet_ioport_read(opaque, addr, size);
126
127
-static uint32_t pcnet_mmio_readl(void *opaque, hwaddr addr)
128
-{
129
- PCNetState *d = opaque;
130
- uint32_t val;
131
-
132
- if (addr & 0x10)
133
- val = pcnet_ioport_readl(d, addr & 0x0f);
134
- else {
135
- addr &= 0x0f;
136
- val = pcnet_aprom_readb(d, addr+3);
137
- val <<= 8;
138
- val |= pcnet_aprom_readb(d, addr+2);
139
- val <<= 8;
140
- val |= pcnet_aprom_readb(d, addr+1);
141
- val <<= 8;
142
- val |= pcnet_aprom_readb(d, addr);
143
+ if (addr < 0x10) {
144
+ if (size == 1) {
145
+ return pcnet_aprom_readb(d, addr);
146
+ } else if ((addr & 1) == 0 && size == 2) {
147
+ return pcnet_aprom_readb(d, addr) |
148
+ (pcnet_aprom_readb(d, addr + 1) << 8);
149
+ } else if ((addr & 3) == 0 && size == 4) {
150
+ return pcnet_aprom_readb(d, addr) |
151
+ (pcnet_aprom_readb(d, addr + 1) << 8) |
152
+ (pcnet_aprom_readb(d, addr + 2) << 16) |
153
+ (pcnet_aprom_readb(d, addr + 3) << 24);
154
+ }
155
+ } else {
156
+ if (size == 2) {
157
+ return pcnet_ioport_readw(d, addr);
158
+ } else if (size == 4) {
159
+ return pcnet_ioport_readl(d, addr);
160
+ }
161
}
162
- trace_pcnet_mmio_readl(opaque, addr, val);
163
- return val;
164
+ return ((uint64_t)1 << (size * 8)) - 1;
165
}
166
167
static const VMStateDescription vmstate_pci_pcnet = {
168
@@ -XXX,XX +XXX,XX @@ static const VMStateDescription vmstate_pci_pcnet = {
169
/* PCI interface */
170
171
static const MemoryRegionOps pcnet_mmio_ops = {
172
- .old_mmio = {
173
- .read = { pcnet_mmio_readb, pcnet_mmio_readw, pcnet_mmio_readl },
174
- .write = { pcnet_mmio_writeb, pcnet_mmio_writew, pcnet_mmio_writel },
175
- },
176
+ .read = pcnet_mmio_read,
177
+ .write = pcnet_mmio_write,
178
+ .valid.min_access_size = 1,
179
+ .valid.max_access_size = 4,
180
+ .impl.min_access_size = 1,
181
+ .impl.max_access_size = 4,
182
.endianness = DEVICE_LITTLE_ENDIAN,
183
};
184
185
diff --git a/hw/net/trace-events b/hw/net/trace-events
186
index XXXXXXX..XXXXXXX 100644
187
--- a/hw/net/trace-events
188
+++ b/hw/net/trace-events
189
@@ -XXX,XX +XXX,XX @@ pcnet_aprom_writeb(void *opaque, uint32_t addr, uint32_t val) "opaque=%p addr=0x
190
pcnet_aprom_readb(void *opaque, uint32_t addr, uint32_t val) "opaque=%p addr=0x%08x val=0x%02x"
191
pcnet_ioport_read(void *opaque, uint64_t addr, unsigned size) "opaque=%p addr=0x%"PRIx64" size=%d"
192
pcnet_ioport_write(void *opaque, uint64_t addr, uint64_t data, unsigned size) "opaque=%p addr=0x%"PRIx64" data=0x%"PRIx64" size=%d"
193
-pcnet_mmio_writeb(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
194
-pcnet_mmio_writew(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
195
-pcnet_mmio_writel(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
196
-pcnet_mmio_readb(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
197
-pcnet_mmio_readw(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
198
-pcnet_mmio_readl(void *opaque, uint64_t addr, uint32_t val) "opaque=%p addr=0x%"PRIx64" val=0x%x"
199
+pcnet_mmio_write(void *opaque, uint64_t addr, uint32_t val, unsigned size) "opaque=%p addr=0x%"PRIx64" val=0x%x size=%d"
200
+pcnet_mmio_read(void *opaque, uint64_t addr, unsigned size) "opaque=%p addr=0x%"PRIx64" size=%d"
201
202
# hw/net/net_rx_pkt.c
203
net_rx_pkt_parsed(bool ip4, bool ip6, bool udp, bool tcp, size_t l3o, size_t l4o, size_t l5o) "RX packet parsed: ip4: %d, ip6: %d, udp: %d, tcp: %d, l3 offset: %zu, l4 offset: %zu, l5 offset: %zu"
204
--
72
--
205
2.19.0
73
2.20.1
206
74
207
75
diff view generated by jsdifflib
1
The ARMv8 architecture defines that an AArch32 CPU starts
1
From: Philippe Mathieu-Daudé <philmd@redhat.com>
2
in SVC mode, unless EL2 is the highest available EL, in
3
which case it starts in Hyp mode. (In ARMv7 a CPU with EL2
4
but not EL3 was not a valid configuration, but we don't
5
specifically reject this if the user asks for one.)
6
2
3
We will move this code in the next commit. Clean it up
4
first to avoid checkpatch.pl errors.
5
6
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
7
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
8
Message-id: 20200423073358.27155-5-philmd@redhat.com
7
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
9
Message-id: 20180823135047.16525-1-peter.maydell@linaro.org
10
---
10
---
11
target/arm/cpu.c | 14 ++++++++++++--
11
target/arm/cpu.c | 9 ++++++---
12
1 file changed, 12 insertions(+), 2 deletions(-)
12
1 file changed, 6 insertions(+), 3 deletions(-)
13
13
14
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
14
diff --git a/target/arm/cpu.c b/target/arm/cpu.c
15
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
16
--- a/target/arm/cpu.c
16
--- a/target/arm/cpu.c
17
+++ b/target/arm/cpu.c
17
+++ b/target/arm/cpu.c
18
@@ -XXX,XX +XXX,XX @@ static void arm_cpu_reset(CPUState *s)
18
@@ -XXX,XX +XXX,XX @@ static bool arm_v7m_cpu_exec_interrupt(CPUState *cs, int interrupt_request)
19
env->cp15.c15_cpar = 1;
19
CPUARMState *env = &cpu->env;
20
}
20
bool ret = false;
21
#else
21
22
- /* SVC mode with interrupts disabled. */
22
- /* ARMv7-M interrupt masking works differently than -A or -R.
23
- env->uncached_cpsr = ARM_CPU_MODE_SVC;
24
+
25
+ /*
23
+ /*
26
+ * If the highest available EL is EL2, AArch32 will start in Hyp
24
+ * ARMv7-M interrupt masking works differently than -A or -R.
27
+ * mode; otherwise it starts in SVC. Note that if we start in
25
* There is no FIQ/IRQ distinction. Instead of I and F bits
28
+ * AArch64 then these values in the uncached_cpsr will be ignored.
26
* masking FIQ and IRQ interrupts, an exception is taken only
29
+ */
27
* if it is higher priority than the current execution priority
30
+ if (arm_feature(env, ARM_FEATURE_EL2) &&
28
@@ -XXX,XX +XXX,XX @@ static void arm1026_initfn(Object *obj)
31
+ !arm_feature(env, ARM_FEATURE_EL3)) {
29
static void arm1136_r2_initfn(Object *obj)
32
+ env->uncached_cpsr = ARM_CPU_MODE_HYP;
30
{
33
+ } else {
31
ARMCPU *cpu = ARM_CPU(obj);
34
+ env->uncached_cpsr = ARM_CPU_MODE_SVC;
32
- /* What qemu calls "arm1136_r2" is actually the 1136 r0p2, ie an
35
+ }
33
+ /*
36
env->daif = PSTATE_D | PSTATE_A | PSTATE_I | PSTATE_F;
34
+ * What qemu calls "arm1136_r2" is actually the 1136 r0p2, ie an
37
35
* older core than plain "arm1136". In particular this does not
38
if (arm_feature(env, ARM_FEATURE_M)) {
36
* have the v6K features.
37
* These ID register values are correct for 1136 but may be wrong
38
@@ -XXX,XX +XXX,XX @@ static const ARMCPUInfo arm_cpus[] = {
39
{ .name = "arm926", .initfn = arm926_initfn },
40
{ .name = "arm946", .initfn = arm946_initfn },
41
{ .name = "arm1026", .initfn = arm1026_initfn },
42
- /* What QEMU calls "arm1136-r2" is actually the 1136 r0p2, i.e. an
43
+ /*
44
+ * What QEMU calls "arm1136-r2" is actually the 1136 r0p2, i.e. an
45
* older core than plain "arm1136". In particular this does not
46
* have the v6K features.
47
*/
39
--
48
--
40
2.19.0
49
2.20.1
41
50
42
51
diff view generated by jsdifflib
1
From: Joel Stanley <joel@jms.id.au>
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
2
3
This contains the NRF51, and the machine that uses it, the BBC
3
Allow name wildcards in qemu_fdt_node_path(). This is useful
4
micro:bit.
4
to find all nodes with a given compatibility string.
5
5
6
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
6
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
8
Signed-off-by: Joel Stanley <joel@jms.id.au>
8
Message-id: 20200423121114.4274-2-edgar.iglesias@gmail.com
9
Message-id: 20180831220920.27113-2-joel@jms.id.au
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
9
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
11
---
10
---
12
MAINTAINERS | 8 ++++++++
11
include/sysemu/device_tree.h | 3 +++
13
1 file changed, 8 insertions(+)
12
device_tree.c | 2 +-
13
2 files changed, 4 insertions(+), 1 deletion(-)
14
14
15
diff --git a/MAINTAINERS b/MAINTAINERS
15
diff --git a/include/sysemu/device_tree.h b/include/sysemu/device_tree.h
16
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
17
--- a/MAINTAINERS
17
--- a/include/sysemu/device_tree.h
18
+++ b/MAINTAINERS
18
+++ b/include/sysemu/device_tree.h
19
@@ -XXX,XX +XXX,XX @@ F: include/hw/*/*aspeed*
19
@@ -XXX,XX +XXX,XX @@ void *load_device_tree_from_sysfs(void);
20
F: hw/net/ftgmac100.c
20
* NULL. If there is no error but no matching node was found, the
21
F: include/hw/net/ftgmac100.h
21
* returned array contains a single element equal to NULL. If an error
22
22
* was encountered when parsing the blob, the function returns NULL
23
+NRF51
23
+ *
24
+M: Joel Stanley <joel@jms.id.au>
24
+ * @name may be NULL to wildcard names and only match compatibility
25
+L: qemu-arm@nongnu.org
25
+ * strings.
26
+S: Maintained
26
*/
27
+F: hw/arm/nrf51_soc.c
27
char **qemu_fdt_node_path(void *fdt, const char *name, char *compat,
28
+F: hw/arm/microbit.c
28
Error **errp);
29
+F: include/hw/arm/nrf51_soc.h
29
diff --git a/device_tree.c b/device_tree.c
30
+
30
index XXXXXXX..XXXXXXX 100644
31
CRIS Machines
31
--- a/device_tree.c
32
-------------
32
+++ b/device_tree.c
33
Axis Dev88
33
@@ -XXX,XX +XXX,XX @@ char **qemu_fdt_node_path(void *fdt, const char *name, char *compat,
34
offset = len;
35
break;
36
}
37
- if (!strcmp(iter_name, name)) {
38
+ if (!name || !strcmp(iter_name, name)) {
39
char *path;
40
41
path = g_malloc(path_len);
34
--
42
--
35
2.19.0
43
2.20.1
36
44
37
45
diff view generated by jsdifflib
1
From: Eric Auger <eric.auger@redhat.com>
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
2
3
At the point smmu_find_add_as() gets called, the bus number might
3
Make compat in qemu_fdt_node_path() const char *.
4
not be computed. Let's change the name of IOMMU memory region and
5
just use the devfn and an incrementing index.
6
4
7
The name only is used for debug.
5
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
8
6
Message-id: 20200423121114.4274-3-edgar.iglesias@gmail.com
9
Signed-off-by: Eric Auger <eric.auger@redhat.com>
10
Message-id: 20180921070138.10114-2-eric.auger@redhat.com
11
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
12
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
13
---
9
---
14
hw/arm/smmu-common.c | 6 +++---
10
include/sysemu/device_tree.h | 2 +-
15
1 file changed, 3 insertions(+), 3 deletions(-)
11
device_tree.c | 2 +-
12
2 files changed, 2 insertions(+), 2 deletions(-)
16
13
17
diff --git a/hw/arm/smmu-common.c b/hw/arm/smmu-common.c
14
diff --git a/include/sysemu/device_tree.h b/include/sysemu/device_tree.h
18
index XXXXXXX..XXXXXXX 100644
15
index XXXXXXX..XXXXXXX 100644
19
--- a/hw/arm/smmu-common.c
16
--- a/include/sysemu/device_tree.h
20
+++ b/hw/arm/smmu-common.c
17
+++ b/include/sysemu/device_tree.h
21
@@ -XXX,XX +XXX,XX @@ static AddressSpace *smmu_find_add_as(PCIBus *bus, void *opaque, int devfn)
18
@@ -XXX,XX +XXX,XX @@ void *load_device_tree_from_sysfs(void);
22
SMMUState *s = opaque;
19
* @name may be NULL to wildcard names and only match compatibility
23
SMMUPciBus *sbus = g_hash_table_lookup(s->smmu_pcibus_by_busptr, bus);
20
* strings.
24
SMMUDevice *sdev;
21
*/
25
+ static uint index;
22
-char **qemu_fdt_node_path(void *fdt, const char *name, char *compat,
26
23
+char **qemu_fdt_node_path(void *fdt, const char *name, const char *compat,
27
if (!sbus) {
24
Error **errp);
28
sbus = g_malloc0(sizeof(SMMUPciBus) +
25
29
@@ -XXX,XX +XXX,XX @@ static AddressSpace *smmu_find_add_as(PCIBus *bus, void *opaque, int devfn)
26
/**
30
27
diff --git a/device_tree.c b/device_tree.c
31
sdev = sbus->pbdev[devfn];
28
index XXXXXXX..XXXXXXX 100644
32
if (!sdev) {
29
--- a/device_tree.c
33
- char *name = g_strdup_printf("%s-%d-%d",
30
+++ b/device_tree.c
34
- s->mrtypename,
31
@@ -XXX,XX +XXX,XX @@ char **qemu_fdt_node_unit_path(void *fdt, const char *name, Error **errp)
35
- pci_bus_num(bus), devfn);
32
return path_array;
36
+ char *name = g_strdup_printf("%s-%d-%d", s->mrtypename, devfn, index++);
33
}
37
+
34
38
sdev = sbus->pbdev[devfn] = g_new0(SMMUDevice, 1);
35
-char **qemu_fdt_node_path(void *fdt, const char *name, char *compat,
39
36
+char **qemu_fdt_node_path(void *fdt, const char *name, const char *compat,
40
sdev->smmu = s;
37
Error **errp)
38
{
39
int offset, len, ret;
41
--
40
--
42
2.19.0
41
2.20.1
43
42
44
43
diff view generated by jsdifflib
1
From: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
2
3
commit 97274d0c05d4 ("hw/char/exynos4210_uart.c: Remove unneeded
3
Move arm_boot_info into XlnxZCU102.
4
handling of NULL chardev") broke Exynos4210 support as it removed
5
NULL 'Chardev *chr' handling from exynos4210_uart_create() and
6
currently exynos4210_init() always passes NULL as 'Chardev *chr'
7
argument to exynos4210_uart_create() calls. Fix it by adding
8
missing serial_hd() calls to exynos4210_init().
9
4
10
Signed-off-by: Bartlomiej Zolnierkiewicz <b.zolnierkie@samsung.com>
11
Reviewed-by: Thomas Huth <thuth@redhat.com>
12
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
5
Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
13
Message-id: 9310418.Wg32kryeWE@amdc3058
6
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
7
Message-id: 20200423121114.4274-4-edgar.iglesias@gmail.com
14
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
8
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
15
---
9
---
16
hw/arm/exynos4210.c | 8 ++++----
10
hw/arm/xlnx-zcu102.c | 9 +++++----
17
1 file changed, 4 insertions(+), 4 deletions(-)
11
1 file changed, 5 insertions(+), 4 deletions(-)
18
12
19
diff --git a/hw/arm/exynos4210.c b/hw/arm/exynos4210.c
13
diff --git a/hw/arm/xlnx-zcu102.c b/hw/arm/xlnx-zcu102.c
20
index XXXXXXX..XXXXXXX 100644
14
index XXXXXXX..XXXXXXX 100644
21
--- a/hw/arm/exynos4210.c
15
--- a/hw/arm/xlnx-zcu102.c
22
+++ b/hw/arm/exynos4210.c
16
+++ b/hw/arm/xlnx-zcu102.c
23
@@ -XXX,XX +XXX,XX @@ Exynos4210State *exynos4210_init(MemoryRegion *system_mem)
17
@@ -XXX,XX +XXX,XX @@ typedef struct XlnxZCU102 {
24
18
25
/*** UARTs ***/
19
bool secure;
26
exynos4210_uart_create(EXYNOS4210_UART0_BASE_ADDR,
20
bool virt;
27
- EXYNOS4210_UART0_FIFO_SIZE, 0, NULL,
21
+
28
+ EXYNOS4210_UART0_FIFO_SIZE, 0, serial_hd(0),
22
+ struct arm_boot_info binfo;
29
s->irq_table[exynos4210_get_irq(EXYNOS4210_UART_INT_GRP, 0)]);
23
} XlnxZCU102;
30
24
31
exynos4210_uart_create(EXYNOS4210_UART1_BASE_ADDR,
25
#define TYPE_ZCU102_MACHINE MACHINE_TYPE_NAME("xlnx-zcu102")
32
- EXYNOS4210_UART1_FIFO_SIZE, 1, NULL,
26
#define ZCU102_MACHINE(obj) \
33
+ EXYNOS4210_UART1_FIFO_SIZE, 1, serial_hd(1),
27
OBJECT_CHECK(XlnxZCU102, (obj), TYPE_ZCU102_MACHINE)
34
s->irq_table[exynos4210_get_irq(EXYNOS4210_UART_INT_GRP, 1)]);
28
35
29
-static struct arm_boot_info xlnx_zcu102_binfo;
36
exynos4210_uart_create(EXYNOS4210_UART2_BASE_ADDR,
30
37
- EXYNOS4210_UART2_FIFO_SIZE, 2, NULL,
31
static bool zcu102_get_secure(Object *obj, Error **errp)
38
+ EXYNOS4210_UART2_FIFO_SIZE, 2, serial_hd(2),
32
{
39
s->irq_table[exynos4210_get_irq(EXYNOS4210_UART_INT_GRP, 2)]);
33
@@ -XXX,XX +XXX,XX @@ static void xlnx_zcu102_init(MachineState *machine)
40
34
41
exynos4210_uart_create(EXYNOS4210_UART3_BASE_ADDR,
35
/* TODO create and connect IDE devices for ide_drive_get() */
42
- EXYNOS4210_UART3_FIFO_SIZE, 3, NULL,
36
43
+ EXYNOS4210_UART3_FIFO_SIZE, 3, serial_hd(3),
37
- xlnx_zcu102_binfo.ram_size = ram_size;
44
s->irq_table[exynos4210_get_irq(EXYNOS4210_UART_INT_GRP, 3)]);
38
- xlnx_zcu102_binfo.loader_start = 0;
45
39
- arm_load_kernel(s->soc.boot_cpu_ptr, machine, &xlnx_zcu102_binfo);
46
/*** SD/MMC host controllers ***/
40
+ s->binfo.ram_size = ram_size;
41
+ s->binfo.loader_start = 0;
42
+ arm_load_kernel(s->soc.boot_cpu_ptr, machine, &s->binfo);
43
}
44
45
static void xlnx_zcu102_machine_instance_init(Object *obj)
47
--
46
--
48
2.19.0
47
2.20.1
49
48
50
49
diff view generated by jsdifflib
1
The GICv2's QEMU interface (sysbus MMIO regions, IRQs,
1
From: "Edgar E. Iglesias" <edgar.iglesias@xilinx.com>
2
etc) is now quite complicated with the addition of the
3
virtualization extensions. Add a comment in the header
4
file which documents it.
5
2
3
Disable unsupported FDT firmware nodes if a user passes us
4
a DTB with nodes enabled that the machine cannot support
5
due to lack of EL3 or EL2 support.
6
7
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
8
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
9
Message-id: 20200423121114.4274-5-edgar.iglesias@gmail.com
6
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
10
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
7
Reviewed-by: Luc Michel <luc.michel@greensocs.com>
8
Message-id: 20180823103818.31189-1-peter.maydell@linaro.org
9
---
11
---
10
include/hw/intc/arm_gic.h | 43 +++++++++++++++++++++++++++++++++++++++
12
hw/arm/xlnx-zcu102.c | 30 ++++++++++++++++++++++++++++++
11
1 file changed, 43 insertions(+)
13
1 file changed, 30 insertions(+)
12
14
13
diff --git a/include/hw/intc/arm_gic.h b/include/hw/intc/arm_gic.h
15
diff --git a/hw/arm/xlnx-zcu102.c b/hw/arm/xlnx-zcu102.c
14
index XXXXXXX..XXXXXXX 100644
16
index XXXXXXX..XXXXXXX 100644
15
--- a/include/hw/intc/arm_gic.h
17
--- a/hw/arm/xlnx-zcu102.c
16
+++ b/include/hw/intc/arm_gic.h
18
+++ b/hw/arm/xlnx-zcu102.c
17
@@ -XXX,XX +XXX,XX @@
19
@@ -XXX,XX +XXX,XX @@
18
* with this program; if not, see <http://www.gnu.org/licenses/>.
20
#include "qemu/error-report.h"
19
*/
21
#include "qemu/log.h"
20
22
#include "sysemu/qtest.h"
21
+/*
23
+#include "sysemu/device_tree.h"
22
+ * QEMU interface:
24
23
+ * + QOM property "num-cpu": number of CPUs to support
25
typedef struct XlnxZCU102 {
24
+ * + QOM property "num-irq": number of IRQs (including both SPIs and PPIs)
26
MachineState parent_obj;
25
+ * + QOM property "revision": GIC version (1 or 2), or 0 for the 11MPCore GIC
27
@@ -XXX,XX +XXX,XX @@ static void zcu102_set_virt(Object *obj, bool value, Error **errp)
26
+ * + QOM property "has-security-extensions": set true if the GIC should
28
s->virt = value;
27
+ * implement the security extensions
29
}
28
+ * + QOM property "has-virtualization-extensions": set true if the GIC should
30
29
+ * implement the virtualization extensions
31
+static void zcu102_modify_dtb(const struct arm_boot_info *binfo, void *fdt)
30
+ * + unnamed GPIO inputs: (where P is number of SPIs, i.e. num-irq - 32)
32
+{
31
+ * [0..P-1] SPIs
33
+ XlnxZCU102 *s = container_of(binfo, XlnxZCU102, binfo);
32
+ * [P..P+31] PPIs for CPU 0
34
+ bool method_is_hvc;
33
+ * [P+32..P+63] PPIs for CPU 1
35
+ char **node_path;
34
+ * ...
36
+ const char *r;
35
+ * + sysbus IRQs: (in order; number will vary depending on number of cores)
37
+ int prop_len;
36
+ * - IRQ for CPU 0
38
+ int i;
37
+ * - IRQ for CPU 1
38
+ * ...
39
+ * - FIQ for CPU 0
40
+ * - FIQ for CPU 1
41
+ * ...
42
+ * - VIRQ for CPU 0 (exists even if virt extensions not present)
43
+ * - VIRQ for CPU 1 (exists even if virt extensions not present)
44
+ * ...
45
+ * - VFIQ for CPU 0 (exists even if virt extensions not present)
46
+ * - VFIQ for CPU 1 (exists even if virt extensions not present)
47
+ * ...
48
+ * - maintenance IRQ for CPU i/f 0 (only if virt extensions present)
49
+ * - maintenance IRQ for CPU i/f 1 (only if virt extensions present)
50
+ * + sysbus MMIO regions: (in order; numbers will vary depending on
51
+ * whether virtualization extensions are present and on number of cores)
52
+ * - distributor registers (GICD*)
53
+ * - CPU interface for the accessing core (GICC*)
54
+ * - virtual interface control registers (GICH*) (only if virt extns present)
55
+ * - virtual CPU interface for the accessing core (GICV*) (only if virt)
56
+ * - CPU 0 CPU interface registers
57
+ * - CPU 1 CPU interface registers
58
+ * ...
59
+ * - CPU 0 virtual interface control registers (only if virt extns present)
60
+ * - CPU 1 virtual interface control registers (only if virt extns present)
61
+ * ...
62
+ */
63
+
39
+
64
#ifndef HW_ARM_GIC_H
40
+ /* If EL3 is enabled, we keep all firmware nodes active. */
65
#define HW_ARM_GIC_H
41
+ if (!s->secure) {
42
+ node_path = qemu_fdt_node_path(fdt, NULL, "xlnx,zynqmp-firmware",
43
+ &error_fatal);
44
+
45
+ for (i = 0; node_path && node_path[i]; i++) {
46
+ r = qemu_fdt_getprop(fdt, node_path[i], "method", &prop_len, NULL);
47
+ method_is_hvc = r && !strcmp("hvc", r);
48
+
49
+ /* Allow HVC based firmware if EL2 is enabled. */
50
+ if (method_is_hvc && s->virt) {
51
+ continue;
52
+ }
53
+ qemu_fdt_setprop_string(fdt, node_path[i], "status", "disabled");
54
+ }
55
+ g_strfreev(node_path);
56
+ }
57
+}
58
+
59
static void xlnx_zcu102_init(MachineState *machine)
60
{
61
XlnxZCU102 *s = ZCU102_MACHINE(machine);
62
@@ -XXX,XX +XXX,XX @@ static void xlnx_zcu102_init(MachineState *machine)
63
64
s->binfo.ram_size = ram_size;
65
s->binfo.loader_start = 0;
66
+ s->binfo.modify_dtb = zcu102_modify_dtb;
67
arm_load_kernel(s->soc.boot_cpu_ptr, machine, &s->binfo);
68
}
66
69
67
--
70
--
68
2.19.0
71
2.20.1
69
72
70
73
diff view generated by jsdifflib