X-Git-Url: http://v3vee.org/palacios/gitweb/gitweb.cgi?a=blobdiff_plain;f=palacios%2Fsrc%2Fpalacios%2Fsvm_halt.c;h=fea77ba87381e142d1a8f3be682b4862cfa7abb8;hb=55ced750335769366957df3c0cc3fbc8b82a3ea9;hp=3c893d3d848149b82144e53165e11d21d8bb426a;hpb=fa7a42a55338e5fa49ec760b367225ca752df4fd;p=palacios.git diff --git a/palacios/src/palacios/svm_halt.c b/palacios/src/palacios/svm_halt.c index 3c893d3..fea77ba 100644 --- a/palacios/src/palacios/svm_halt.c +++ b/palacios/src/palacios/svm_halt.c @@ -1,36 +1,64 @@ +/* + * This file is part of the Palacios Virtual Machine Monitor developed + * by the V3VEE Project with funding from the United States National + * Science Foundation and the Department of Energy. + * + * The V3VEE Project is a joint project between Northwestern University + * and the University of New Mexico. You can find out more at + * http://www.v3vee.org + * + * Copyright (c) 2008, Peter Dinda + * Copyright (c) 2008, Jack Lange + * Copyright (c) 2008, The V3VEE Project + * All rights reserved. + * + * Author: Peter Dinda + * + * This is free software. You are permitted to use, + * redistribute, and modify it as specified in the file "V3VEE_LICENSE". + */ + #include #include -// From GeekOS -void Yield(void); - - -int handle_svm_halt(struct guest_info * info) -{ - // What we should do is starting waiting on an OS event that will - // result in an injection of an interrupt. - - // What we will hackishly do instead is resume on any event - // Plus is this totally GeekOS specific - ullong_t yield_start = 0; - ullong_t yield_stop = 0; - uint32_t gap = 0; - PrintDebug("GeekOS Yield\n"); +#ifndef DEBUG_HALT +#undef PrintDebug +#define PrintDebug(fmt, args...) +#endif - rdtscll(yield_start); - Yield(); - rdtscll(yield_stop); - //v3_update_time(info, yield_stop - yield_start); - gap = yield_stop - yield_start; - v3_raise_irq(info, 0); - - PrintDebug("GeekOS Yield Done (%d cycles)\n", gap); - - info->rip+=1; - - return 0; +// +// This should trigger a #GP if cpl!=0, otherwise, yield to host +// +int v3_handle_svm_halt(struct guest_info * info) +{ + if (info->cpl!=0) { + v3_raise_exception(info, GPF_EXCEPTION); + } else { + + ullong_t yield_start = 0; + ullong_t yield_stop = 0; + uint32_t gap = 0; + + PrintDebug("CPU Yield\n"); + + rdtscll(yield_start); + V3_Yield(); + rdtscll(yield_stop); + + + //v3_update_time(info, yield_stop - yield_start); + gap = yield_stop - yield_start; + if (!v3_intr_pending(info)) { + v3_raise_irq(info, 0); + } + + PrintDebug("CPU Yield Done (%d cycles)\n", gap); + + info->rip+=1; + } + return 0; }