|  | // Copyright 2016 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 "goldfish_dma.h" | 
|  |  | 
|  | #define ALLOW_DEPRECATED_QEMU_PIPE_HEADERS | 
|  | #include <qemu_pipe.h> | 
|  |  | 
|  | #if PLATFORM_SDK_VERSION < 26 | 
|  | #include <cutils/log.h> | 
|  | #else | 
|  | #include <log/log.h> | 
|  | #endif | 
|  | #include <errno.h> | 
|  | #ifdef __ANDROID__ | 
|  | #include <linux/ioctl.h> | 
|  | #include <linux/types.h> | 
|  | #include <sys/cdefs.h> | 
|  | #endif | 
|  | #include <sys/ioctl.h> | 
|  | #include <sys/mman.h> | 
|  | #include <fcntl.h> | 
|  | #include <stdlib.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  |  | 
|  | /* There is an ioctl associated with goldfish dma driver. | 
|  | * Make it conflict with ioctls that are not likely to be used | 
|  | * in the emulator. | 
|  | * 'G'	00-3F	drivers/misc/sgi-gru/grulib.h	conflict! | 
|  | * 'G'	00-0F	linux/gigaset_dev.h	conflict! | 
|  | */ | 
|  | #define GOLDFISH_DMA_IOC_MAGIC	'G' | 
|  |  | 
|  | #define GOLDFISH_DMA_IOC_LOCK			_IOWR(GOLDFISH_DMA_IOC_MAGIC, 0, struct goldfish_dma_ioctl_info) | 
|  | #define GOLDFISH_DMA_IOC_UNLOCK			_IOWR(GOLDFISH_DMA_IOC_MAGIC, 1, struct goldfish_dma_ioctl_info) | 
|  | #define GOLDFISH_DMA_IOC_GETOFF			_IOWR(GOLDFISH_DMA_IOC_MAGIC, 2, struct goldfish_dma_ioctl_info) | 
|  | #define GOLDFISH_DMA_IOC_CREATE_REGION	_IOWR(GOLDFISH_DMA_IOC_MAGIC, 3, struct goldfish_dma_ioctl_info) | 
|  |  | 
|  | struct goldfish_dma_ioctl_info { | 
|  | uint64_t phys_begin; | 
|  | uint64_t size; | 
|  | }; | 
|  |  | 
|  | int goldfish_dma_create_region(uint32_t sz, struct goldfish_dma_context* res) { | 
|  |  | 
|  | res->fd = qemu_pipe_open("opengles"); | 
|  | res->mapped_addr = 0; | 
|  | res->size = 0; | 
|  |  | 
|  | if (res->fd > 0) { | 
|  | // now alloc | 
|  | struct goldfish_dma_ioctl_info info; | 
|  | info.size = sz; | 
|  | int alloc_res = ioctl(res->fd, GOLDFISH_DMA_IOC_CREATE_REGION, &info); | 
|  |  | 
|  | if (alloc_res) { | 
|  | ALOGE("%s: failed to allocate DMA region. errno=%d", | 
|  | __FUNCTION__, errno); | 
|  | close(res->fd); | 
|  | res->fd = -1; | 
|  | return alloc_res; | 
|  | } | 
|  |  | 
|  | res->size = sz; | 
|  | ALOGV("%s: successfully allocated goldfish DMA region with size %u cxt=%p fd=%d", | 
|  | __FUNCTION__, sz, res, res->fd); | 
|  | return 0; | 
|  | } else { | 
|  | ALOGE("%s: could not obtain fd to device! fd %d errno=%d\n", | 
|  | __FUNCTION__, res->fd, errno); | 
|  | return ENODEV; | 
|  | } | 
|  | } | 
|  |  | 
|  | void* goldfish_dma_map(struct goldfish_dma_context* cxt) { | 
|  | ALOGV("%s: on fd %d errno=%d", __FUNCTION__, cxt->fd, errno); | 
|  | void *mapped = mmap(0, cxt->size, PROT_WRITE, MAP_SHARED, cxt->fd, 0); | 
|  | ALOGV("%s: cxt=%p mapped=%p size=%u errno=%d", | 
|  | __FUNCTION__, cxt, mapped, cxt->size, errno); | 
|  |  | 
|  | if (mapped == MAP_FAILED) { | 
|  | mapped = NULL; | 
|  | } | 
|  |  | 
|  | cxt->mapped_addr = reinterpret_cast<uint64_t>(mapped); | 
|  | return mapped; | 
|  | } | 
|  |  | 
|  | int goldfish_dma_unmap(struct goldfish_dma_context* cxt) { | 
|  | ALOGV("%s: cxt=%p mapped=0x%" PRIu64, __FUNCTION__, cxt, cxt->mapped_addr); | 
|  | munmap(reinterpret_cast<void *>(cxt->mapped_addr), cxt->size); | 
|  | cxt->mapped_addr = 0; | 
|  | cxt->size = 0; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | void goldfish_dma_write(struct goldfish_dma_context* cxt, | 
|  | const void* to_write, | 
|  | uint32_t sz) { | 
|  | ALOGV("%s: cxt=%p mapped=0x%" PRIu64 " to_write=%p size=%u", | 
|  | __FUNCTION__, cxt, cxt->mapped_addr, to_write, sz); | 
|  | memcpy(reinterpret_cast<void *>(cxt->mapped_addr), to_write, sz); | 
|  | } | 
|  |  | 
|  | void goldfish_dma_free(goldfish_dma_context* cxt) { | 
|  | close(cxt->fd); | 
|  | } | 
|  |  | 
|  | uint64_t goldfish_dma_guest_paddr(const struct goldfish_dma_context* cxt) { | 
|  | struct goldfish_dma_ioctl_info info; | 
|  | ioctl(cxt->fd, GOLDFISH_DMA_IOC_GETOFF, &info); | 
|  | return info.phys_begin; | 
|  | } |