shm_open() creates and opens a new POSIX shared memory object. A POSIX shared memory object allows creating memory backend with an associated file descriptor that can be shared with external processes (e.g. vhost-user). The new `memory-backend-shm` can be used as an alternative when `memory-backend-memfd` is not available (Linux only), since shm_open() should be provided by any POSIX-compliant operating system. This backend mimics memfd, allocating memory that is practically anonymous. In theory shm_open() requires a name, but this is allocated for a short time interval and shm_unlink() is called right after shm_open(). After that, only fd is shared with external processes (e.g., vhost-user) as if it were associated with anonymous memory. In the future we may also allow the user to specify the name to be passed to shm_open(), but for now we keep the backend simple, mimicking anonymous memory such as memfd. Acked-by: David Hildenbrand <david@redhat.com> Acked-by: Stefan Hajnoczi <stefanha@redhat.com> Acked-by: Markus Armbruster <armbru@redhat.com> (QAPI schema) Signed-off-by: Stefano Garzarella <sgarzare@redhat.com> Message-Id: <20240618100519.145853-1-sgarzare@redhat.com> Reviewed-by: Michael S. Tsirkin <mst@redhat.com> Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
		
			
				
	
	
		
			124 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * QEMU host POSIX shared memory object backend
 | 
						|
 *
 | 
						|
 * Copyright (C) 2024 Red Hat Inc
 | 
						|
 *
 | 
						|
 * Authors:
 | 
						|
 *   Stefano Garzarella <sgarzare@redhat.com>
 | 
						|
 *
 | 
						|
 * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | 
						|
 * See the COPYING file in the top-level directory.
 | 
						|
 */
 | 
						|
 | 
						|
#include "qemu/osdep.h"
 | 
						|
#include "sysemu/hostmem.h"
 | 
						|
#include "qapi/error.h"
 | 
						|
 | 
						|
#define TYPE_MEMORY_BACKEND_SHM "memory-backend-shm"
 | 
						|
 | 
						|
OBJECT_DECLARE_SIMPLE_TYPE(HostMemoryBackendShm, MEMORY_BACKEND_SHM)
 | 
						|
 | 
						|
struct HostMemoryBackendShm {
 | 
						|
    HostMemoryBackend parent_obj;
 | 
						|
};
 | 
						|
 | 
						|
static bool
 | 
						|
shm_backend_memory_alloc(HostMemoryBackend *backend, Error **errp)
 | 
						|
{
 | 
						|
    g_autoptr(GString) shm_name = g_string_new(NULL);
 | 
						|
    g_autofree char *backend_name = NULL;
 | 
						|
    uint32_t ram_flags;
 | 
						|
    int fd, oflag;
 | 
						|
    mode_t mode;
 | 
						|
 | 
						|
    if (!backend->size) {
 | 
						|
        error_setg(errp, "can't create shm backend with size 0");
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!backend->share) {
 | 
						|
        error_setg(errp, "can't create shm backend with `share=off`");
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
     * Let's use `mode = 0` because we don't want other processes to open our
 | 
						|
     * memory unless we share the file descriptor with them.
 | 
						|
     */
 | 
						|
    mode = 0;
 | 
						|
    oflag = O_RDWR | O_CREAT | O_EXCL;
 | 
						|
    backend_name = host_memory_backend_get_name(backend);
 | 
						|
 | 
						|
    /*
 | 
						|
     * Some operating systems allow creating anonymous POSIX shared memory
 | 
						|
     * objects (e.g. FreeBSD provides the SHM_ANON constant), but this is not
 | 
						|
     * defined by POSIX, so let's create a unique name.
 | 
						|
     *
 | 
						|
     * From Linux's shm_open(3) man-page:
 | 
						|
     *   For  portable  use,  a shared  memory  object should be identified
 | 
						|
     *   by a name of the form /somename;"
 | 
						|
     */
 | 
						|
    g_string_printf(shm_name, "/qemu-" FMT_pid "-shm-%s", getpid(),
 | 
						|
                    backend_name);
 | 
						|
 | 
						|
    fd = shm_open(shm_name->str, oflag, mode);
 | 
						|
    if (fd < 0) {
 | 
						|
        error_setg_errno(errp, errno,
 | 
						|
                         "failed to create POSIX shared memory");
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    /*
 | 
						|
     * We have the file descriptor, so we no longer need to expose the
 | 
						|
     * POSIX shared memory object. However it will remain allocated as long as
 | 
						|
     * there are file descriptors pointing to it.
 | 
						|
     */
 | 
						|
    shm_unlink(shm_name->str);
 | 
						|
 | 
						|
    if (ftruncate(fd, backend->size) == -1) {
 | 
						|
        error_setg_errno(errp, errno,
 | 
						|
                         "failed to resize POSIX shared memory to %" PRIu64,
 | 
						|
                         backend->size);
 | 
						|
        close(fd);
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    ram_flags = RAM_SHARED;
 | 
						|
    ram_flags |= backend->reserve ? 0 : RAM_NORESERVE;
 | 
						|
 | 
						|
    return memory_region_init_ram_from_fd(&backend->mr, OBJECT(backend),
 | 
						|
                                              backend_name, backend->size,
 | 
						|
                                              ram_flags, fd, 0, errp);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
shm_backend_instance_init(Object *obj)
 | 
						|
{
 | 
						|
    HostMemoryBackendShm *m = MEMORY_BACKEND_SHM(obj);
 | 
						|
 | 
						|
    MEMORY_BACKEND(m)->share = true;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
shm_backend_class_init(ObjectClass *oc, void *data)
 | 
						|
{
 | 
						|
    HostMemoryBackendClass *bc = MEMORY_BACKEND_CLASS(oc);
 | 
						|
 | 
						|
    bc->alloc = shm_backend_memory_alloc;
 | 
						|
}
 | 
						|
 | 
						|
static const TypeInfo shm_backend_info = {
 | 
						|
    .name = TYPE_MEMORY_BACKEND_SHM,
 | 
						|
    .parent = TYPE_MEMORY_BACKEND,
 | 
						|
    .instance_init = shm_backend_instance_init,
 | 
						|
    .class_init = shm_backend_class_init,
 | 
						|
    .instance_size = sizeof(HostMemoryBackendShm),
 | 
						|
};
 | 
						|
 | 
						|
static void register_types(void)
 | 
						|
{
 | 
						|
    type_register_static(&shm_backend_info);
 | 
						|
}
 | 
						|
 | 
						|
type_init(register_types);
 |