xen-devel.lists.xenproject.org archive mirror
 help / color / mirror / Atom feed
From: Ian Campbell <ian.campbell@citrix.com>
To: ian.jackson@eu.citrix.com, wei.liu2@citrix.com, xen-devel@lists.xen.org
Cc: Ian Campbell <ian.campbell@citrix.com>, stefano.stabellini@eu.citrix.com
Subject: [PATCH QEMU-XEN v3 5/8] xen: Switch uses of xc_map_foreign_bulk to use libxenforeignmemory API.
Date: Wed, 7 Oct 2015 15:16:03 +0100	[thread overview]
Message-ID: <1444227366-1015-5-git-send-email-ian.campbell@citrix.com> (raw)
In-Reply-To: <1444226543.1410.53.camel@citrix.com>

In Xen 4.7 we are refactoring parts libxenctrl into a number of
separate libraries which will provide backward and forward API and ABI
compatiblity.

One such library will be libxenforeignmemory which provides access to
privileged foreign mappings and which will provide an interface
equivalent to xc_map_foreign_bulk.

In preparation for adding support for libxenforeignmemory add support
to the <=4.0 and <=4.6 compat code in xen_common.h to allow us to
switch to using the new API. These shims will disappear for versions
of Xen which include libxenforeignmemory.

Since libxenforeignmemory will have its own handle type but for <= 4.6
the functionality is provided by using a libxenctrl handle we
introduce a new global xen_fmem alongside the existing xen_xc. In fact
we make xen_fmem a pointer to the existing xen_xc, which then works
correctly with both <=4.0 (xc handle is an int) and <=4.6 (xc handle
is a pointer). In the latter case xen_fmem is actually a double
indirect pointer, but it all falls out in the wash.

Unlike libxenctrl libxenforeignmemory has an explicit unmap function,
rather than just specifying that munmap should be used, so the unmap
paths are updated to use xenforeignmemory_unmap, which is a shim for
munmap on these versions of xen. The mappings in xen-hvm.c do not
appear to be unmapped (which makes sense for a qemu-dm process)

In fb_disconnect this results in a change from simply mmap over the
existing mapping (with an implciit munmap) to expliclty unmapping with
xenforeignmemory_unmap and then mapping the required anonymous memory
in the same hole. I don't think this is a problem since any other
thread which was racily touching this region would already be running
the risk of hitting the mapping halfway through the call. If this is
thought to be a problem then we could consider adding an extra API to
the libxenforeignmemory interface to replace a foreign mapping with
anonymous shared memory, but I'd prefer not to.

Build tested with 4.0 and 4.5.

Signed-off-by: Ian Campbell <ian.campbell@citrix.com>
---
I noticed in xen_console.c that the decision to use a foreign
privileged memory mapping vs a grant dev is made using different
variables in con_initialise vs con_disconnect. The former uses
xendev->dev while the latter uses xendev->gnttabdev. Is this a latent
bug?
---
 hw/char/xen_console.c        |  8 ++++----
 hw/display/xenfb.c           | 15 ++++++++-------
 hw/xen/xen_backend.c         |  3 ++-
 include/hw/xen/xen_backend.h |  1 +
 include/hw/xen/xen_common.h  |  8 ++++++++
 xen-common.c                 |  1 +
 xen-hvm.c                    | 18 +++++++++---------
 xen-mapcache.c               |  6 +++---
 8 files changed, 36 insertions(+), 24 deletions(-)

