X-Git-Url: http://v3vee.org/palacios/gitweb/gitweb.cgi?a=blobdiff_plain;f=palacios%2Fsrc%2Fdevices%2F8254.c;h=3472669f0bbefdc468ce624c0815332667d390be;hb=fbe9ba34153fdfd6c67a59b5284b2d23de70e878;hp=5a78e937734cf82c0133f849f833f4197f436c91;hpb=72420d58d18ec71d4777d029daaf0c6a1c820b32;p=palacios.git diff --git a/palacios/src/devices/8254.c b/palacios/src/devices/8254.c index 5a78e93..3472669 100644 --- a/palacios/src/devices/8254.c +++ b/palacios/src/devices/8254.c @@ -89,6 +89,8 @@ struct pit { struct v3_timer * timer; + struct v3_vm_info * vm; + struct channel ch_0; struct channel ch_1; struct channel ch_2; @@ -297,10 +299,13 @@ static void pit_update_timer(struct guest_info * info, ullong_t cpu_cycles, ullo // update counter with remainder (mod reload) state->pit_counter = state->pit_reload - cpu_cycles; - //PrintDebug("8254 PIT: Handling %d crystal tics\n", oscillations); + if (oscillations) { + PrintDebug("8254 PIT: Handling %d crystal tics\n", oscillations); + } + if (handle_crystal_tics(state, &(state->ch_0), oscillations) == 1) { // raise interrupt - PrintDebug("8254 PIT: Injecting Timer interrupt to guest\n"); + // PrintDebug("8254 PIT: Injecting Timer interrupt to guest\n"); v3_raise_irq(info->vm_info, 0); } @@ -642,9 +647,9 @@ static void init_channel(struct channel * ch) { -static int pit_free(struct vm_device * dev) { - struct pit * state = (struct pit *)dev->private_data; - struct guest_info * info = &(dev->vm->cores[0]); +static int pit_free(void * private_data) { + struct pit * state = (struct pit *)private_data; + struct guest_info * info = &(state->vm->cores[0]); if (state->timer) { @@ -655,11 +660,63 @@ static int pit_free(struct vm_device * dev) { return 0; } +static int pit_checkpoint(struct vm_device *dev, v3_keyed_stream_t stream) +{ + struct pit *p = (struct pit *) (dev->private_data); -static struct v3_device_ops dev_ops = { - .free = pit_free, + v3_keyed_stream_key_t ks; + + ks = v3_keyed_stream_open_key(stream,dev->name); + + if (!ks) { + return -1; + } + + STD_SAVE(stream,ks,p->pit_counter); + STD_SAVE(stream,ks,p->pit_reload); + STD_SAVE(stream,ks,p->ch_0); + STD_SAVE(stream,ks,p->ch_1); + STD_SAVE(stream,ks,p->ch_2); + STD_SAVE(stream,ks,p->speaker); + + v3_keyed_stream_close_key(stream,ks); + + return 0; + + +} +static int pit_restore(struct vm_device *dev, v3_keyed_stream_t stream) +{ + struct pit *p = (struct pit *) (dev->private_data); + v3_keyed_stream_key_t ks; + + ks = v3_keyed_stream_open_key(stream,dev->name); + + if (!ks) { + return -1; + } + + STD_LOAD(stream,ks,p->pit_counter); + STD_LOAD(stream,ks,p->pit_reload); + STD_LOAD(stream,ks,p->ch_0); + STD_LOAD(stream,ks,p->ch_1); + STD_LOAD(stream,ks,p->ch_2); + STD_LOAD(stream,ks,p->speaker); + + v3_keyed_stream_close_key(stream,ks); + + return 0; + + +} + + +static struct v3_device_ops dev_ops = { + .free = (int (*)(void *))pit_free, + .checkpoint = pit_checkpoint, + .restore = pit_restore, }; #include @@ -674,12 +731,14 @@ static int pit_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) { // just hardcode the core context struct guest_info * info = &(vm->cores[0]); - uint_t cpu_khz = V3_CPU_KHZ(); + uint_t cpu_khz = info->time_state.guest_cpu_freq; ullong_t reload_val = (ullong_t)cpu_khz * 1000; pit_state = (struct pit *)V3_Malloc(sizeof(struct pit)); + V3_ASSERT(pit_state != NULL); pit_state->speaker = 0; + pit_state->vm = vm; dev = v3_add_device(vm, dev_id, &dev_ops, pit_state);