This implements the Nested KVM HV hcall API for spapr under TCG. The L2 is switched in when the H_ENTER_NESTED hcall is made, and the L1 is switched back in returned from the hcall when a HV exception is sent to the vhyp. Register state is copied in and out according to the nested KVM HV hcall API specification. The hdecr timer is started when the L2 is switched in, and it provides the HDEC / 0x980 return to L1. The MMU re-uses the bare metal radix 2-level page table walker by using the get_pate method to point the MMU to the nested partition table entry. MMU faults due to partition scope errors raise HV exceptions and accordingly are routed back to the L1. The MMU does not tag translations for the L1 (direct) vs L2 (nested) guests, so the TLB is flushed on any L1<->L2 transition (hcall entry and exit). Reviewed-by: Fabiano Rosas <farosas@linux.ibm.com> Signed-off-by: Nicholas Piggin <npiggin@gmail.com> Reviewed-by: Cédric Le Goater <clg@kaod.org> [ clg: checkpatch fixes ] Message-Id: <20220216102545.1808018-10-npiggin@gmail.com> Signed-off-by: Cédric Le Goater <clg@kaod.org>
		
			
				
	
	
		
			64 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			64 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * sPAPR CPU core device.
 | |
|  *
 | |
|  * Copyright (C) 2016 Bharata B Rao <bharata@linux.vnet.ibm.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.
 | |
|  */
 | |
| #ifndef HW_SPAPR_CPU_CORE_H
 | |
| #define HW_SPAPR_CPU_CORE_H
 | |
| 
 | |
| #include "hw/cpu/core.h"
 | |
| #include "hw/qdev-core.h"
 | |
| #include "target/ppc/cpu-qom.h"
 | |
| #include "target/ppc/cpu.h"
 | |
| #include "qom/object.h"
 | |
| 
 | |
| #define TYPE_SPAPR_CPU_CORE "spapr-cpu-core"
 | |
| OBJECT_DECLARE_TYPE(SpaprCpuCore, SpaprCpuCoreClass,
 | |
|                     SPAPR_CPU_CORE)
 | |
| 
 | |
| #define SPAPR_CPU_CORE_TYPE_NAME(model) model "-" TYPE_SPAPR_CPU_CORE
 | |
| 
 | |
| struct SpaprCpuCore {
 | |
|     /*< private >*/
 | |
|     CPUCore parent_obj;
 | |
| 
 | |
|     /*< public >*/
 | |
|     PowerPCCPU **threads;
 | |
|     int node_id;
 | |
|     bool pre_3_0_migration; /* older machine don't know about SpaprCpuState */
 | |
| };
 | |
| 
 | |
| struct SpaprCpuCoreClass {
 | |
|     DeviceClass parent_class;
 | |
|     const char *cpu_type;
 | |
| };
 | |
| 
 | |
| const char *spapr_get_cpu_core_type(const char *cpu_type);
 | |
| void spapr_cpu_set_entry_state(PowerPCCPU *cpu, target_ulong nip,
 | |
|                                target_ulong r1, target_ulong r3,
 | |
|                                target_ulong r4);
 | |
| 
 | |
| typedef struct SpaprCpuState {
 | |
|     uint64_t vpa_addr;
 | |
|     uint64_t slb_shadow_addr, slb_shadow_size;
 | |
|     uint64_t dtl_addr, dtl_size;
 | |
|     bool prod; /* not migrated, only used to improve dispatch latencies */
 | |
|     struct ICPState *icp;
 | |
|     struct XiveTCTX *tctx;
 | |
| 
 | |
|     /* Fields for nested-HV support */
 | |
|     bool in_nested; /* true while the L2 is executing */
 | |
|     CPUPPCState *nested_host_state; /* holds the L1 state while L2 executes */
 | |
|     int64_t nested_tb_offset; /* L1->L2 TB offset */
 | |
| } SpaprCpuState;
 | |
| 
 | |
| static inline SpaprCpuState *spapr_cpu_state(PowerPCCPU *cpu)
 | |
| {
 | |
|     return (SpaprCpuState *)cpu->machine_data;
 | |
| }
 | |
| 
 | |
| #endif
 |