/*
 * Copyright (C) 2018 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include "mem_map.h"
#include <sys/mman.h>
#include "logging.h"

#include <zircon/process.h>
#include <zircon/syscalls.h>

namespace art {

static zx_handle_t fuchsia_lowmem_vmar = ZX_HANDLE_INVALID;
static zx_vaddr_t fuchsia_lowmem_base = 0;
static size_t fuchsia_lowmem_size = 0;

static const char map_name[] = "mmap-android";
static constexpr uintptr_t FUCHSIA_LOWER_MEM_START = 0x80000000;
static constexpr uintptr_t FUCHSIA_LOWER_MEM_SIZE  = 0x60000000;

void MemMap::TargetMMapInit() {
  if (fuchsia_lowmem_vmar != ZX_HANDLE_INVALID) {
    return;
  }

  zx_info_vmar_t vmarinfo;
  CHECK_EQ(zx_object_get_info(zx_vmar_root_self(),
                              ZX_INFO_VMAR,
                              &vmarinfo,
                              sizeof(vmarinfo),
                              nullptr,
                              nullptr), ZX_OK) << "could not find info from root vmar";

  uintptr_t lower_mem_start = FUCHSIA_LOWER_MEM_START - vmarinfo.base;
  fuchsia_lowmem_size = FUCHSIA_LOWER_MEM_SIZE;
  uint32_t allocflags = ZX_VM_FLAG_CAN_MAP_READ |
                        ZX_VM_FLAG_CAN_MAP_WRITE |
                        ZX_VM_FLAG_CAN_MAP_EXECUTE |
                        ZX_VM_FLAG_SPECIFIC;
  CHECK_EQ(zx_vmar_allocate(zx_vmar_root_self(),
                            lower_mem_start,
                            fuchsia_lowmem_size,
                            allocflags,
                            &fuchsia_lowmem_vmar,
                            &fuchsia_lowmem_base), ZX_OK) << "could not allocate lowmem vmar";
}

void* MemMap::TargetMMap(void* start, size_t len, int prot, int flags, int fd, off_t fd_off) {
  zx_status_t status;
  uintptr_t mem = 0;

  bool mmap_lower = (flags & MAP_32BIT) != 0;

  // for file-based mapping use system library
  if ((flags & MAP_ANONYMOUS) == 0) {
    if (start != nullptr) {
      flags |= MAP_FIXED;
    }
    CHECK(!mmap_lower) << "cannot map files into low memory for Fuchsia";
    return mmap(start, len, prot, flags, fd, fd_off);
  }

  uint32_t vmarflags = 0;
  if ((prot & PROT_READ) != 0) {
    vmarflags |= ZX_VM_FLAG_PERM_READ;
  }
  if ((prot & PROT_WRITE) != 0) {
    vmarflags |= ZX_VM_FLAG_PERM_WRITE;
  }
  if ((prot & PROT_EXEC) != 0) {
    vmarflags |= ZX_VM_FLAG_PERM_EXECUTE;
  }

  if (len == 0) {
    errno = EINVAL;
    return MAP_FAILED;
  }

  zx_info_vmar_t vmarinfo;
  size_t vmaroffset = 0;
  if (start != nullptr) {
    vmarflags |= ZX_VM_FLAG_SPECIFIC;
    status = zx_object_get_info((mmap_lower ? fuchsia_lowmem_vmar : zx_vmar_root_self()),
                                ZX_INFO_VMAR,
                                &vmarinfo,
                                sizeof(vmarinfo),
                                nullptr,
                                nullptr);
    if (status < 0 || reinterpret_cast<uintptr_t>(start) < vmarinfo.base) {
      errno = EINVAL;
      return MAP_FAILED;
    }
    vmaroffset = reinterpret_cast<uintptr_t>(start) - vmarinfo.base;
  }

  zx_handle_t vmo;
  if (zx_vmo_create(len, 0, &vmo) < 0) {
    errno = ENOMEM;
    return MAP_FAILED;
  }
  zx_vmo_get_size(vmo, &len);
  zx_object_set_property(vmo, ZX_PROP_NAME, map_name, strlen(map_name));

  if (mmap_lower) {
    status = zx_vmar_map(fuchsia_lowmem_vmar, vmaroffset, vmo, fd_off, len, vmarflags, &mem);
  } else {
    status = zx_vmar_map(zx_vmar_root_self(), vmaroffset, vmo, fd_off, len, vmarflags, &mem);
  }
  zx_handle_close(vmo);
  if (status != ZX_OK) {
    return MAP_FAILED;
  }

  return reinterpret_cast<void *>(mem);
}

int MemMap::TargetMUnmap(void* start, size_t len) {
  uintptr_t addr = reinterpret_cast<uintptr_t>(start);
  zx_handle_t alloc_vmar = zx_vmar_root_self();
  if (addr >= fuchsia_lowmem_base && addr < fuchsia_lowmem_base + fuchsia_lowmem_size) {
    alloc_vmar = fuchsia_lowmem_vmar;
  }
  zx_status_t status = zx_vmar_unmap(alloc_vmar, addr, len);
  if (status < 0) {
    errno = EINVAL;
    return -1;
  }
  return 0;
}

}  // namespace art
