2 Palacios main control interface
7 #include <linux/module.h>
8 #include <linux/moduleparam.h>
9 #include <linux/errno.h>
10 #include <linux/percpu.h>
12 #include <linux/uaccess.h>
13 #include <linux/device.h>
14 #include <linux/cdev.h>
18 #include <linux/file.h>
19 #include <linux/spinlock.h>
20 #include <linux/kthread.h>
22 #include <linux/proc_fs.h>
28 #include "linux-exts.h"
32 MODULE_LICENSE("GPL");
35 int cpu_list[NR_CPUS] = {};
37 module_param_array(cpu_list, int, &cpu_list_len, 0644);
38 MODULE_PARM_DESC(cpu_list, "Comma-delimited list of CPUs that Palacios will run on");
44 static int v3_major_num = 0;
46 static struct v3_guest * guest_map[MAX_VMS] = {[0 ... MAX_VMS - 1] = 0};
47 static struct proc_dir_entry *dir = 0;
49 struct class * v3_class = NULL;
50 static struct cdev ctrl_dev;
52 static int register_vm(struct v3_guest * guest) {
55 for (i = 0; i < MAX_VMS; i++) {
56 if (guest_map[i] == NULL) {
67 static long v3_dev_ioctl(struct file * filp,
68 unsigned int ioctl, unsigned long arg) {
69 void __user * argp = (void __user *)arg;
70 DEBUG("V3 IOCTL %d\n", ioctl);
74 case V3_CREATE_GUEST:{
76 struct v3_guest_img user_image;
77 struct v3_guest * guest = palacios_alloc(sizeof(struct v3_guest));
80 ERROR("Palacios: Error allocating Kernel guest_image\n");
84 memset(guest, 0, sizeof(struct v3_guest));
86 INFO("Palacios: Creating V3 Guest...\n");
88 vm_minor = register_vm(guest);
91 ERROR("Palacios Error: Too many VMs are currently running\n");
95 guest->vm_dev = MKDEV(v3_major_num, vm_minor);
97 if (copy_from_user(&user_image, argp, sizeof(struct v3_guest_img))) {
98 ERROR("Palacios Error: copy from user error getting guest image...\n");
102 guest->img_size = user_image.size;
104 DEBUG("Palacios: Allocating kernel memory for guest image (%llu bytes)\n", user_image.size);
105 guest->img = vmalloc(guest->img_size);
107 if (IS_ERR(guest->img)) {
108 ERROR("Palacios Error: Could not allocate space for guest image\n");
112 if (copy_from_user(guest->img, user_image.guest_data, guest->img_size)) {
113 ERROR("Palacios: Error loading guest data\n");
117 strncpy(guest->name, user_image.name, 127);
119 INIT_LIST_HEAD(&(guest->exts));
121 if (create_palacios_vm(guest) == -1) {
122 ERROR("Palacios: Error creating guest\n");
132 guest_map[vm_minor] = NULL;
134 palacios_free(guest);
140 case V3_FREE_GUEST: {
141 unsigned long vm_idx = arg;
142 struct v3_guest * guest = guest_map[vm_idx];
145 ERROR("No VM at index %ld\n",vm_idx);
149 INFO("Freeing VM (%s) (%p)\n", guest->name, guest);
151 free_palacios_vm(guest);
152 guest_map[vm_idx] = NULL;
155 case V3_ADD_MEMORY: {
156 struct v3_mem_region mem;
158 memset(&mem, 0, sizeof(struct v3_mem_region));
160 if (copy_from_user(&mem, argp, sizeof(struct v3_mem_region))) {
161 ERROR("copy from user error getting mem_region...\n");
165 DEBUG("Adding %llu pages to Palacios memory\n", mem.num_pages);
167 if (add_palacios_memory(mem.base_addr, mem.num_pages) == -1) {
168 ERROR("Error adding memory to Palacios\n");
176 ERROR("\tUnhandled\n");
185 static struct file_operations v3_ctrl_fops = {
186 .owner = THIS_MODULE,
187 .unlocked_ioctl = v3_dev_ioctl,
188 .compat_ioctl = v3_dev_ioctl,
193 struct proc_dir_entry *palacios_get_procdir(void)
198 static int read_guests(char * buf, char ** start, off_t off, int count,
199 int * eof, void * data)
204 for(i = 0; i < MAX_VMS; i++) {
205 if (guest_map[i] != NULL) {
207 len += snprintf(buf+len, count-len,
208 "%s\t/dev/v3-vm%d\n",
209 guest_map[i]->name, i);
217 static int show_mem(char * buf, char ** start, off_t off, int count,
218 int * eof, void * data)
222 len = snprintf(buf,count, "%p\n", (void *)get_palacios_base_addr());
223 len += snprintf(buf+len,count-len, "%lld\n", get_palacios_num_pages());
229 static int __init v3_init(void) {
230 dev_t dev = MKDEV(0, 0); // We dynamicallly assign the major number
236 // Initialize Palacios
240 // initialize extensions
241 init_lnx_extensions();
244 v3_class = class_create(THIS_MODULE, "vms");
245 if (IS_ERR(v3_class)) {
246 ERROR("Failed to register V3 VM device class\n");
247 return PTR_ERR(v3_class);
250 INFO("intializing V3 Control device\n");
252 ret = alloc_chrdev_region(&dev, 0, MAX_VMS + 1, "v3vee");
255 ERROR("Error registering device region for V3 devices\n");
259 v3_major_num = MAJOR(dev);
261 dev = MKDEV(v3_major_num, MAX_VMS + 1);
264 DEBUG("Creating V3 Control device: Major %d, Minor %d\n", v3_major_num, MINOR(dev));
265 cdev_init(&ctrl_dev, &v3_ctrl_fops);
266 ctrl_dev.owner = THIS_MODULE;
267 ctrl_dev.ops = &v3_ctrl_fops;
268 cdev_add(&ctrl_dev, dev, 1);
270 device_create(v3_class, NULL, dev, NULL, "v3vee");
273 ERROR("Error adding v3 control device\n");
277 dir = proc_mkdir("v3vee", NULL);
279 struct proc_dir_entry *entry;
281 entry = create_proc_read_entry("v3-guests", 0444, dir,
284 INFO("/proc/v3vee/v3-guests successfully created\n");
286 ERROR("Could not create proc entry\n");
290 entry = create_proc_read_entry("v3-mem", 0444, dir,
293 INFO("/proc/v3vee/v3-mem successfully added\n");
295 ERROR("Could not create proc entry\n");
299 ERROR("Could not create proc entry\n");
306 unregister_chrdev_region(MKDEV(v3_major_num, 0), MAX_VMS + 1);
308 class_destroy(v3_class);
314 static void __exit v3_exit(void) {
315 extern u32 pg_allocs;
321 // should probably try to stop any guests
325 dev_t dev = MKDEV(v3_major_num, MAX_VMS + 1);
327 INFO("Removing V3 Control device\n");
332 DEBUG("Palacios Mallocs = %d, Frees = %d\n", mallocs, frees);
333 DEBUG("Palacios Page Allocs = %d, Page Frees = %d\n", pg_allocs, pg_frees);
335 unregister_chrdev_region(MKDEV(v3_major_num, 0), MAX_VMS + 1);
339 device_destroy(v3_class, dev);
340 class_destroy(v3_class);
343 deinit_lnx_extensions();
345 palacios_deinit_mm();
347 remove_proc_entry("v3-guests", dir);
348 remove_proc_entry("v3-mem", dir);
349 remove_proc_entry("v3vee", NULL);
351 DEBUG("Palacios Module Mallocs = %d, Frees = %d\n", mod_allocs, mod_frees);
356 module_init(v3_init);
357 module_exit(v3_exit);
361 void * trace_malloc(size_t size, gfp_t flags) {
365 addr = kmalloc(size, flags);
371 void trace_free(const void * objp) {