diff --git a/hw/char/xen_console.c b/hw/char/xen_console.c
index 6ef477a..d28a5da 100644
--- a/hw/char/xen_console.c
+++ b/hw/char/xen_console.c
@@ -230,9 +230,9 @@ static int con_initialise(struct XenDevice *xendev)
 
     if (!xendev->dev) {
 	xen_pfn_t mfn = con->ring_ref;
-        con->sring = xc_map_foreign_bulk(xen_xc, con->xendev.dom,
-					 PROT_READ|PROT_WRITE,
-					 &mfn, &err, 1);
+        con->sring = xenforeignmemory_map(xen_fmem, con->xendev.dom,
+					  PROT_READ|PROT_WRITE,
+					  &mfn, &err, 1);
     } else {
         con->sring = xengnttab_map_grant_ref(xendev->gnttabdev, con->xendev.dom,
                                              con->ring_ref,
@@ -277,7 +277,7 @@ static void con_disconnect(struct XenDevice *xendev)
 
     if (con->sring) {
         if (!xendev->gnttabdev) {
-            munmap(con->sring, XC_PAGE_SIZE);
+            xenforeignmemory_unmap(xen_fmem, con->sring, XC_PAGE_SIZE);
         } else {
             xengnttab_munmap(xendev->gnttabdev, con->sring, 1);
         }
diff --git a/hw/display/xenfb.c b/hw/display/xenfb.c
index f9442b7..ac4ace4 100644
--- a/hw/display/xenfb.c
+++ b/hw/display/xenfb.c
@@ -103,8 +103,8 @@ static int common_bind(struct common *c)
     if (xenstore_read_fe_int(&c->xendev, "event-channel", &c->xendev.remote_port) == -1)
 	return -1;
 
-    c->page = xc_map_foreign_bulk(xen_xc, c->xendev.dom,
-				  PROT_READ | PROT_WRITE, &mfn, &err, 1);
+    c->page = xenforeignmemory_map(xen_fmem, c->xendev.dom,
+				   PROT_READ | PROT_WRITE, &mfn, &err, 1);
     if (c->page == NULL)
 	return -1;
 
@@ -119,7 +119,7 @@ static void common_unbind(struct common *c)
 {
     xen_be_unbind_evtchn(&c->xendev);
     if (c->page) {
-	munmap(c->page, XC_PAGE_SIZE);
+        xenforeignmemory_unmap(xen_fmem, c->page, XC_PAGE_SIZE);
 	c->page = NULL;
     }
 }
@@ -491,14 +491,14 @@ static int xenfb_map_fb(struct XenFB *xenfb)
     errs = g_malloc0(sizeof(int) * n_fbdirs);
 
     xenfb_copy_mfns(mode, n_fbdirs, pgmfns, pd);
-    map = xc_map_foreign_bulk(xen_xc, xenfb->c.xendev.dom,
-			      PROT_READ, pgmfns, errs, n_fbdirs);
+    map = xenforeignmemory_map(xen_fmem, xenfb->c.xendev.dom,
+			       PROT_READ, pgmfns, errs, n_fbdirs);
     if (map == NULL)
 	goto out;
     xenfb_copy_mfns(mode, xenfb->fbpages, fbmfns, map);
-    munmap(map, n_fbdirs * XC_PAGE_SIZE);
+    xenforeignmemory_unmap(xen_fmem, map, n_fbdirs * XC_PAGE_SIZE);
 
-    xenfb->pixels = xc_map_foreign_bulk(xen_xc, xenfb->c.xendev.dom,
+    xenfb->pixels = xenforeignmemory_map(xen_fmem, xenfb->c.xendev.dom,
             PROT_READ, fbmfns, errs, xenfb->fbpages);
     if (xenfb->pixels == NULL)
 	goto out;
@@ -907,6 +907,7 @@ static void fb_disconnect(struct XenDevice *xendev)
      *   Replacing the framebuffer with anonymous shared memory
      *   instead.  This releases the guest pages and keeps qemu happy.
      */
+    xenforeignmemory_unmap(xen_fmem, fb->pixels, fb->fbpages * XC_PAGE_SIZE);
     fb->pixels = mmap(fb->pixels, fb->fbpages * XC_PAGE_SIZE,
                       PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANON,
                       -1, 0);
diff --git a/hw/xen/xen_backend.c b/hw/xen/xen_backend.c
index 9e616c6..8b16617 100644
--- a/hw/xen/xen_backend.c
+++ b/hw/xen/xen_backend.c
@@ -45,6 +45,7 @@
 
 /* public */
 XenXC xen_xc = XC_HANDLER_INITIAL_VALUE;
+xenforeignmemory_handle *xen_fmem = NULL;
 struct xs_handle *xenstore = NULL;
 const char *xen_protocol;
 
@@ -718,7 +719,7 @@ int xen_be_init(void)
         goto err;
     }
 
-    if (xen_xc == XC_HANDLER_INITIAL_VALUE) {
+    if (xen_xc == XC_HANDLER_INITIAL_VALUE || xen_fmem == NULL) {
         /* Check if xen_init() have been called */
         goto err;
     }
diff --git a/include/hw/xen/xen_backend.h b/include/hw/xen/xen_backend.h
index 8e8857b..e0d52ee 100644
--- a/include/hw/xen/xen_backend.h
+++ b/include/hw/xen/xen_backend.h
@@ -57,6 +57,7 @@ struct XenDevice {
 
 /* variables */
 extern XenXC xen_xc;
+extern xenforeignmemory_handle *xen_fmem;
 extern struct xs_handle *xenstore;
 extern const char *xen_protocol;
 
diff --git a/include/hw/xen/xen_common.h b/include/hw/xen/xen_common.h
index 3955d86..28f3fb1 100644
--- a/include/hw/xen/xen_common.h
+++ b/include/hw/xen/xen_common.h
@@ -41,6 +41,7 @@ static inline void *xc_map_foreign_bulk(int xc_handle, uint32_t dom, int prot,
 typedef int XenXC;
 typedef int xenevtchn_handle;
 typedef int xengnttab_handle;
+typedef int xenforeignmemory_handle;
 
 #  define XC_INTERFACE_FMT "%i"
 #  define XC_HANDLER_INITIAL_VALUE    -1
@@ -101,6 +102,9 @@ static inline XenXC xen_xc_interface_open(void *logger, void *dombuild_logger,
     return xc_interface_open();
 }
 
+#define xenforeignmemory_map(h,d,p,a,e,n) xc_map_foreign_bulk(*h,d,p,a,e,n)
+#define xenforeignmemory_unmap(h,p,s) munmap(p, s)
+
 static inline int xc_fd(int xen_xc)
 {
     return xen_xc;
@@ -146,6 +150,7 @@ static inline void xs_close(struct xs_handle *xsh)
 #else
 
 typedef xc_interface *XenXC;
+typedef xc_interface *xenforeignmemory_handle;
 typedef xc_evtchn xenevtchn_handle;
 typedef xc_gnttab xengnttab_handle;
 
@@ -174,6 +179,9 @@ static inline XenXC xen_xc_interface_open(void *logger, void *dombuild_logger,
     return xc_interface_open(logger, dombuild_logger, open_flags);
 }
 
+#define xenforeignmemory_map(h,d,p,a,e,n) xc_map_foreign_bulk(*h,d,p,a,e,n)
+#define xenforeignmemory_unmap(h,p,s) munmap(p, s)
+
 /* FIXME There is now way to have the xen fd */
 static inline int xc_fd(xc_interface *xen_xc)
 {
diff --git a/xen-common.c b/xen-common.c
index 56359ca..d319dcb 100644
--- a/xen-common.c
+++ b/xen-common.c
@@ -117,6 +117,7 @@ static int xen_init(MachineState *ms)
         xen_be_printf(NULL, 0, "can't open xen interface\n");
         return -1;
     }
+    xen_fmem = &xen_xc;
     qemu_add_vm_change_state_handler(xen_change_state_handler, NULL);
 
     return 0;
diff --git a/xen-hvm.c b/xen-hvm.c
index 3a56099..9603047 100644
--- a/xen-hvm.c
+++ b/xen-hvm.c
@@ -1213,9 +1213,9 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
     DPRINTF("buffered io page at pfn %lx\n", bufioreq_pfn);
     DPRINTF("buffered io evtchn is %x\n", bufioreq_evtchn);
 
-    state->shared_page = xc_map_foreign_bulk(xen_xc, xen_domid,
-					     PROT_READ|PROT_WRITE,
-					     &ioreq_pfn, &map_err, 1);
+    state->shared_page = xenforeignmemory_map(xen_fmem, xen_domid,
+					      PROT_READ|PROT_WRITE,
+					      &ioreq_pfn, &map_err, 1);
     if (state->shared_page == NULL) {
         hw_error("map shared IO page returned error %d (%d) handle=" XC_INTERFACE_FMT,
                  errno, map_err, xen_xc);
@@ -1225,8 +1225,8 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
     if (!rc) {
         DPRINTF("shared vmport page at pfn %lx\n", ioreq_pfn);
         state->shared_vmport_page =
-		xc_map_foreign_bulk(xen_xc, xen_domid, PROT_READ|PROT_WRITE,
-				    &ioreq_pfn, &map_err, 1);
+		xenforeignmemory_map(xen_fmem, xen_domid, PROT_READ|PROT_WRITE,
+				     &ioreq_pfn, &map_err, 1);
         if (state->shared_vmport_page == NULL) {
             hw_error("map shared vmport IO page returned error %d (%d) handle="
                      XC_INTERFACE_FMT, errno, map_err, xen_xc);
@@ -1235,10 +1235,10 @@ int xen_hvm_init(ram_addr_t *below_4g_mem_size, ram_addr_t *above_4g_mem_size,
         hw_error("get vmport regs pfn returned error %d, rc=%d", errno, rc);
     }
 
-    state->buffered_io_page = xc_map_foreign_bulk(xen_xc, xen_domid,
-						  PROT_READ|PROT_WRITE,
-                                                  &bufioreq_pfn,
-						  &map_err, 1);
+    state->buffered_io_page = xenforeignmemory_map(xen_fmem, xen_domid,
+						   PROT_READ|PROT_WRITE,
+						   &bufioreq_pfn,
+						   &map_err, 1);
     if (state->buffered_io_page == NULL) {
         hw_error("map buffered IO page returned error %d (%d)", errno, map_err);
     }
diff --git a/xen-mapcache.c b/xen-mapcache.c
index 66da1a6..223fe4d 100644
--- a/xen-mapcache.c
+++ b/xen-mapcache.c
@@ -169,10 +169,10 @@ static void xen_remap_bucket(MapCacheEntry *entry,
         pfns[i] = (address_index << (MCACHE_BUCKET_SHIFT-XC_PAGE_SHIFT)) + i;
     }
 
-    vaddr_base = xc_map_foreign_bulk(xen_xc, xen_domid, PROT_READ|PROT_WRITE,
-                                     pfns, err, nb_pfn);
+    vaddr_base = xenforeignmemory_map(xen_fmem, xen_domid, PROT_READ|PROT_WRITE,
+				      pfns, err, nb_pfn);
     if (vaddr_base == NULL) {
-        perror("xc_map_foreign_bulk");
+        perror("xenforeignmemory_map");
         exit(-1);
     }
 
-- 
2.1.4

  parent reply	other threads:[~2015-10-07 14:16 UTC|newest]

Thread overview: 88+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2015-10-07 14:02 [PATCH v3 0/<VARIOUS>] Begin to disentangle libxenctrl and provide some stable libraries Ian Campbell
2015-10-07 14:10 ` Ian Jackson
2015-10-07 14:18   ` Ian Campbell
2015-10-07 14:28     ` [Minios-devel] " Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 01/22] tools/Rules.mk: Properly handle libraries with recursive dependencies Ian Campbell
2015-10-08 11:21   ` Ian Jackson
2015-10-08 11:38     ` Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 02/22] tools: Refactor "xentoollog" into its own library Ian Campbell
2015-10-07 14:42   ` Andrew Cooper
2015-10-07 14:50     ` Ian Campbell
2015-10-08 10:17       ` Ian Jackson
2015-10-08 11:23   ` Ian Jackson
2015-10-08 11:34     ` Ian Campbell
2015-10-08 13:51       ` Ian Jackson
2015-10-07 14:15 ` [PATCH XEN v3 03/22] tools/libxc: Remove osdep indirection for xc_evtchn Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 04/22] tools: Refactor /dev/xen/evtchn wrappers into libxenevtchn Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 05/22] tools: Arrange to check public headers for ANSI compatiblity Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 06/22] tools/libxc: Remove osdep indirection for xc_gnt{shr, tab} Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 07/22] tools: Refactor /dev/xen/gnt{dev, shr} wrappers into libxengnttab Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 08/22] tools/libxc: Remove osdep indirection for privcmd Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 09/22] tools: Refactor hypercall calling wrappers into libxencall Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 10/22] tools/libxc: drop xc_map_foreign_bulk_compat wrappers Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 11/22] tools: Remove xc_map_foreign_batch Ian Campbell
2015-10-07 15:08   ` George Dunlap
2015-10-07 15:20     ` Ian Campbell
2015-10-14 15:05     ` Ian Campbell
2015-10-14 15:11       ` George Dunlap
2015-10-07 14:15 ` [PATCH XEN v3 12/22] tools: Implement xc_map_foreign_range(s) in terms of common helper Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 13/22] tools: Refactor foreign memory mapping into libxenforeignmemory Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 14/22] tools: foreignmemory: provide xenforeignmemory_unmap Ian Campbell
2015-10-07 15:03   ` Andrew Cooper
2015-10-07 15:16     ` Ian Campbell
2015-10-07 15:20       ` Andrew Cooper
2015-10-07 15:30         ` Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 15/22] tools/libs/evtchn: Review and update doc comments Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 16/22] tools/libs: Clean up hard tabs Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 17/22] tools/libs/gnttab: Review and update doc comments Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 18/22] tools/libs/call: Update some log messages to not refer to xc Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 19/22] tools/libs/call: Avoid xc_memalign in netbsd and solaris backends Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 20/22] tools/libs/foreignmemory: Mention restrictions on fork in docs Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 21/22] tools: Update CFLAGS for qemu-xen to allow it to use new libraries Ian Campbell
2015-10-07 14:15 ` [PATCH XEN v3 22/22] HACK: Add a .config to pull all the right bits from my xenbits trees Ian Campbell
2015-10-07 14:15 ` [PATCH QEMU-XEN v3 1/8] xen: Switch to libxenevtchn interface for compat shims Ian Campbell
2015-10-07 14:36   ` Andrew Cooper
2015-10-07 14:48     ` Ian Campbell
2015-10-14 13:16   ` Stefano Stabellini
2015-10-14 13:25     ` Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 2/8] xen: Switch to libxengnttab " Ian Campbell
2015-10-14 13:30   ` Stefano Stabellini
2015-10-14 13:40     ` Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 3/8] xen: Switch uses of xc_map_foreign_range into xc_map_foreign_bulk Ian Campbell
2015-10-14 14:01   ` Stefano Stabellini
2015-10-14 14:20     ` Ian Campbell
2015-10-14 14:21       ` Stefano Stabellini
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 4/8] xen: Switch uses of xc_map_foreign_pages " Ian Campbell
2015-10-14 14:03   ` Stefano Stabellini
2015-10-07 14:16 ` Ian Campbell [this message]
2015-10-14 14:10   ` [PATCH QEMU-XEN v3 5/8] xen: Switch uses of xc_map_foreign_bulk to use libxenforeignmemory API Stefano Stabellini
2015-10-14 14:31     ` Ian Campbell
2015-10-14 15:41       ` Stefano Stabellini
2015-10-14 15:59         ` Ian Campbell
2015-10-14 16:06           ` Ian Campbell
2015-10-14 16:29           ` Stefano Stabellini
2015-10-14 16:48             ` Ian Campbell
2015-10-14 17:17               ` Stefano Stabellini
2015-10-15 11:21                 ` Ian Campbell
2015-10-15 14:49                   ` Stefano Stabellini
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 6/8] xen: Use stable library interfaces when they are available Ian Campbell
2015-10-14 14:52   ` Stefano Stabellini
2015-10-15 14:23     ` Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 7/8] xen: domainbuild: reopen libxenctrl interface after forking for domain watcher Ian Campbell
2015-10-14 14:22   ` Stefano Stabellini
2015-10-07 14:16 ` [PATCH QEMU-XEN v3 8/8] xen: make it possible to build without the Xen PV domain builder Ian Campbell
2015-10-14 14:24   ` Stefano Stabellini
2015-10-07 14:16 ` [PATCH QEMU-XEN-TRADITIONAL v3 1/5] qemu-xen-traditional: Use xentoollog as a separate library Ian Campbell
2015-10-08 11:33   ` Ian Jackson
2015-10-08 11:45     ` Ian Campbell
2015-10-08 13:53       ` Ian Jackson
2015-10-08 14:05         ` Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN-TRADITIONAL v3 2/5] qemu-xen-traditional: Use libxenevtchn Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN-TRADITIONAL v3 3/5] qemu-xen-traditional: Use libxengnttab Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN-TRADITIONAL v3 4/5] qemu-xen-traditional: Add libxencall to rpath-link Ian Campbell
2015-10-07 14:16 ` [PATCH QEMU-XEN-TRADITIONAL v3 5/5] qemu-xen-traditional: Add libxenforeignmemory " Ian Campbell
2015-10-07 14:16 ` [PATCH MINI-OS v3 1/5] mini-os: Include libxentoollog with libxc Ian Campbell
2015-10-07 14:16 ` [PATCH MINI-OS v3 2/5] mini-os: Include libxenevtchn " Ian Campbell
2015-10-07 14:16 ` [PATCH MINI-OS v3 3/5] mini-os: Include libxengnttab " Ian Campbell
2015-10-07 14:16 ` [PATCH MINI-OS v3 4/5] mini-os: Include libxencall " Ian Campbell
2015-10-07 14:16 ` [PATCH MINI-OS v3 5/5] mini-os: Include libxenforeignmemory " Ian Campbell

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=1444227366-1015-5-git-send-email-ian.campbell@citrix.com \
    --to=ian.campbell@citrix.com \
    --cc=ian.jackson@eu.citrix.com \
    --cc=stefano.stabellini@eu.citrix.com \
    --cc=wei.liu2@citrix.com \
    --cc=xen-devel@lists.xen.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).