/* * Copyright (C) 2008 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 /* * Implementation of the user-space ashmem API for the simulator, which lacks an * ashmem-enabled kernel. See ashmem-dev.c for the real ashmem-based version. A * disk-backed temp file is the best option that is consistently supported * across all host platforms. */ #include #include #include #include #include #include #include #include #include #include #include #include #include using android::base::unique_fd; static bool ashmem_validate_stat(int fd, struct stat* buf) { int result = fstat(fd, buf); if (result == -1) { return false; } // Check if this is an ashmem region. Since there's no such thing on the host, // we can't actually implement that. Check that it's at least a regular file. if (!S_ISREG(buf->st_mode)) { errno = ENOTTY; return false; } // In Win32, unlike Unix, the temp file is not unlinked immediately after // creation. #if !defined(_WIN32) if (buf->st_nlink != 0) { errno = ENOTTY; return false; } #endif return true; } int ashmem_valid(int fd) { struct stat buf; return ashmem_validate_stat(fd, &buf); } int ashmem_create_region(const char* /*ignored*/, size_t size) { // Files returned by tmpfile are automatically removed. std::unique_ptr tmp(tmpfile(), &fclose); if (!tmp) { return -1; } int fd = fileno(tmp.get()); if (fd == -1) { return -1; } unique_fd dupfd = unique_fd(dup(fd)); if (dupfd == -1) { return -1; } if (TEMP_FAILURE_RETRY(ftruncate(dupfd, size)) == -1) { return -1; } return dupfd.release(); } int ashmem_set_prot_region(int /*fd*/, int /*prot*/) { return 0; } int ashmem_pin_region(int /*fd*/, size_t /*offset*/, size_t /*len*/) { return 0 /*ASHMEM_NOT_PURGED*/; } int ashmem_unpin_region(int /*fd*/, size_t /*offset*/, size_t /*len*/) { return 0 /*ASHMEM_IS_UNPINNED*/; } int ashmem_get_size_region(int fd) { struct stat buf; if (!ashmem_validate_stat(fd, &buf)) { return -1; } return buf.st_size; }