129443dabSPekka Enberg #include "kvm/kvm.h" 2*6b1994caSSasha Levin #include "kvm/rbtree-interval.h" 329443dabSPekka Enberg 429443dabSPekka Enberg #include <stdio.h> 5*6b1994caSSasha Levin #include <stdlib.h> 6*6b1994caSSasha Levin 73fdf659dSSasha Levin #include <linux/types.h> 8*6b1994caSSasha Levin #include <linux/rbtree.h> 9*6b1994caSSasha Levin 10*6b1994caSSasha Levin #define mmio_node(n) rb_entry(n, struct mmio_mapping, node) 11*6b1994caSSasha Levin 12*6b1994caSSasha Levin struct mmio_mapping { 13*6b1994caSSasha Levin struct rb_int_node node; 14*6b1994caSSasha Levin void (*kvm_mmio_callback_fn)(u64 addr, u8 *data, u32 len, u8 is_write); 15*6b1994caSSasha Levin }; 16*6b1994caSSasha Levin 17*6b1994caSSasha Levin static struct rb_root mmio_tree = RB_ROOT; 18*6b1994caSSasha Levin 19*6b1994caSSasha Levin static struct mmio_mapping *mmio_search(struct rb_root *root, u64 addr, u64 len) 20*6b1994caSSasha Levin { 21*6b1994caSSasha Levin struct rb_int_node *node; 22*6b1994caSSasha Levin 23*6b1994caSSasha Levin node = rb_int_search_range(root, addr, addr + len); 24*6b1994caSSasha Levin if (node == NULL) 25*6b1994caSSasha Levin return NULL; 26*6b1994caSSasha Levin 27*6b1994caSSasha Levin return mmio_node(node); 28*6b1994caSSasha Levin } 29*6b1994caSSasha Levin 30*6b1994caSSasha Levin /* Find lowest match, Check for overlap */ 31*6b1994caSSasha Levin static struct mmio_mapping *mmio_search_single(struct rb_root *root, u64 addr) 32*6b1994caSSasha Levin { 33*6b1994caSSasha Levin struct rb_int_node *node; 34*6b1994caSSasha Levin 35*6b1994caSSasha Levin node = rb_int_search_single(root, addr); 36*6b1994caSSasha Levin if (node == NULL) 37*6b1994caSSasha Levin return NULL; 38*6b1994caSSasha Levin 39*6b1994caSSasha Levin return mmio_node(node); 40*6b1994caSSasha Levin } 41*6b1994caSSasha Levin 42*6b1994caSSasha Levin static int mmio_insert(struct rb_root *root, struct mmio_mapping *data) 43*6b1994caSSasha Levin { 44*6b1994caSSasha Levin return rb_int_insert(root, &data->node); 45*6b1994caSSasha Levin } 4629443dabSPekka Enberg 473fdf659dSSasha Levin static const char *to_direction(u8 is_write) 4829443dabSPekka Enberg { 4929443dabSPekka Enberg if (is_write) 5029443dabSPekka Enberg return "write"; 5129443dabSPekka Enberg 5229443dabSPekka Enberg return "read"; 5329443dabSPekka Enberg } 5429443dabSPekka Enberg 55*6b1994caSSasha Levin bool kvm__register_mmio(u64 phys_addr, u64 phys_addr_len, void (*kvm_mmio_callback_fn)(u64 addr, u8 *data, u32 len, u8 is_write)) 56*6b1994caSSasha Levin { 57*6b1994caSSasha Levin struct mmio_mapping *mmio; 58*6b1994caSSasha Levin 59*6b1994caSSasha Levin mmio = malloc(sizeof(*mmio)); 60*6b1994caSSasha Levin if (mmio == NULL) 61*6b1994caSSasha Levin return false; 62*6b1994caSSasha Levin 63*6b1994caSSasha Levin *mmio = (struct mmio_mapping) { 64*6b1994caSSasha Levin .node = RB_INT_INIT(phys_addr, phys_addr + phys_addr_len), 65*6b1994caSSasha Levin .kvm_mmio_callback_fn = kvm_mmio_callback_fn, 66*6b1994caSSasha Levin }; 67*6b1994caSSasha Levin 68*6b1994caSSasha Levin return mmio_insert(&mmio_tree, mmio); 69*6b1994caSSasha Levin } 70*6b1994caSSasha Levin 71*6b1994caSSasha Levin bool kvm__deregister_mmio(u64 phys_addr) 72*6b1994caSSasha Levin { 73*6b1994caSSasha Levin struct mmio_mapping *mmio; 74*6b1994caSSasha Levin 75*6b1994caSSasha Levin mmio = mmio_search_single(&mmio_tree, phys_addr); 76*6b1994caSSasha Levin if (mmio == NULL) 77*6b1994caSSasha Levin return false; 78*6b1994caSSasha Levin 79*6b1994caSSasha Levin rb_int_erase(&mmio_tree, &mmio->node); 80*6b1994caSSasha Levin free(mmio); 81*6b1994caSSasha Levin return true; 82*6b1994caSSasha Levin } 83*6b1994caSSasha Levin 8443835ac9SSasha Levin bool kvm__emulate_mmio(struct kvm *kvm, u64 phys_addr, u8 *data, u32 len, u8 is_write) 8529443dabSPekka Enberg { 86*6b1994caSSasha Levin struct mmio_mapping *mmio = mmio_search(&mmio_tree, phys_addr, len); 87*6b1994caSSasha Levin 88*6b1994caSSasha Levin if (mmio) 89*6b1994caSSasha Levin mmio->kvm_mmio_callback_fn(phys_addr, data, len, is_write); 90*6b1994caSSasha Levin else 913fdf659dSSasha Levin fprintf(stderr, "Warning: Ignoring MMIO %s at %016llx (length %u)\n", 9229443dabSPekka Enberg to_direction(is_write), phys_addr, len); 9329443dabSPekka Enberg 9429443dabSPekka Enberg return true; 9529443dabSPekka Enberg } 96