mmap_unlock();
return new_addr;
}
+
+static bool can_passthrough_madv_dontneed(abi_ulong start, abi_ulong end)
+{
+ ulong addr;
+
+ if ((start | end) & ~qemu_host_page_mask) {
+ return false;
+ }
+
+ for (addr = start; addr < end; addr += TARGET_PAGE_SIZE) {
+ if (!(page_get_flags(addr) & PAGE_ANON)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+abi_long target_madvise(abi_ulong start, abi_ulong len_in, int advice)
+{
+ abi_ulong len, end;
+ int ret = 0;
+
+ if (start & ~TARGET_PAGE_MASK) {
+ return -TARGET_EINVAL;
+ }
+ len = TARGET_PAGE_ALIGN(len_in);
+
+ if (len_in && !len) {
+ return -TARGET_EINVAL;
+ }
+
+ end = start + len;
+ if (end < start) {
+ return -TARGET_EINVAL;
+ }
+
+ if (end == start) {
+ return 0;
+ }
+
+ if (!guest_range_valid_untagged(start, len)) {
+ return -TARGET_EINVAL;
+ }
+
+ /*
+ * A straight passthrough may not be safe because qemu sometimes turns
+ * private file-backed mappings into anonymous mappings.
+ *
+ * This is a hint, so ignoring and returning success is ok.
+ *
+ * This breaks MADV_DONTNEED, completely implementing which is quite
+ * complicated. However, there is one low-hanging fruit: host-page-aligned
+ * anonymous mappings. In this case passthrough is safe, so do it.
+ */
+ mmap_lock();
+ if ((advice & MADV_DONTNEED) &&
+ can_passthrough_madv_dontneed(start, end)) {
+ ret = get_errno(madvise(g2h_untagged(start), len, MADV_DONTNEED));
+ }
+ mmap_unlock();
+
+ return ret;
+}
}
}
-static inline abi_long get_errno(abi_long ret)
+abi_long get_errno(abi_long ret)
{
if (ret == -1)
return -host_to_target_errno(errno);
#ifdef TARGET_NR_madvise
case TARGET_NR_madvise:
- /* A straight passthrough may not be safe because qemu sometimes
- turns private file-backed mappings into anonymous mappings.
- This will break MADV_DONTNEED.
- This is a hint, so ignoring and returning success is ok. */
- return 0;
+ return target_madvise(arg1, arg2, arg3);
#endif
#ifdef TARGET_NR_fcntl64
case TARGET_NR_fcntl64:
abi_long arg8);
extern __thread CPUState *thread_cpu;
G_NORETURN void cpu_loop(CPUArchState *env);
+abi_long get_errno(abi_long ret);
const char *target_strerror(int err);
int get_osversion(void);
void init_qemu_uname_release(void);
abi_long target_mremap(abi_ulong old_addr, abi_ulong old_size,
abi_ulong new_size, unsigned long flags,
abi_ulong new_addr);
+abi_long target_madvise(abi_ulong start, abi_ulong len_in, int advice);
extern unsigned long last_brk;
extern abi_ulong mmap_next_start;
abi_ulong mmap_find_vma(abi_ulong, abi_ulong, abi_ulong);