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