1b2441318SGreg Kroah-Hartman /* SPDX-License-Identifier: GPL-2.0 */ 221ebbedaSXiao Guangrong #ifndef _ASM_X86_KVM_PAGE_TRACK_H 321ebbedaSXiao Guangrong #define _ASM_X86_KVM_PAGE_TRACK_H 421ebbedaSXiao Guangrong 521ebbedaSXiao Guangrong enum kvm_page_track_mode { 621ebbedaSXiao Guangrong KVM_PAGE_TRACK_WRITE, 721ebbedaSXiao Guangrong KVM_PAGE_TRACK_MAX, 821ebbedaSXiao Guangrong }; 921ebbedaSXiao Guangrong 100eb05bf2SXiao Guangrong /* 110eb05bf2SXiao Guangrong * The notifier represented by @kvm_page_track_notifier_node is linked into 120eb05bf2SXiao Guangrong * the head which will be notified when guest is triggering the track event. 130eb05bf2SXiao Guangrong * 140eb05bf2SXiao Guangrong * Write access on the head is protected by kvm->mmu_lock, read access 150eb05bf2SXiao Guangrong * is protected by track_srcu. 160eb05bf2SXiao Guangrong */ 170eb05bf2SXiao Guangrong struct kvm_page_track_notifier_head { 180eb05bf2SXiao Guangrong struct srcu_struct track_srcu; 190eb05bf2SXiao Guangrong struct hlist_head track_notifier_list; 200eb05bf2SXiao Guangrong }; 210eb05bf2SXiao Guangrong 220eb05bf2SXiao Guangrong struct kvm_page_track_notifier_node { 230eb05bf2SXiao Guangrong struct hlist_node node; 240eb05bf2SXiao Guangrong 250eb05bf2SXiao Guangrong /* 260eb05bf2SXiao Guangrong * It is called when guest is writing the write-tracked page 270eb05bf2SXiao Guangrong * and write emulation is finished at that time. 280eb05bf2SXiao Guangrong * 290eb05bf2SXiao Guangrong * @vcpu: the vcpu where the write access happened. 300eb05bf2SXiao Guangrong * @gpa: the physical address written by guest. 310eb05bf2SXiao Guangrong * @new: the data was written to the address. 320eb05bf2SXiao Guangrong * @bytes: the written length. 33d126363dSJike Song * @node: this node 340eb05bf2SXiao Guangrong */ 350eb05bf2SXiao Guangrong void (*track_write)(struct kvm_vcpu *vcpu, gpa_t gpa, const u8 *new, 36d126363dSJike Song int bytes, struct kvm_page_track_notifier_node *node); 37ae7cd873SXiaoguang Chen /* 38ae7cd873SXiaoguang Chen * It is called when memory slot is being moved or removed 39ae7cd873SXiaoguang Chen * users can drop write-protection for the pages in that memory slot 40ae7cd873SXiaoguang Chen * 41ae7cd873SXiaoguang Chen * @kvm: the kvm where memory slot being moved or removed 42ae7cd873SXiaoguang Chen * @slot: the memory slot being moved or removed 43d126363dSJike Song * @node: this node 44ae7cd873SXiaoguang Chen */ 45d126363dSJike Song void (*track_flush_slot)(struct kvm *kvm, struct kvm_memory_slot *slot, 46d126363dSJike Song struct kvm_page_track_notifier_node *node); 470eb05bf2SXiao Guangrong }; 480eb05bf2SXiao Guangrong 49eb7511bfSHaimin Zhang int kvm_page_track_init(struct kvm *kvm); 502beb6dadSPaolo Bonzini void kvm_page_track_cleanup(struct kvm *kvm); 510eb05bf2SXiao Guangrong 52e96c81eeSSean Christopherson void kvm_page_track_free_memslot(struct kvm_memory_slot *slot); 5321ebbedaSXiao Guangrong int kvm_page_track_create_memslot(struct kvm_memory_slot *slot, 5421ebbedaSXiao Guangrong unsigned long npages); 55f29d4d78SXiao Guangrong 56f29d4d78SXiao Guangrong void kvm_slot_page_track_add_page(struct kvm *kvm, 57f29d4d78SXiao Guangrong struct kvm_memory_slot *slot, gfn_t gfn, 58f29d4d78SXiao Guangrong enum kvm_page_track_mode mode); 59f29d4d78SXiao Guangrong void kvm_slot_page_track_remove_page(struct kvm *kvm, 60f29d4d78SXiao Guangrong struct kvm_memory_slot *slot, gfn_t gfn, 61f29d4d78SXiao Guangrong enum kvm_page_track_mode mode); 623d0c27adSXiao Guangrong bool kvm_page_track_is_active(struct kvm_vcpu *vcpu, gfn_t gfn, 633d0c27adSXiao Guangrong enum kvm_page_track_mode mode); 64*88810413SDavid Matlack bool kvm_slot_page_track_is_active(struct kvm_memory_slot *slot, gfn_t gfn, 65*88810413SDavid Matlack enum kvm_page_track_mode mode); 660eb05bf2SXiao Guangrong 670eb05bf2SXiao Guangrong void 680eb05bf2SXiao Guangrong kvm_page_track_register_notifier(struct kvm *kvm, 690eb05bf2SXiao Guangrong struct kvm_page_track_notifier_node *n); 700eb05bf2SXiao Guangrong void 710eb05bf2SXiao Guangrong kvm_page_track_unregister_notifier(struct kvm *kvm, 720eb05bf2SXiao Guangrong struct kvm_page_track_notifier_node *n); 730eb05bf2SXiao Guangrong void kvm_page_track_write(struct kvm_vcpu *vcpu, gpa_t gpa, const u8 *new, 740eb05bf2SXiao Guangrong int bytes); 75ae7cd873SXiaoguang Chen void kvm_page_track_flush_slot(struct kvm *kvm, struct kvm_memory_slot *slot); 7621ebbedaSXiao Guangrong #endif 77