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 to VMM/Guest model dynamic switch for Virtio NIC
[palacios.git] / palacios / src / devices / nvram.c
index 943cbf3..f2c45a3 100644 (file)
@@ -29,6 +29,7 @@
 #include <palacios/vmm_host_events.h>
 #include <palacios/vm_guest.h>
 
+
 #ifndef CONFIG_DEBUG_NVRAM
 #undef PrintDebug
 #define PrintDebug(fmt, args...)
@@ -107,11 +108,13 @@ struct nvram_internal {
     struct vm_device * ide;
 
     struct v3_vm_info * vm;
+    
+    struct v3_timer   *timer;
 
     v3_lock_t nvram_lock;
 
-    uint32_t        us;   //microseconds - for clock update - zeroed every second
-    uint32_t        pus;  //microseconds - for periodic interrupt - cleared every period
+    uint64_t        us;   //microseconds - for clock update - zeroed every second
+    uint64_t        pus;  //microseconds - for periodic interrupt - cleared every period
 };
 
 
@@ -275,7 +278,7 @@ static uint8_t days_in_month(uint8_t month, uint8_t bcd) {
 }
 
 
-static void update_time(struct nvram_internal * data, uint32_t period_us) {
+static void update_time(struct nvram_internal * data, uint64_t period_us) {
     struct rtc_stata * stata = (struct rtc_stata *)&((data->mem_state[NVRAM_REG_STAT_A]));
     struct rtc_statb * statb = (struct rtc_statb *)&((data->mem_state[NVRAM_REG_STAT_B]));
     struct rtc_statc * statc = (struct rtc_statc *)&((data->mem_state[NVRAM_REG_STAT_C]));
@@ -299,10 +302,7 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
     uint8_t nextday = 0;
     uint32_t  periodic_period;
 
-    //PrintDebug("nvram: sizeof(struct rtc_stata)=%d\n", sizeof(struct rtc_stata));
-
-
-    //PrintDebug("nvram: update_time\n",statb->pi);
+    PrintDebug("nvram: update_time by %llu microseocnds\n",period_us);
   
     // We will set these flags on exit
     statc->irq = 0;
@@ -320,7 +320,7 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
        carry = add_to(sec, &carry, bcd);
 
        if (carry) { 
-           PrintDebug("nvram: somehow managed to get a carry in second update\n"); 
+           PrintError("nvram: somehow managed to get a carry in second update\n"); 
        }
 
        if ( (bcd && (*sec == 0x60)) || 
@@ -331,7 +331,7 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
            carry = 1;
            carry = add_to(min, &carry, bcd);
            if (carry) { 
-               PrintDebug("nvram: somehow managed to get a carry in minute update\n"); 
+               PrintError("nvram: somehow managed to get a carry in minute update\n"); 
            }
 
            if ( (bcd && (*min == 0x60)) || 
@@ -352,7 +352,7 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
                carry = 1;
                carry = add_to(&hour24, &carry, bcd);
                if (carry) { 
-                   PrintDebug("nvram: somehow managed to get a carry in hour update\n"); 
+                   PrintError("nvram: somehow managed to get a carry in hour update\n"); 
                }
 
                if ( (bcd && (hour24 == 0x24)) || 
@@ -453,7 +453,7 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
 
     statc->irq = (statc->pf || statc->af || statc->uf);
   
-    //PrintDebug("nvram: time is now: YMDHMS: 0x%x:0x%x:0x%x:0x%x:0x%x,0x%x bcd=%d\n", *year, *month, *monthday, *hour, *min, *sec,bcd);
+    PrintDebug("nvram: time is now: YMDHMS: 0x%x:0x%x:0x%x:0x%x:0x%x,0x%x bcd=%d\n", *year, *month, *monthday, *hour, *min, *sec,bcd);
   
     // Interrupt associated VM, if needed
     if (statc->irq) { 
@@ -463,24 +463,22 @@ static void update_time(struct nvram_internal * data, uint32_t period_us) {
 }
 
 
-static int handle_timer_event(struct v3_vm_info * vm, 
-                             struct v3_timer_event * evt, 
-                             void * priv_data) {
+static void nvram_update_timer(struct guest_info *vm,
+                              ullong_t           cpu_cycles,
+                              ullong_t           cpu_freq,
+                              void              *priv_data)
+{
+    struct nvram_internal *nvram_state = (struct nvram_internal *)priv_data;
+    uint64_t period_us;
 
+    
+    period_us = (1000000*cpu_cycles/cpu_freq);
 
-    struct nvram_internal * data = priv_data;
+    update_time(nvram_state,period_us);
 
-    if (data) {
-       addr_t irq_state = v3_lock_irqsave(data->nvram_lock);
-       update_time(data, evt->period_us);
-       v3_unlock_irqrestore(data->nvram_lock, irq_state);
-    }
-  
-    return 0;
 }
 
 
-
 static void set_memory_size(struct nvram_internal * nvram, addr_t bytes) {
     // 1. Conventional Mem: 0-640k in K
     // 2. Extended Mem: 0-16MB in K
@@ -720,7 +718,7 @@ static int nvram_write_reg_port(struct guest_info * core, uint16_t port,
     struct nvram_internal * data = priv_data;
     
     memcpy(&(data->thereg), src, 1);
-    PrintDebug("Writing To NVRAM reg: 0x%x\n", data->thereg);
+    PrintDebug("nvram: Writing To NVRAM reg: 0x%x\n", data->thereg);
 
     return 1;
 }
@@ -733,7 +731,7 @@ static int nvram_read_data_port(struct guest_info * core, uint16_t port,
     addr_t irq_state = v3_lock_irqsave(data->nvram_lock);
 
     if (get_memory(data, data->thereg, (uint8_t *)dst) == -1) {
-       PrintError("Register %d (0x%x) Not set\n", data->thereg, data->thereg);
+       PrintError("nvram: Register %d (0x%x) Not set\n", data->thereg, data->thereg);
 
        v3_unlock_irqrestore(data->nvram_lock, irq_state);
 
@@ -745,7 +743,7 @@ static int nvram_read_data_port(struct guest_info * core, uint16_t port,
        }
     }
 
-    PrintDebug("nvram_read_data_port(0x%x)  =  0x%x\n", data->thereg, *(uint8_t *)dst);
+    PrintDebug("nvram: nvram_read_data_port(0x%x)  =  0x%x\n", data->thereg, *(uint8_t *)dst);
 
     // hack
     if (data->thereg == NVRAM_REG_STAT_A) { 
@@ -769,7 +767,7 @@ static int nvram_write_data_port(struct guest_info * core, uint16_t port,
 
     v3_unlock_irqrestore(data->nvram_lock, irq_state);
 
-    PrintDebug("nvram_write_data_port(0x%x) = 0x%x\n", 
+    PrintDebug("nvram: nvram_write_data_port(0x%x) = 0x%x\n", 
               data->thereg, data->mem_state[data->thereg]);
 
     return 1;
@@ -779,8 +777,13 @@ static int nvram_write_data_port(struct guest_info * core, uint16_t port,
 
 
 static int nvram_free(struct nvram_internal * nvram_state) {
-
+    
     // unregister host events
+    struct guest_info *info = &(nvram_state->vm->cores[0]);
+
+    if (nvram_state->timer) { 
+       v3_remove_timer(info,nvram_state->timer);
+    }
 
     V3_Free(nvram_state);
     return 0;
@@ -788,6 +791,9 @@ static int nvram_free(struct nvram_internal * nvram_state) {
 
 
 
+static struct v3_timer_ops timer_ops = {
+    .update_timer = nvram_update_timer,
+};
 
 
 static struct v3_device_ops dev_ops = {  
@@ -805,7 +811,7 @@ static int nvram_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) {
     int ret = 0;
 
     if (!ide) {
-       PrintError("Could not find IDE device\n");
+       PrintError("nvram: Could not find IDE device\n");
        return -1;
     }
 
@@ -820,7 +826,7 @@ static int nvram_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) {
     struct vm_device * dev = v3_add_device(vm, dev_id, &dev_ops, nvram_state);
 
     if (dev == NULL) {
-       PrintError("Could not attach device %s\n", dev_id);
+       PrintError("nvram: Could not attach device %s\n", dev_id);
        V3_Free(nvram_state);
        return -1;
     }
@@ -832,12 +838,17 @@ static int nvram_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) {
     ret |= v3_dev_hook_io(dev, NVRAM_DATA_PORT, &nvram_read_data_port, &nvram_write_data_port);
   
     if (ret != 0) {
-       PrintError("Error hooking NVRAM IO ports\n");
+       PrintError("nvram: Error hooking NVRAM IO ports\n");
        v3_remove_device(dev);
        return -1;
     }
 
-    v3_hook_host_event(vm, HOST_TIMER_EVT, V3_HOST_EVENT_HANDLER(handle_timer_event), nvram_state);
+    nvram_state->timer = v3_add_timer(&(vm->cores[0]),&timer_ops,nvram_state);
+
+    if (nvram_state->timer == NULL ) { 
+       v3_remove_device(dev);
+       return -1;
+    }
 
     return 0;
 }