return table;
}
-uint16_t get_cpu_mhz(void)
+struct shared_info *get_shared_info(void)
{
+ static struct shared_info *shared_info = NULL;
struct xen_add_to_physmap xatp;
- struct shared_info *shared_info = (struct shared_info *)0xfffff000;
+
+ if ( shared_info != NULL )
+ return shared_info;
+
+ xatp.domid = DOMID_SELF;
+ xatp.space = XENMAPSPACE_shared_info;
+ xatp.idx = 0;
+ xatp.gpfn = 0xfffffu;
+ shared_info = (struct shared_info *)(xatp.gpfn << PAGE_SHIFT);
+ if ( hypercall_memory_op(XENMEM_add_to_physmap, &xatp) != 0 )
+ BUG();
+
+ return shared_info;
+}
+
+uint16_t get_cpu_mhz(void)
+{
+ struct shared_info *shared_info = get_shared_info();
struct vcpu_time_info *info = &shared_info->vcpu_info[0].time;
uint64_t cpu_khz;
uint32_t tsc_to_nsec_mul, version;
if ( cpu_mhz != 0 )
return cpu_mhz;
- /* Map shared-info page. */
- xatp.domid = DOMID_SELF;
- xatp.space = XENMAPSPACE_shared_info;
- xatp.idx = 0;
- xatp.gpfn = (unsigned long)shared_info >> 12;
- if ( hypercall_memory_op(XENMEM_add_to_physmap, &xatp) != 0 )
- BUG();
-
/* Get a consistent snapshot of scale factor (multiplier and shift). */
do {
version = info->version;
#define pci_writew(devfn, reg, val) (pci_write(devfn, reg, 2, (uint16_t)val))
#define pci_writel(devfn, reg, val) (pci_write(devfn, reg, 4, (uint32_t)val))
+/* Get a pointer to the shared-info page */
+struct shared_info *get_shared_info(void);
+
/* Get CPU speed in MHz. */
uint16_t get_cpu_mhz(void);
(unsigned long) rings, (unsigned long) event);
}
-/* Reset the xenbus connection so the next kernel can start again.
- * We zero out the whole ring -- the backend can handle this, and it's
- * not going to surprise any frontends since it's equivalent to never
- * having used the rings. */
+/* Reset the xenbus connection so the next kernel can start again. */
void xenbus_shutdown(void)
{
ASSERT(rings != NULL);
+
+ /* We zero out the whole ring -- the backend can handle this, and it's
+ * not going to surprise any frontends since it's equivalent to never
+ * having used the rings. */
memset(rings, 0, sizeof *rings);
+
+ /* Clear the xenbus event-channel too */
+ get_shared_info()->evtchn_pending[event / sizeof (unsigned long)]
+ &= ~(1UL << ((event % sizeof (unsigned long))));
+
rings = NULL;
}