mirror of
				https://git.proxmox.com/git/qemu
				synced 2025-10-25 23:04:04 +00:00 
			
		
		
		
	 b40acf99be
			
		
	
	
		b40acf99be
		
	
	
	
	
		
			
			The current ioport dispatcher is a complex beast, mostly due to the need to deal with old portio interface users. But we can overcome it without converting all portio users by embedding the required base address of a MemoryRegionPortio access into that data structure. That removes the need to have the additional MemoryRegionIORange structure in the loop on every access. To handle old portio memory ops, we simply install dispatching handlers for portio memory regions when registering them with the memory core. This removes the need for the old_portio field. We can drop the additional aliasing of ioport regions and also the special address space listener. cpu_in and cpu_out now simply call address_space_read/write. And we can concentrate portio handling in a single source file. Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
		
			
				
	
	
		
			125 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			125 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Declarations for obsolete exec.c functions
 | |
|  *
 | |
|  * Copyright 2011 Red Hat, Inc. and/or its affiliates
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Avi Kivity <avi@redhat.com>
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or
 | |
|  * later.  See the COPYING file in the top-level directory.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * This header is for use by exec.c and memory.c ONLY.  Do not include it.
 | |
|  * The functions declared here will be removed soon.
 | |
|  */
 | |
| 
 | |
| #ifndef MEMORY_INTERNAL_H
 | |
| #define MEMORY_INTERNAL_H
 | |
| 
 | |
| #ifndef CONFIG_USER_ONLY
 | |
| #include "hw/xen/xen.h"
 | |
| 
 | |
| 
 | |
| typedef struct AddressSpaceDispatch AddressSpaceDispatch;
 | |
| 
 | |
| void address_space_init_dispatch(AddressSpace *as);
 | |
| void address_space_destroy_dispatch(AddressSpace *as);
 | |
| 
 | |
| extern const MemoryRegionOps unassigned_mem_ops;
 | |
| 
 | |
| bool memory_region_access_valid(MemoryRegion *mr, hwaddr addr,
 | |
|                                 unsigned size, bool is_write);
 | |
| 
 | |
| ram_addr_t qemu_ram_alloc_from_ptr(ram_addr_t size, void *host,
 | |
|                                    MemoryRegion *mr);
 | |
| ram_addr_t qemu_ram_alloc(ram_addr_t size, MemoryRegion *mr);
 | |
| void *qemu_get_ram_ptr(ram_addr_t addr);
 | |
| void qemu_ram_free(ram_addr_t addr);
 | |
| void qemu_ram_free_from_ptr(ram_addr_t addr);
 | |
| 
 | |
| #define VGA_DIRTY_FLAG       0x01
 | |
| #define CODE_DIRTY_FLAG      0x02
 | |
| #define MIGRATION_DIRTY_FLAG 0x08
 | |
| 
 | |
| static inline int cpu_physical_memory_get_dirty_flags(ram_addr_t addr)
 | |
| {
 | |
|     return ram_list.phys_dirty[addr >> TARGET_PAGE_BITS];
 | |
| }
 | |
| 
 | |
| /* read dirty bit (return 0 or 1) */
 | |
| static inline int cpu_physical_memory_is_dirty(ram_addr_t addr)
 | |
| {
 | |
|     return cpu_physical_memory_get_dirty_flags(addr) == 0xff;
 | |
| }
 | |
| 
 | |
| static inline int cpu_physical_memory_get_dirty(ram_addr_t start,
 | |
|                                                 ram_addr_t length,
 | |
|                                                 int dirty_flags)
 | |
| {
 | |
|     int ret = 0;
 | |
|     ram_addr_t addr, end;
 | |
| 
 | |
|     end = TARGET_PAGE_ALIGN(start + length);
 | |
|     start &= TARGET_PAGE_MASK;
 | |
|     for (addr = start; addr < end; addr += TARGET_PAGE_SIZE) {
 | |
|         ret |= cpu_physical_memory_get_dirty_flags(addr) & dirty_flags;
 | |
|     }
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static inline int cpu_physical_memory_set_dirty_flags(ram_addr_t addr,
 | |
|                                                       int dirty_flags)
 | |
| {
 | |
|     return ram_list.phys_dirty[addr >> TARGET_PAGE_BITS] |= dirty_flags;
 | |
| }
 | |
| 
 | |
| static inline void cpu_physical_memory_set_dirty(ram_addr_t addr)
 | |
| {
 | |
|     cpu_physical_memory_set_dirty_flags(addr, 0xff);
 | |
| }
 | |
| 
 | |
| static inline int cpu_physical_memory_clear_dirty_flags(ram_addr_t addr,
 | |
|                                                         int dirty_flags)
 | |
| {
 | |
|     int mask = ~dirty_flags;
 | |
| 
 | |
|     return ram_list.phys_dirty[addr >> TARGET_PAGE_BITS] &= mask;
 | |
| }
 | |
| 
 | |
| static inline void cpu_physical_memory_set_dirty_range(ram_addr_t start,
 | |
|                                                        ram_addr_t length,
 | |
|                                                        int dirty_flags)
 | |
| {
 | |
|     ram_addr_t addr, end;
 | |
| 
 | |
|     end = TARGET_PAGE_ALIGN(start + length);
 | |
|     start &= TARGET_PAGE_MASK;
 | |
|     for (addr = start; addr < end; addr += TARGET_PAGE_SIZE) {
 | |
|         cpu_physical_memory_set_dirty_flags(addr, dirty_flags);
 | |
|     }
 | |
|     xen_modified_memory(addr, length);
 | |
| }
 | |
| 
 | |
| static inline void cpu_physical_memory_mask_dirty_range(ram_addr_t start,
 | |
|                                                         ram_addr_t length,
 | |
|                                                         int dirty_flags)
 | |
| {
 | |
|     ram_addr_t addr, end;
 | |
| 
 | |
|     end = TARGET_PAGE_ALIGN(start + length);
 | |
|     start &= TARGET_PAGE_MASK;
 | |
|     for (addr = start; addr < end; addr += TARGET_PAGE_SIZE) {
 | |
|         cpu_physical_memory_clear_dirty_flags(addr, dirty_flags);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void cpu_physical_memory_reset_dirty(ram_addr_t start, ram_addr_t end,
 | |
|                                      int dirty_flags);
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #endif
 |