mirror of
https://github.com/projectacrn/acrn-hypervisor.git
synced 2025-07-29 22:47:21 +00:00
hv: get correct fault address for copy_to/from_gva
When doing copy_to/from_gva, it's possible the guest no page happens on none-first page. In this case, we need get correct fault address from gva2gpa. Signed-off-by: Yin Fengwei <fengwei.yin@intel.com> Reviewed-by: Jason Chen CJ <jason.cj.chen@intel.com> Acked-by: Anthony Xu <anthony.xu@intel.com> Acked-by: Eddie Dong <Eddie.dong@intel.com>
This commit is contained in:
parent
55105dbdeb
commit
63fe48c27f
@ -375,7 +375,8 @@ static inline int copy_gpa(struct vm *vm, void *h_ptr_arg, uint64_t gpa_arg,
|
||||
}
|
||||
|
||||
static inline int copy_gva(struct vcpu *vcpu, void *h_ptr_arg, uint64_t gva_arg,
|
||||
uint32_t size_arg, uint32_t *err_code, bool cp_from_vm)
|
||||
uint32_t size_arg, uint32_t *err_code, uint64_t *fault_addr,
|
||||
bool cp_from_vm)
|
||||
{
|
||||
void *h_ptr = h_ptr_arg;
|
||||
uint64_t gpa = 0UL;
|
||||
@ -396,6 +397,7 @@ static inline int copy_gva(struct vcpu *vcpu, void *h_ptr_arg, uint64_t gva_arg,
|
||||
while (size > 0U) {
|
||||
ret = gva2gpa(vcpu, gva, &gpa, err_code);
|
||||
if (ret < 0) {
|
||||
*fault_addr = gva;
|
||||
pr_err("error[%d] in GVA2GPA, err_code=0x%x",
|
||||
ret, *err_code);
|
||||
return ret;
|
||||
@ -433,15 +435,15 @@ int copy_to_gpa(struct vm *vm, void *h_ptr, uint64_t gpa, uint32_t size)
|
||||
}
|
||||
|
||||
int copy_from_gva(struct vcpu *vcpu, void *h_ptr, uint64_t gva,
|
||||
uint32_t size, uint32_t *err_code)
|
||||
uint32_t size, uint32_t *err_code, uint64_t *fault_addr)
|
||||
{
|
||||
return copy_gva(vcpu, h_ptr, gva, size, err_code, 1);
|
||||
return copy_gva(vcpu, h_ptr, gva, size, err_code, fault_addr, 1);
|
||||
}
|
||||
|
||||
int copy_to_gva(struct vcpu *vcpu, void *h_ptr, uint64_t gva,
|
||||
uint32_t size, uint32_t *err_code)
|
||||
uint32_t size, uint32_t *err_code, uint64_t *fault_addr)
|
||||
{
|
||||
return copy_gva(vcpu, h_ptr, gva, size, err_code, 0);
|
||||
return copy_gva(vcpu, h_ptr, gva, size, err_code, fault_addr, 0);
|
||||
}
|
||||
|
||||
void init_e820(void)
|
||||
|
@ -1773,6 +1773,7 @@ static int vie_init(struct instr_emul_vie *vie, struct vcpu *vcpu)
|
||||
uint64_t guest_rip_gva = vcpu_get_rip(vcpu);
|
||||
uint32_t inst_len = vcpu->arch_vcpu.inst_len;
|
||||
uint32_t err_code;
|
||||
uint64_t fault_addr;
|
||||
int ret;
|
||||
|
||||
if (inst_len > VIE_INST_SIZE || inst_len == 0U) {
|
||||
@ -1785,10 +1786,10 @@ static int vie_init(struct instr_emul_vie *vie, struct vcpu *vcpu)
|
||||
|
||||
err_code = PAGE_FAULT_ID_FLAG;
|
||||
ret = copy_from_gva(vcpu, vie->inst, guest_rip_gva,
|
||||
inst_len, &err_code);
|
||||
inst_len, &err_code, &fault_addr);
|
||||
if (ret < 0) {
|
||||
if (ret == -EFAULT) {
|
||||
vcpu_inject_pf(vcpu, guest_rip_gva, err_code);
|
||||
vcpu_inject_pf(vcpu, fault_addr, err_code);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
@ -27,7 +27,7 @@ uint64_t get_microcode_version(void)
|
||||
((uhdr.data_size != 0U) ? uhdr.data_size : 2000U)
|
||||
void acrn_update_ucode(struct vcpu *vcpu, uint64_t v)
|
||||
{
|
||||
uint64_t gva;
|
||||
uint64_t gva, fault_addr;
|
||||
struct ucode_header uhdr;
|
||||
uint32_t data_page_num;
|
||||
size_t data_size;
|
||||
@ -38,10 +38,11 @@ void acrn_update_ucode(struct vcpu *vcpu, uint64_t v)
|
||||
gva = v - sizeof(struct ucode_header);
|
||||
|
||||
err_code = 0U;
|
||||
err = copy_from_gva(vcpu, &uhdr, gva, sizeof(uhdr), &err_code);
|
||||
err = copy_from_gva(vcpu, &uhdr, gva, sizeof(uhdr), &err_code,
|
||||
&fault_addr);
|
||||
if (err < 0) {
|
||||
if (err == -EFAULT) {
|
||||
vcpu_inject_pf(vcpu, gva, err_code);
|
||||
vcpu_inject_pf(vcpu, fault_addr, err_code);
|
||||
}
|
||||
return;
|
||||
}
|
||||
@ -56,10 +57,11 @@ void acrn_update_ucode(struct vcpu *vcpu, uint64_t v)
|
||||
}
|
||||
|
||||
err_code = 0U;
|
||||
err = copy_from_gva(vcpu, ucode_ptr, gva, data_size, &err_code);
|
||||
err = copy_from_gva(vcpu, ucode_ptr, gva, data_size, &err_code,
|
||||
&fault_addr);
|
||||
if (err < 0) {
|
||||
if (err == -EFAULT) {
|
||||
vcpu_inject_pf(vcpu, gva, err_code);
|
||||
vcpu_inject_pf(vcpu, fault_addr, err_code);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
@ -97,11 +97,11 @@ static void dump_guest_reg(struct vcpu *vcpu)
|
||||
static void dump_guest_stack(struct vcpu *vcpu)
|
||||
{
|
||||
uint32_t i;
|
||||
uint64_t tmp[DUMP_STACK_SIZE];
|
||||
uint64_t tmp[DUMP_STACK_SIZE], fault_addr;
|
||||
uint32_t err_code = 0;
|
||||
|
||||
if (copy_from_gva(vcpu, tmp, vcpu_get_gpreg(vcpu, CPU_REG_RSP),
|
||||
DUMP_STACK_SIZE, &err_code) < 0) {
|
||||
DUMP_STACK_SIZE, &err_code, &fault_addr) < 0) {
|
||||
printf("\r\nUnabled to Copy Guest Stack:\r\n");
|
||||
return;
|
||||
}
|
||||
@ -143,11 +143,11 @@ static void show_guest_call_trace(struct vcpu *vcpu)
|
||||
* if the address is invalid, it will cause hv page fault
|
||||
* then halt system */
|
||||
while ((count < CALL_TRACE_HIERARCHY_MAX) && (bp != 0)) {
|
||||
uint64_t parent_bp = 0UL;
|
||||
uint64_t parent_bp = 0UL, fault_addr;
|
||||
|
||||
err_code = 0U;
|
||||
err = copy_from_gva(vcpu, &parent_bp, bp, sizeof(parent_bp),
|
||||
&err_code);
|
||||
&err_code, &fault_addr);
|
||||
if (err < 0) {
|
||||
printf("\r\nUnabled to get Guest parent BP\r\n");
|
||||
return;
|
||||
|
@ -592,7 +592,7 @@ int shell_vcpu_dumpreg(int argc, char **argv)
|
||||
char temp_str[MAX_STR_SIZE];
|
||||
struct vm *vm;
|
||||
struct vcpu *vcpu;
|
||||
uint64_t i;
|
||||
uint64_t i, fault_addr;
|
||||
uint64_t tmp[DUMPREG_SP_SIZE];
|
||||
uint32_t err_code = 0;
|
||||
|
||||
@ -676,7 +676,8 @@ int shell_vcpu_dumpreg(int argc, char **argv)
|
||||
|
||||
/* dump sp */
|
||||
status = copy_from_gva(vcpu, tmp, vcpu_get_gpreg(vcpu, CPU_REG_RSP),
|
||||
DUMPREG_SP_SIZE*sizeof(uint64_t), &err_code);
|
||||
DUMPREG_SP_SIZE*sizeof(uint64_t), &err_code,
|
||||
&fault_addr);
|
||||
if (status < 0) {
|
||||
/* copy_from_gva fail */
|
||||
shell_puts("Cannot handle user gva yet!\r\n");
|
||||
|
@ -113,9 +113,9 @@ extern vm_sw_loader_t vm_sw_loader;
|
||||
int copy_from_gpa(struct vm *vm, void *h_ptr, uint64_t gpa, uint32_t size);
|
||||
int copy_to_gpa(struct vm *vm, void *h_ptr, uint64_t gpa, uint32_t size);
|
||||
int copy_from_gva(struct vcpu *vcpu, void *h_ptr, uint64_t gva,
|
||||
uint32_t size, uint32_t *err_code);
|
||||
uint32_t size, uint32_t *err_code, uint64_t *fault_addr);
|
||||
int copy_to_gva(struct vcpu *vcpu, void *h_ptr, uint64_t gva,
|
||||
uint32_t size, uint32_t *err_code);
|
||||
uint32_t size, uint32_t *err_code, uint64_t *fault_addr);
|
||||
|
||||
uint64_t create_guest_init_gdt(struct vm *vm, uint32_t *limit);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user