Palacios Public Git Repository

To checkout Palacios execute

  git clone http://v3vee.org/palacios/palacios.web/palacios.git
This will give you the master branch. You probably want the devel branch or one of the release branches. To switch to the devel branch, simply execute
  cd palacios
  git checkout --track -b devel origin/devel
The other branches are similar.


Fix for option processing
[palacios.git] / palacios / src / devices / filedisk.c
1 /* 
2  * This file is part of the Palacios Virtual Machine Monitor developed
3  * by the V3VEE Project with funding from the United States National 
4  * Science Foundation and the Department of Energy.  
5  *
6  * The V3VEE Project is a joint project between Northwestern University
7  * and the University of New Mexico.  You can find out more at 
8  * http://www.v3vee.org
9  *
10  * Copyright (c) 2008, Jack Lange <jarusl@cs.northwestern.edu> 
11  * Copyright (c) 2008, The V3VEE Project <http://www.v3vee.org> 
12  * All rights reserved.
13  *
14  * Author: Jack Lange <jarusl@cs.northwestern.edu>
15  *
16  * This is free software.  You are permitted to use,
17  * redistribute, and modify it as specified in the file "V3VEE_LICENSE".
18  */
19
20 #include <palacios/vmm.h>
21 #include <palacios/vmm_dev_mgr.h>
22
23 #include <interfaces/vmm_file.h>
24 #include <palacios/vm_guest.h>
25
26 #ifndef V3_CONFIG_DEBUG_FILEDISK
27 #undef PrintDebug
28 #define PrintDebug(fmt, args...)
29 #endif
30
31 struct disk_state {
32     uint64_t capacity; // in bytes
33
34     v3_file_t fd;
35 };
36
37
38
39 static int write_all(v3_file_t fd, char * buf, uint64_t offset, uint64_t length) {
40     uint64_t bytes_written = 0;
41     
42     PrintDebug(VM_NONE, VCORE_NONE, "Writing %llu bytes\n", length - bytes_written);
43     while (bytes_written < length) {
44         int tmp_bytes = v3_file_write(fd, buf + bytes_written, length - bytes_written, offset + bytes_written);
45         PrintDebug(VM_NONE, VCORE_NONE, "Wrote %d bytes\n", tmp_bytes);
46         
47         if (tmp_bytes <= 0 ) {
48             PrintError(VM_NONE, VCORE_NONE, "Write failed\n");
49             return -1;
50         }
51         
52         bytes_written += tmp_bytes;
53     }
54     
55     return 0;
56 }
57
58
59 static int read_all(v3_file_t fd, char * buf, uint64_t offset, uint64_t length) {
60     uint64_t bytes_read = 0;
61     
62     PrintDebug(VM_NONE, VCORE_NONE, "Reading %llu bytes\n", length - bytes_read);
63     while (bytes_read < length) {
64         int tmp_bytes = v3_file_read(fd, buf + bytes_read, length - bytes_read, offset + bytes_read);
65         PrintDebug(VM_NONE, VCORE_NONE, "Read %d bytes\n", tmp_bytes);
66         
67         if (tmp_bytes <= 0) {
68             PrintError(VM_NONE, VCORE_NONE, "Read failed\n");
69             return -1;
70         }
71         
72         bytes_read += tmp_bytes;
73     }
74     
75     return 0;
76 }
77
78 static int read(uint8_t * buf, uint64_t lba, uint64_t num_bytes, void * private_data) {
79     struct disk_state * disk = (struct disk_state *)private_data;
80
81     PrintDebug(VM_NONE, VCORE_NONE, "Reading %llu bytes from %llu to 0x%p\n", num_bytes, lba, buf);
82
83     if (lba + num_bytes > disk->capacity) {
84         PrintError(VM_NONE, VCORE_NONE, "Out of bounds read: lba=%llu, num_bytes=%llu, capacity=%llu\n",
85                    lba, num_bytes, disk->capacity);
86         return -1;
87     }
88
89     return read_all(disk->fd, buf, lba, num_bytes);
90 }
91
92
93 static int write(uint8_t * buf, uint64_t lba, uint64_t num_bytes, void * private_data) {
94     struct disk_state * disk = (struct disk_state *)private_data;
95
96     PrintDebug(VM_NONE, VCORE_NONE, "Writing %llu bytes from 0x%p to %llu\n", num_bytes,  buf, lba);
97
98     if (lba + num_bytes > disk->capacity) {
99         PrintError(VM_NONE, VCORE_NONE, "Out of bounds read: lba=%llu, num_bytes=%llu, capacity=%llu\n",
100                    lba, num_bytes, disk->capacity);
101         return -1;
102     }
103
104
105     return write_all(disk->fd,  buf, lba, num_bytes);
106 }
107
108
109 static uint64_t get_capacity(void * private_data) {
110     struct disk_state * disk = (struct disk_state *)private_data;
111
112     PrintDebug(VM_NONE, VCORE_NONE, "Querying FILEDISK capacity %llu\n", disk->capacity);
113
114     return disk->capacity;
115 }
116
117 static struct v3_dev_blk_ops blk_ops = {
118     .read = read, 
119     .write = write,
120     .get_capacity = get_capacity,
121 };
122
123
124
125
126 static int disk_free(struct disk_state * disk) {
127     v3_file_close(disk->fd);
128     
129     V3_Free(disk);
130     return 0;
131 }
132
133 static struct v3_device_ops dev_ops = {
134     .free = (int (*)(void *))disk_free,
135 };
136
137
138
139
140 static int disk_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) {
141     struct disk_state * disk = NULL;
142     char * path = v3_cfg_val(cfg, "path");
143     char * dev_id = v3_cfg_val(cfg, "ID");
144     char * writable = v3_cfg_val(cfg, "writable");
145     char * writeable = v3_cfg_val(cfg, "writeable");
146
147     v3_cfg_tree_t * frontend_cfg = v3_cfg_subtree(cfg, "frontend");
148     int flags = FILE_OPEN_MODE_READ;
149
150     if ( ((writable) && (writable[0] == '1')) ||
151          ((writeable) && (writeable[0] == '1')) ) {
152         flags |= FILE_OPEN_MODE_WRITE;
153     }
154
155     if (path == NULL) {
156         PrintError(vm, VCORE_NONE, "Missing path (%s) for %s\n", path, dev_id);
157         return -1;
158     }
159
160     disk = (struct disk_state *)V3_Malloc(sizeof(struct disk_state));
161
162     if (disk == NULL) {
163         PrintError(vm, VCORE_NONE, "Could not allocate disk\n");
164         return -1;
165     }
166
167     memset(disk, 0, sizeof(struct disk_state));
168
169     struct vm_device * dev = v3_add_device(vm, dev_id, &dev_ops, disk);
170
171     if (dev == NULL) {
172         PrintError(vm, VCORE_NONE, "Could not attach device %s\n", dev_id);
173         V3_Free(disk);
174         return -1;
175     }
176
177
178     disk->fd = v3_file_open(vm, path, flags);
179
180     if (disk->fd == NULL) {
181         PrintError(vm, VCORE_NONE, "Could not open file disk:%s\n", path);
182         v3_remove_device(dev);
183         return -1;
184     }
185
186     disk->capacity = v3_file_size(disk->fd);
187
188     V3_Print(vm, VCORE_NONE, "Registering FILEDISK %s (path=%s, fd=%lu, size=%llu, writeable=%d)\n",
189              dev_id, path, (addr_t)disk->fd, disk->capacity,
190              flags & FILE_OPEN_MODE_WRITE);
191
192
193     if (v3_dev_connect_blk(vm, v3_cfg_val(frontend_cfg, "tag"), 
194                            &blk_ops, frontend_cfg, disk) == -1) {
195         PrintError(vm, VCORE_NONE, "Could not connect %s to frontend %s\n", 
196                    dev_id, v3_cfg_val(frontend_cfg, "tag"));
197         v3_remove_device(dev);
198         return -1;
199     }
200     
201
202     return 0;
203 }
204
205
206 device_register("FILEDISK", disk_init)