mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-08 02:03:56 -06:00
target-arm queue:
* hw/arm/virt: Disable pl011 clock migration if needed * target/arm: Make M-profile VTOR loads on reset handle memory aliasing * target/arm: Set ARMMMUFaultInfo.level in user-only arm_cpu_tlb_fill -----BEGIN PGP SIGNATURE----- iQJNBAABCAA3FiEE4aXFk81BneKOgxXPPCUl7RQ2DN4FAmBZ+oEZHHBldGVyLm1h eWRlbGxAbGluYXJvLm9yZwAKCRA8JSXtFDYM3jnoD/0aToosqYaDppJ7kxRIuOof 9hXcQXIpMFBk0x4sAPhSnXSxnw4uevBJWl5zcn4GvqazdToq7ocn+38DVR0oxSg/ VDZVTKOgE0yBMmsOCCrW5spZM4M/SQxa7ebfZU0uT9yu4W9XVT8Cs6EZ8wNuFUOo hs8zTf483wD9pW+dq195HGjvPpvk999DCeBFiYZHCncuwWyuQveduiKv6/V1NPQF KOlLoXto/lbcDdrOV0yPfhiLXMrXA0TIN3SDwE2QrPqTEQJWR0TaiUQ17HLCHgDG xNfh+nM5sP56B8a/aRwx9z+BKv0/+KjlYO4lZ165xRA+NGKpV8E7A7VjUZEbHM1R x+c/rCsk/dbDAVjJi9hb+RFsqpsdmz/6KhnaHBmO/beZXqeFjaD2JN+XpnrOsquk UUJJb0HS5f+hyBLU1jQsdeRX0Wl8N6xipd4pw0+K60I3Aefv+8/bqr6LYxJafqd4 bhFvuDCecAE00FreexiWi32tHRV10aAWhxcpggh1n3dyflJkxjouRdeWZ7UAlMVK 1yifIZqUsQ3Q+ZDUq9QZZJRbkldy7z0Kuw3xGMKZJnC40h+szs2Xl7ARZPu87fdZ 4zdDOq3blfnLyLVoHO01bi+trA9XvNF/cn0CWw0eIRPo8NUr7rG6aROBHBMECq4v ZBljEj9Wrr1PgN/yxpxREQ== =eihR -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/pmaydell/tags/pull-target-arm-20210323' into staging target-arm queue: * hw/arm/virt: Disable pl011 clock migration if needed * target/arm: Make M-profile VTOR loads on reset handle memory aliasing * target/arm: Set ARMMMUFaultInfo.level in user-only arm_cpu_tlb_fill # gpg: Signature made Tue 23 Mar 2021 14:26:09 GMT # gpg: using RSA key E1A5C593CD419DE28E8315CF3C2525ED14360CDE # gpg: issuer "peter.maydell@linaro.org" # gpg: Good signature from "Peter Maydell <peter.maydell@linaro.org>" [ultimate] # gpg: aka "Peter Maydell <pmaydell@gmail.com>" [ultimate] # gpg: aka "Peter Maydell <pmaydell@chiark.greenend.org.uk>" [ultimate] # Primary key fingerprint: E1A5 C593 CD41 9DE2 8E83 15CF 3C25 25ED 1436 0CDE * remotes/pmaydell/tags/pull-target-arm-20210323: target/arm: Set ARMMMUFaultInfo.level in user-only arm_cpu_tlb_fill target/arm: Make M-profile VTOR loads on reset handle memory aliasing hw/core/loader: Add new function rom_ptr_for_as() memory: Add offset_in_region to flatview_cb arguments memory: Document flatview_for_each_range() memory: Make flatview_cb return bool, not int hw/arm/virt: Disable pl011 clock migration if needed Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
1a4d83b564
10 changed files with 157 additions and 10 deletions
|
@ -322,10 +322,18 @@ static const MemoryRegionOps pl011_ops = {
|
||||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static bool pl011_clock_needed(void *opaque)
|
||||||
|
{
|
||||||
|
PL011State *s = PL011(opaque);
|
||||||
|
|
||||||
|
return s->migrate_clk;
|
||||||
|
}
|
||||||
|
|
||||||
static const VMStateDescription vmstate_pl011_clock = {
|
static const VMStateDescription vmstate_pl011_clock = {
|
||||||
.name = "pl011/clock",
|
.name = "pl011/clock",
|
||||||
.version_id = 1,
|
.version_id = 1,
|
||||||
.minimum_version_id = 1,
|
.minimum_version_id = 1,
|
||||||
|
.needed = pl011_clock_needed,
|
||||||
.fields = (VMStateField[]) {
|
.fields = (VMStateField[]) {
|
||||||
VMSTATE_CLOCK(clk, PL011State),
|
VMSTATE_CLOCK(clk, PL011State),
|
||||||
VMSTATE_END_OF_LIST()
|
VMSTATE_END_OF_LIST()
|
||||||
|
@ -363,6 +371,7 @@ static const VMStateDescription vmstate_pl011 = {
|
||||||
|
|
||||||
static Property pl011_properties[] = {
|
static Property pl011_properties[] = {
|
||||||
DEFINE_PROP_CHR("chardev", PL011State, chr),
|
DEFINE_PROP_CHR("chardev", PL011State, chr),
|
||||||
|
DEFINE_PROP_BOOL("migrate-clk", PL011State, migrate_clk, true),
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1383,6 +1383,81 @@ void *rom_ptr(hwaddr addr, size_t size)
|
||||||
return rom->data + (addr - rom->addr);
|
return rom->data + (addr - rom->addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct FindRomCBData {
|
||||||
|
size_t size; /* Amount of data we want from ROM, in bytes */
|
||||||
|
MemoryRegion *mr; /* MR at the unaliased guest addr */
|
||||||
|
hwaddr xlat; /* Offset of addr within mr */
|
||||||
|
void *rom; /* Output: rom data pointer, if found */
|
||||||
|
} FindRomCBData;
|
||||||
|
|
||||||
|
static bool find_rom_cb(Int128 start, Int128 len, const MemoryRegion *mr,
|
||||||
|
hwaddr offset_in_region, void *opaque)
|
||||||
|
{
|
||||||
|
FindRomCBData *cbdata = opaque;
|
||||||
|
hwaddr alias_addr;
|
||||||
|
|
||||||
|
if (mr != cbdata->mr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
alias_addr = int128_get64(start) + cbdata->xlat - offset_in_region;
|
||||||
|
cbdata->rom = rom_ptr(alias_addr, cbdata->size);
|
||||||
|
if (!cbdata->rom) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
/* Found a match, stop iterating */
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *rom_ptr_for_as(AddressSpace *as, hwaddr addr, size_t size)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Find any ROM data for the given guest address range. If there
|
||||||
|
* is a ROM blob then return a pointer to the host memory
|
||||||
|
* corresponding to 'addr'; otherwise return NULL.
|
||||||
|
*
|
||||||
|
* We look not only for ROM blobs that were loaded directly to
|
||||||
|
* addr, but also for ROM blobs that were loaded to aliases of
|
||||||
|
* that memory at other addresses within the AddressSpace.
|
||||||
|
*
|
||||||
|
* Note that we do not check @as against the 'as' member in the
|
||||||
|
* 'struct Rom' returned by rom_ptr(). The Rom::as is the
|
||||||
|
* AddressSpace which the rom blob should be written to, whereas
|
||||||
|
* our @as argument is the AddressSpace which we are (effectively)
|
||||||
|
* reading from, and the same underlying RAM will often be visible
|
||||||
|
* in multiple AddressSpaces. (A common example is a ROM blob
|
||||||
|
* written to the 'system' address space but then read back via a
|
||||||
|
* CPU's cpu->as pointer.) This does mean we might potentially
|
||||||
|
* return a false-positive match if a ROM blob was loaded into an
|
||||||
|
* AS which is entirely separate and distinct from the one we're
|
||||||
|
* querying, but this issue exists also for rom_ptr() and hasn't
|
||||||
|
* caused any problems in practice.
|
||||||
|
*/
|
||||||
|
FlatView *fv;
|
||||||
|
void *rom;
|
||||||
|
hwaddr len_unused;
|
||||||
|
FindRomCBData cbdata = {};
|
||||||
|
|
||||||
|
/* Easy case: there's data at the actual address */
|
||||||
|
rom = rom_ptr(addr, size);
|
||||||
|
if (rom) {
|
||||||
|
return rom;
|
||||||
|
}
|
||||||
|
|
||||||
|
RCU_READ_LOCK_GUARD();
|
||||||
|
|
||||||
|
fv = address_space_to_flatview(as);
|
||||||
|
cbdata.mr = flatview_translate(fv, addr, &cbdata.xlat, &len_unused,
|
||||||
|
false, MEMTXATTRS_UNSPECIFIED);
|
||||||
|
if (!cbdata.mr) {
|
||||||
|
/* Nothing at this address, so there can't be any aliasing */
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
cbdata.size = size;
|
||||||
|
flatview_for_each_range(fv, find_rom_cb, &cbdata);
|
||||||
|
return cbdata.rom;
|
||||||
|
}
|
||||||
|
|
||||||
void hmp_info_roms(Monitor *mon, const QDict *qdict)
|
void hmp_info_roms(Monitor *mon, const QDict *qdict)
|
||||||
{
|
{
|
||||||
Rom *rom;
|
Rom *rom;
|
||||||
|
|
|
@ -52,6 +52,7 @@ GlobalProperty hw_compat_5_1[] = {
|
||||||
{ "virtio-scsi-device", "num_queues", "1"},
|
{ "virtio-scsi-device", "num_queues", "1"},
|
||||||
{ "nvme", "use-intel-id", "on"},
|
{ "nvme", "use-intel-id", "on"},
|
||||||
{ "pvpanic", "events", "1"}, /* PVPANIC_PANICKED */
|
{ "pvpanic", "events", "1"}, /* PVPANIC_PANICKED */
|
||||||
|
{ "pl011", "migrate-clk", "off" },
|
||||||
};
|
};
|
||||||
const size_t hw_compat_5_1_len = G_N_ELEMENTS(hw_compat_5_1);
|
const size_t hw_compat_5_1_len = G_N_ELEMENTS(hw_compat_5_1);
|
||||||
|
|
||||||
|
|
|
@ -776,11 +776,35 @@ static inline FlatView *address_space_to_flatview(AddressSpace *as)
|
||||||
return qatomic_rcu_read(&as->current_map);
|
return qatomic_rcu_read(&as->current_map);
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef int (*flatview_cb)(Int128 start,
|
/**
|
||||||
|
* typedef flatview_cb: callback for flatview_for_each_range()
|
||||||
|
*
|
||||||
|
* @start: start address of the range within the FlatView
|
||||||
|
* @len: length of the range in bytes
|
||||||
|
* @mr: MemoryRegion covering this range
|
||||||
|
* @offset_in_region: offset of the first byte of the range within @mr
|
||||||
|
* @opaque: data pointer passed to flatview_for_each_range()
|
||||||
|
*
|
||||||
|
* Returns: true to stop the iteration, false to keep going.
|
||||||
|
*/
|
||||||
|
typedef bool (*flatview_cb)(Int128 start,
|
||||||
Int128 len,
|
Int128 len,
|
||||||
const MemoryRegion*, void*);
|
const MemoryRegion *mr,
|
||||||
|
hwaddr offset_in_region,
|
||||||
|
void *opaque);
|
||||||
|
|
||||||
void flatview_for_each_range(FlatView *fv, flatview_cb cb , void *opaque);
|
/**
|
||||||
|
* flatview_for_each_range: Iterate through a FlatView
|
||||||
|
* @fv: the FlatView to iterate through
|
||||||
|
* @cb: function to call for each range
|
||||||
|
* @opaque: opaque data pointer to pass to @cb
|
||||||
|
*
|
||||||
|
* A FlatView is made up of a list of non-overlapping ranges, each of
|
||||||
|
* which is a slice of a MemoryRegion. This function iterates through
|
||||||
|
* each range in @fv, calling @cb. The callback function can terminate
|
||||||
|
* iteration early by returning 'true'.
|
||||||
|
*/
|
||||||
|
void flatview_for_each_range(FlatView *fv, flatview_cb cb, void *opaque);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct MemoryRegionSection: describes a fragment of a #MemoryRegion
|
* struct MemoryRegionSection: describes a fragment of a #MemoryRegion
|
||||||
|
|
|
@ -50,6 +50,7 @@ struct PL011State {
|
||||||
CharBackend chr;
|
CharBackend chr;
|
||||||
qemu_irq irq[6];
|
qemu_irq irq[6];
|
||||||
Clock *clk;
|
Clock *clk;
|
||||||
|
bool migrate_clk;
|
||||||
const unsigned char *id;
|
const unsigned char *id;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -290,6 +290,37 @@ void rom_transaction_end(bool commit);
|
||||||
|
|
||||||
int rom_copy(uint8_t *dest, hwaddr addr, size_t size);
|
int rom_copy(uint8_t *dest, hwaddr addr, size_t size);
|
||||||
void *rom_ptr(hwaddr addr, size_t size);
|
void *rom_ptr(hwaddr addr, size_t size);
|
||||||
|
/**
|
||||||
|
* rom_ptr_for_as: Return a pointer to ROM blob data for the address
|
||||||
|
* @as: AddressSpace to look for the ROM blob in
|
||||||
|
* @addr: Address within @as
|
||||||
|
* @size: size of data required in bytes
|
||||||
|
*
|
||||||
|
* Returns: pointer into the data which backs the matching ROM blob,
|
||||||
|
* or NULL if no blob covers the address range.
|
||||||
|
*
|
||||||
|
* This function looks for a ROM blob which covers the specified range
|
||||||
|
* of bytes of length @size starting at @addr within the address space
|
||||||
|
* @as. This is useful for code which runs as part of board
|
||||||
|
* initialization or CPU reset which wants to read data that is part
|
||||||
|
* of a user-supplied guest image or other guest memory contents, but
|
||||||
|
* which runs before the ROM loader's reset function has copied the
|
||||||
|
* blobs into guest memory.
|
||||||
|
*
|
||||||
|
* rom_ptr_for_as() will look not just for blobs loaded directly to
|
||||||
|
* the specified address, but also for blobs which were loaded to an
|
||||||
|
* alias of the region at a different location in the AddressSpace.
|
||||||
|
* In other words, if a machine model has RAM at address 0x0000_0000
|
||||||
|
* which is aliased to also appear at 0x1000_0000, rom_ptr_for_as()
|
||||||
|
* will return the correct data whether the guest image was linked and
|
||||||
|
* loaded at 0x0000_0000 or 0x1000_0000. Contrast rom_ptr(), which
|
||||||
|
* will only return data if the image load address is an exact match
|
||||||
|
* with the queried address.
|
||||||
|
*
|
||||||
|
* New code should prefer to use rom_ptr_for_as() instead of
|
||||||
|
* rom_ptr().
|
||||||
|
*/
|
||||||
|
void *rom_ptr_for_as(AddressSpace *as, hwaddr addr, size_t size);
|
||||||
void hmp_info_roms(Monitor *mon, const QDict *qdict);
|
void hmp_info_roms(Monitor *mon, const QDict *qdict);
|
||||||
|
|
||||||
#define rom_add_file_fixed(_f, _a, _i) \
|
#define rom_add_file_fixed(_f, _a, _i) \
|
||||||
|
|
|
@ -671,9 +671,11 @@ void flatview_for_each_range(FlatView *fv, flatview_cb cb , void *opaque)
|
||||||
assert(cb);
|
assert(cb);
|
||||||
|
|
||||||
FOR_EACH_FLAT_RANGE(fr, fv) {
|
FOR_EACH_FLAT_RANGE(fr, fv) {
|
||||||
if (cb(fr->addr.start, fr->addr.size, fr->mr, opaque))
|
if (cb(fr->addr.start, fr->addr.size, fr->mr,
|
||||||
|
fr->offset_in_region, opaque)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static MemoryRegion *memory_region_get_flatview_root(MemoryRegion *mr)
|
static MemoryRegion *memory_region_get_flatview_root(MemoryRegion *mr)
|
||||||
|
|
|
@ -331,7 +331,7 @@ static void arm_cpu_reset(DeviceState *dev)
|
||||||
|
|
||||||
/* Load the initial SP and PC from offset 0 and 4 in the vector table */
|
/* Load the initial SP and PC from offset 0 and 4 in the vector table */
|
||||||
vecbase = env->v7m.vecbase[env->v7m.secure];
|
vecbase = env->v7m.vecbase[env->v7m.secure];
|
||||||
rom = rom_ptr(vecbase, 8);
|
rom = rom_ptr_for_as(s->as, vecbase, 8);
|
||||||
if (rom) {
|
if (rom) {
|
||||||
/* Address zero is covered by ROM which hasn't yet been
|
/* Address zero is covered by ROM which hasn't yet been
|
||||||
* copied into physical memory.
|
* copied into physical memory.
|
||||||
|
|
|
@ -163,6 +163,7 @@ bool arm_cpu_tlb_fill(CPUState *cs, vaddr address, int size,
|
||||||
} else {
|
} else {
|
||||||
fi.type = ARMFault_Translation;
|
fi.type = ARMFault_Translation;
|
||||||
}
|
}
|
||||||
|
fi.level = 3;
|
||||||
|
|
||||||
/* now we have a real cpu fault */
|
/* now we have a real cpu fault */
|
||||||
cpu_restore_state(cs, retaddr, true);
|
cpu_restore_state(cs, retaddr, true);
|
||||||
|
|
|
@ -98,19 +98,22 @@ struct get_io_cb_info {
|
||||||
address_range result;
|
address_range result;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int get_io_address_cb(Int128 start, Int128 size,
|
static bool get_io_address_cb(Int128 start, Int128 size,
|
||||||
const MemoryRegion *mr, void *opaque) {
|
const MemoryRegion *mr,
|
||||||
|
hwaddr offset_in_region,
|
||||||
|
void *opaque)
|
||||||
|
{
|
||||||
struct get_io_cb_info *info = opaque;
|
struct get_io_cb_info *info = opaque;
|
||||||
if (g_hash_table_lookup(fuzzable_memoryregions, mr)) {
|
if (g_hash_table_lookup(fuzzable_memoryregions, mr)) {
|
||||||
if (info->index == 0) {
|
if (info->index == 0) {
|
||||||
info->result.addr = (ram_addr_t)start;
|
info->result.addr = (ram_addr_t)start;
|
||||||
info->result.size = (ram_addr_t)size;
|
info->result.size = (ram_addr_t)size;
|
||||||
info->found = 1;
|
info->found = 1;
|
||||||
return 1;
|
return true;
|
||||||
}
|
}
|
||||||
info->index--;
|
info->index--;
|
||||||
}
|
}
|
||||||
return 0;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue