xref: /kvmtool/mmio.c (revision f588adbbff064bec620968f651cfa9d0b2d9276b)
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