2015-08-21 07:04:50 +00:00
|
|
|
/*
|
|
|
|
* os-posix-lib.c
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003-2008 Fabrice Bellard
|
|
|
|
* Copyright (c) 2010 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* QEMU library functions on POSIX which are shared between QEMU and
|
|
|
|
* the QEMU tools.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2018-02-23 18:56:50 +00:00
|
|
|
#if defined(__linux__) && \
|
2018-06-15 15:53:45 +00:00
|
|
|
(defined(__x86_64__) || defined(__arm__) || defined(__aarch64__) \
|
|
|
|
|| defined(__powerpc64__))
|
2015-08-21 07:04:50 +00:00
|
|
|
/* Use 2 MiB alignment so transparent hugepages can be used by KVM.
|
|
|
|
Valgrind does not support alignments larger than 1 MiB,
|
|
|
|
therefore we need special code which handles running on Valgrind. */
|
|
|
|
# define QEMU_VMALLOC_ALIGN (512 * 4096)
|
|
|
|
#elif defined(__linux__) && defined(__s390x__)
|
|
|
|
/* Use 1 MiB (segment size) alignment so gmap can be used by KVM. */
|
|
|
|
# define QEMU_VMALLOC_ALIGN (256 * 4096)
|
2018-03-05 19:09:52 +00:00
|
|
|
#elif defined(__linux__) && defined(__sparc__)
|
|
|
|
#include <sys/shm.h>
|
|
|
|
# define QEMU_VMALLOC_ALIGN MAX(getpagesize(), SHMLBA)
|
2015-08-21 07:04:50 +00:00
|
|
|
#else
|
|
|
|
# define QEMU_VMALLOC_ALIGN getpagesize()
|
|
|
|
#endif
|
|
|
|
#define HUGETLBFS_MAGIC 0x958458f6
|
|
|
|
|
2018-02-19 06:27:52 +00:00
|
|
|
#include "qemu/osdep.h"
|
include/qemu/osdep.h: Don't include qapi/error.h
Commit 57cb38b included qapi/error.h into qemu/osdep.h to get the
Error typedef. Since then, we've moved to include qemu/osdep.h
everywhere. Its file comment explains: "To avoid getting into
possible circular include dependencies, this file should not include
any other QEMU headers, with the exceptions of config-host.h,
compiler.h, os-posix.h and os-win32.h, all of which are doing a
similar job to this file and are under similar constraints."
qapi/error.h doesn't do a similar job, and it doesn't adhere to
similar constraints: it includes qapi-types.h. That's in excess of
100KiB of crap most .c files don't actually need.
Add the typedef to qemu/typedefs.h, and include that instead of
qapi/error.h. Include qapi/error.h in .c files that need it and don't
get it now. Include qapi-types.h in qom/object.h for uint16List.
Update scripts/clean-includes accordingly. Update it further to match
reality: replace config.h by config-target.h, add sysemu/os-posix.h,
sysemu/os-win32.h. Update the list of includes in the qemu/osdep.h
comment quoted above similarly.
This reduces the number of objects depending on qapi/error.h from "all
of them" to less than a third. Unfortunately, the number depending on
qapi-types.h shrinks only a little. More work is needed for that one.
Backports commit da34e65cb4025728566d6504a99916f6e7e1dd6a from qemu
2018-02-22 04:05:15 +00:00
|
|
|
#include "qapi/error.h"
|
2017-01-20 13:13:21 +00:00
|
|
|
#include "unicorn/platform.h"
|
2015-08-21 07:04:50 +00:00
|
|
|
#include "sysemu/sysemu.h"
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <libgen.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <sys/signal.h>
|
2018-02-22 14:24:12 +00:00
|
|
|
#include "qemu/cutils.h"
|
2015-08-21 07:04:50 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_LINUX
|
2015-12-17 05:20:57 +00:00
|
|
|
#if !defined(__CYGWIN__)
|
2015-08-21 07:04:50 +00:00
|
|
|
#include <sys/syscall.h>
|
2015-12-17 05:20:57 +00:00
|
|
|
#endif
|
2015-08-21 07:04:50 +00:00
|
|
|
#include <sys/vfs.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __FreeBSD__
|
|
|
|
#include <sys/sysctl.h>
|
2018-03-01 13:55:35 +00:00
|
|
|
#include <sys/user.h>
|
2015-08-21 07:04:50 +00:00
|
|
|
#endif
|
|
|
|
|
2018-02-25 09:10:25 +00:00
|
|
|
#include "qemu/mmap-alloc.h"
|
2018-02-16 20:17:07 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
void *qemu_oom_check(void *ptr)
|
|
|
|
{
|
|
|
|
if (ptr == NULL) {
|
|
|
|
fprintf(stderr, "Failed to allocate memory: %s\n", strerror(errno));
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *qemu_try_memalign(size_t alignment, size_t size)
|
|
|
|
{
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
if (alignment < sizeof(void*)) {
|
|
|
|
alignment = sizeof(void*);
|
|
|
|
}
|
|
|
|
|
2018-03-08 13:55:22 +00:00
|
|
|
#if defined(CONFIG_POSIX_MEMALIGN)
|
2015-08-21 07:04:50 +00:00
|
|
|
int ret;
|
|
|
|
ret = posix_memalign(&ptr, alignment, size);
|
|
|
|
if (ret != 0) {
|
|
|
|
errno = ret;
|
|
|
|
ptr = NULL;
|
|
|
|
}
|
|
|
|
#elif defined(CONFIG_BSD)
|
|
|
|
ptr = valloc(size);
|
|
|
|
#else
|
|
|
|
ptr = memalign(alignment, size);
|
|
|
|
#endif
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *qemu_memalign(size_t alignment, size_t size)
|
|
|
|
{
|
|
|
|
return qemu_oom_check(qemu_try_memalign(alignment, size));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* alloc shared memory pages */
|
|
|
|
void *qemu_anon_ram_alloc(size_t size, uint64_t *alignment)
|
|
|
|
{
|
|
|
|
size_t align = QEMU_VMALLOC_ALIGN;
|
2018-02-16 20:17:07 +00:00
|
|
|
void *ptr = qemu_ram_mmap(-1, size, align, false);
|
2015-08-21 07:04:50 +00:00
|
|
|
|
|
|
|
if (ptr == MAP_FAILED) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (alignment) {
|
|
|
|
*alignment = align;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_vfree(void *ptr)
|
|
|
|
{
|
|
|
|
free(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_anon_ram_free(void *ptr, size_t size)
|
|
|
|
{
|
2018-02-16 20:17:07 +00:00
|
|
|
qemu_ram_munmap(ptr, size);
|
2015-08-21 07:04:50 +00:00
|
|
|
}
|