mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-16 06:31:51 -06:00
exec/ioport: Add portio_list_set_address()
Some SuperI/O devices such as the VIA south bridges or the PC87312 controller are able to relocate their SuperI/O functions. Add a convenience function for implementing this in the VIA south bridges. This convenience function relies on previous simplifications in exec/ioport which avoids some duplicate synchronization of I/O port base addresses. The naming of the function is inspired by its memory_region_set_address() pendant. Signed-off-by: Bernhard Beschow <shentey@gmail.com> Message-Id: <20240114123911.4877-6-shentey@gmail.com> Reviewed-by: Michael S. Tsirkin <mst@redhat.com> Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
This commit is contained in:
parent
4edee342f8
commit
ad2b652341
3 changed files with 24 additions and 2 deletions
|
@ -431,10 +431,10 @@ data doesn't match the stored device data well; it allows an
|
||||||
intermediate temporary structure to be populated with migration
|
intermediate temporary structure to be populated with migration
|
||||||
data and then transferred to the main structure.
|
data and then transferred to the main structure.
|
||||||
|
|
||||||
If you use memory API functions that update memory layout outside
|
If you use memory or portio_list API functions that update memory layout outside
|
||||||
initialization (i.e., in response to a guest action), this is a strong
|
initialization (i.e., in response to a guest action), this is a strong
|
||||||
indication that you need to call these functions in a ``post_load`` callback.
|
indication that you need to call these functions in a ``post_load`` callback.
|
||||||
Examples of such memory API functions are:
|
Examples of such API functions are:
|
||||||
|
|
||||||
- memory_region_add_subregion()
|
- memory_region_add_subregion()
|
||||||
- memory_region_del_subregion()
|
- memory_region_del_subregion()
|
||||||
|
@ -443,6 +443,7 @@ Examples of such memory API functions are:
|
||||||
- memory_region_set_enabled()
|
- memory_region_set_enabled()
|
||||||
- memory_region_set_address()
|
- memory_region_set_address()
|
||||||
- memory_region_set_alias_offset()
|
- memory_region_set_alias_offset()
|
||||||
|
- portio_list_set_address()
|
||||||
|
|
||||||
Iterative device migration
|
Iterative device migration
|
||||||
--------------------------
|
--------------------------
|
||||||
|
|
|
@ -54,6 +54,7 @@ typedef struct PortioList {
|
||||||
const struct MemoryRegionPortio *ports;
|
const struct MemoryRegionPortio *ports;
|
||||||
Object *owner;
|
Object *owner;
|
||||||
struct MemoryRegion *address_space;
|
struct MemoryRegion *address_space;
|
||||||
|
uint32_t addr;
|
||||||
unsigned nr;
|
unsigned nr;
|
||||||
struct MemoryRegion **regions;
|
struct MemoryRegion **regions;
|
||||||
void *opaque;
|
void *opaque;
|
||||||
|
@ -70,5 +71,6 @@ void portio_list_add(PortioList *piolist,
|
||||||
struct MemoryRegion *address_space,
|
struct MemoryRegion *address_space,
|
||||||
uint32_t addr);
|
uint32_t addr);
|
||||||
void portio_list_del(PortioList *piolist);
|
void portio_list_del(PortioList *piolist);
|
||||||
|
void portio_list_set_address(PortioList *piolist, uint32_t addr);
|
||||||
|
|
||||||
#endif /* IOPORT_H */
|
#endif /* IOPORT_H */
|
||||||
|
|
|
@ -133,6 +133,7 @@ void portio_list_init(PortioList *piolist,
|
||||||
piolist->nr = 0;
|
piolist->nr = 0;
|
||||||
piolist->regions = g_new0(MemoryRegion *, n);
|
piolist->regions = g_new0(MemoryRegion *, n);
|
||||||
piolist->address_space = NULL;
|
piolist->address_space = NULL;
|
||||||
|
piolist->addr = 0;
|
||||||
piolist->opaque = opaque;
|
piolist->opaque = opaque;
|
||||||
piolist->owner = owner;
|
piolist->owner = owner;
|
||||||
piolist->name = name;
|
piolist->name = name;
|
||||||
|
@ -282,6 +283,7 @@ void portio_list_add(PortioList *piolist,
|
||||||
unsigned int off_low, off_high, off_last, count;
|
unsigned int off_low, off_high, off_last, count;
|
||||||
|
|
||||||
piolist->address_space = address_space;
|
piolist->address_space = address_space;
|
||||||
|
piolist->addr = start;
|
||||||
|
|
||||||
/* Handle the first entry specially. */
|
/* Handle the first entry specially. */
|
||||||
off_last = off_low = pio_start->offset;
|
off_last = off_low = pio_start->offset;
|
||||||
|
@ -322,6 +324,23 @@ void portio_list_del(PortioList *piolist)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void portio_list_set_address(PortioList *piolist, uint32_t addr)
|
||||||
|
{
|
||||||
|
MemoryRegionPortioList *mrpio;
|
||||||
|
unsigned i, j;
|
||||||
|
|
||||||
|
for (i = 0; i < piolist->nr; ++i) {
|
||||||
|
mrpio = container_of(piolist->regions[i], MemoryRegionPortioList, mr);
|
||||||
|
memory_region_set_address(&mrpio->mr,
|
||||||
|
mrpio->mr.addr - piolist->addr + addr);
|
||||||
|
for (j = 0; mrpio->ports[j].size; ++j) {
|
||||||
|
mrpio->ports[j].offset += addr - piolist->addr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
piolist->addr = addr;
|
||||||
|
}
|
||||||
|
|
||||||
static void memory_region_portio_list_finalize(Object *obj)
|
static void memory_region_portio_list_finalize(Object *obj)
|
||||||
{
|
{
|
||||||
MemoryRegionPortioList *mrpio = MEMORY_REGION_PORTIO_LIST(obj);
|
MemoryRegionPortioList *mrpio = MEMORY_REGION_PORTIO_LIST(obj);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue