129443dabSPekka Enberg #include "kvm/kvm.h" 26b1994caSSasha Levin #include "kvm/rbtree-interval.h" 3*f588adbbSSasha Levin #include "kvm/brlock.h" 429443dabSPekka Enberg 529443dabSPekka Enberg #include <stdio.h> 66b1994caSSasha Levin #include <stdlib.h> 76b1994caSSasha Levin 83fdf659dSSasha Levin #include <linux/types.h> 96b1994caSSasha Levin #include <linux/rbtree.h> 106b1994caSSasha Levin 116b1994caSSasha Levin #define mmio_node(n) rb_entry(n, struct mmio_mapping, node) 126b1994caSSasha Levin 136b1994caSSasha Levin struct mmio_mapping { 146b1994caSSasha Levin struct rb_int_node node; 156b1994caSSasha Levin void (*kvm_mmio_callback_fn)(u64 addr, u8 *data, u32 len, u8 is_write); 166b1994caSSasha Levin }; 176b1994caSSasha Levin 186b1994caSSasha Levin static struct rb_root mmio_tree = RB_ROOT; 196b1994caSSasha Levin 206b1994caSSasha Levin static struct mmio_mapping *mmio_search(struct rb_root *root, u64 addr, u64 len) 216b1994caSSasha Levin { 226b1994caSSasha Levin struct rb_int_node *node; 236b1994caSSasha Levin 246b1994caSSasha Levin node = rb_int_search_range(root, addr, addr + len); 256b1994caSSasha Levin if (node == NULL) 266b1994caSSasha Levin return NULL; 276b1994caSSasha Levin 286b1994caSSasha Levin return mmio_node(node); 296b1994caSSasha Levin } 306b1994caSSasha Levin 316b1994caSSasha Levin /* Find lowest match, Check for overlap */ 326b1994caSSasha Levin static struct mmio_mapping *mmio_search_single(struct rb_root *root, u64 addr) 336b1994caSSasha Levin { 346b1994caSSasha Levin struct rb_int_node *node; 356b1994caSSasha Levin 366b1994caSSasha Levin node = rb_int_search_single(root, addr); 376b1994caSSasha Levin if (node == NULL) 386b1994caSSasha Levin return NULL; 396b1994caSSasha Levin 406b1994caSSasha Levin return mmio_node(node); 416b1994caSSasha Levin } 426b1994caSSasha Levin 436b1994caSSasha Levin static int mmio_insert(struct rb_root *root, struct mmio_mapping *data) 446b1994caSSasha Levin { 456b1994caSSasha Levin return rb_int_insert(root, &data->node); 466b1994caSSasha Levin } 4729443dabSPekka Enberg 483fdf659dSSasha Levin static const char *to_direction(u8 is_write) 4929443dabSPekka Enberg { 5029443dabSPekka Enberg if (is_write) 5129443dabSPekka Enberg return "write"; 5229443dabSPekka Enberg 5329443dabSPekka Enberg return "read"; 5429443dabSPekka Enberg } 5529443dabSPekka Enberg 566b1994caSSasha 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)) 576b1994caSSasha Levin { 586b1994caSSasha Levin struct mmio_mapping *mmio; 59*f588adbbSSasha Levin int ret; 606b1994caSSasha Levin 616b1994caSSasha Levin mmio = malloc(sizeof(*mmio)); 626b1994caSSasha Levin if (mmio == NULL) 636b1994caSSasha Levin return false; 646b1994caSSasha Levin 656b1994caSSasha Levin *mmio = (struct mmio_mapping) { 666b1994caSSasha Levin .node = RB_INT_INIT(phys_addr, phys_addr + phys_addr_len), 676b1994caSSasha Levin .kvm_mmio_callback_fn = kvm_mmio_callback_fn, 686b1994caSSasha Levin }; 696b1994caSSasha Levin 70*f588adbbSSasha Levin br_write_lock(); 71*f588adbbSSasha Levin ret = mmio_insert(&mmio_tree, mmio); 72*f588adbbSSasha Levin br_write_unlock(); 73*f588adbbSSasha Levin 74*f588adbbSSasha Levin return ret; 756b1994caSSasha Levin } 766b1994caSSasha Levin 776b1994caSSasha Levin bool kvm__deregister_mmio(u64 phys_addr) 786b1994caSSasha Levin { 796b1994caSSasha Levin struct mmio_mapping *mmio; 806b1994caSSasha Levin 81*f588adbbSSasha Levin br_write_lock(); 826b1994caSSasha Levin mmio = mmio_search_single(&mmio_tree, phys_addr); 83*f588adbbSSasha Levin if (mmio == NULL) { 84*f588adbbSSasha Levin br_write_unlock(); 856b1994caSSasha Levin return false; 86*f588adbbSSasha Levin } 876b1994caSSasha Levin 886b1994caSSasha Levin rb_int_erase(&mmio_tree, &mmio->node); 89*f588adbbSSasha Levin br_write_unlock(); 90*f588adbbSSasha Levin 916b1994caSSasha Levin free(mmio); 926b1994caSSasha Levin return true; 936b1994caSSasha Levin } 946b1994caSSasha Levin 9543835ac9SSasha Levin bool kvm__emulate_mmio(struct kvm *kvm, u64 phys_addr, u8 *data, u32 len, u8 is_write) 9629443dabSPekka Enberg { 97*f588adbbSSasha Levin struct mmio_mapping *mmio; 98*f588adbbSSasha Levin 99*f588adbbSSasha Levin br_read_lock(); 100*f588adbbSSasha Levin mmio = mmio_search(&mmio_tree, phys_addr, len); 1016b1994caSSasha Levin 1026b1994caSSasha Levin if (mmio) 1036b1994caSSasha Levin mmio->kvm_mmio_callback_fn(phys_addr, data, len, is_write); 1046b1994caSSasha Levin else 1053fdf659dSSasha Levin fprintf(stderr, "Warning: Ignoring MMIO %s at %016llx (length %u)\n", 10629443dabSPekka Enberg to_direction(is_write), phys_addr, len); 107*f588adbbSSasha Levin br_read_unlock(); 10829443dabSPekka Enberg 10929443dabSPekka Enberg return true; 11029443dabSPekka Enberg } 111