Index: third_party/android_crazy_linker/src/tests/test_util.h |
diff --git a/third_party/android_crazy_linker/src/tests/test_util.h b/third_party/android_crazy_linker/src/tests/test_util.h |
new file mode 100644 |
index 0000000000000000000000000000000000000000..28fd39fa9f93ea7e19682d96e7abb0b77f8be96b |
--- /dev/null |
+++ b/third_party/android_crazy_linker/src/tests/test_util.h |
@@ -0,0 +1,465 @@ |
+// Copyright (c) 2013 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+// A set of common helper functions used by crazy_linker tests. |
+// IMPORTANT: ALL FUNCTIONS HERE ARE INLINED. This avoids adding a |
+// dependency on another source file for all tests that include this |
+// header. |
+ |
+#ifndef TEST_UTIL_H |
+#define TEST_UTIL_H |
+ |
+#include <crazy_linker.h> |
+ |
+#include <dirent.h> |
+#include <errno.h> |
+#include <limits.h> |
+#include <stdarg.h> |
+#include <stdio.h> |
+#include <stdlib.h> |
+#include <sys/mman.h> |
+#include <sys/socket.h> |
+#include <sys/stat.h> |
+#include <sys/uio.h> |
+#include <unistd.h> |
+ |
+namespace { |
+ |
+// Print an error message and exit the process. |
+// Message must be terminated by a newline. |
+inline void Panic(const char* fmt, ...) { |
+ va_list args; |
+ fprintf(stderr, "PANIC: "); |
+ va_start(args, fmt); |
+ vfprintf(stderr, fmt, args); |
+ va_end(args); |
+ exit(1); |
+} |
+ |
+// Print an error message, the errno message, then exit the process. |
+// Message must not be terminated by a newline. |
+inline void PanicErrno(const char* fmt, ...) { |
+ int old_errno = errno; |
+ va_list args; |
+ fprintf(stderr, "PANIC: "); |
+ va_start(args, fmt); |
+ vfprintf(stderr, fmt, args); |
+ va_end(args); |
+ fprintf(stderr, ": %s\n", strerror(old_errno)); |
+ exit(1); |
+} |
+ |
+// Simple string class. |
+class String { |
+ public: |
+ String() : str_(NULL), len_(0) {} |
+ |
+ String(const String& other) { String(other.str_, other.len_); } |
+ |
+ String(const char* str) { String(str, strlen(str)); } |
+ |
+ String(const char* str, size_t len) : str_(NULL), len_(0) { |
+ Append(str, len); |
+ } |
+ |
+ ~String() { |
+ if (str_) { |
+ free(str_); |
+ str_ = NULL; |
+ } |
+ } |
+ |
+ String& operator+=(const char* str) { |
+ Append(str, strlen(str)); |
+ return *this; |
+ } |
+ |
+ String& operator+=(const String& other) { |
+ Append(other.str_, other.len_); |
+ return *this; |
+ } |
+ |
+ String& operator+=(char ch) { |
+ Append(&ch, 1); |
+ return *this; |
+ } |
+ |
+ const char* c_str() const { return len_ ? str_ : ""; } |
+ char* ptr() { return str_; } |
+ size_t size() const { return len_; } |
+ |
+ void Append(const char* str, size_t len) { |
+ size_t old_len = len_; |
+ Resize(len_ + len); |
+ memcpy(str_ + old_len, str, len); |
+ } |
+ |
+ void Resize(size_t len) { |
+ str_ = reinterpret_cast<char*>(realloc(str_, len + 1)); |
+ if (len > len_) |
+ memset(str_ + len_, '\0', len - len_); |
+ str_[len] = '\0'; |
+ len_ = len; |
+ } |
+ |
+ void Format(const char* fmt, ...) { |
+ va_list args; |
+ va_start(args, fmt); |
+ Resize(128); |
+ for (;;) { |
+ va_list args2; |
+ va_copy(args2, args); |
+ int ret = vsnprintf(str_, len_ + 1, fmt, args2); |
+ va_end(args2); |
+ if (ret < static_cast<int>(len_ + 1)) |
+ break; |
+ |
+ Resize(len_ * 2); |
+ } |
+ } |
+ |
+ private: |
+ char* str_; |
+ size_t len_; |
+}; |
+ |
+// Helper class to create a temporary directory that gets deleted on scope exit, |
+// as well as all regular files it contains. |
+class TempDirectory { |
+ public: |
+ TempDirectory() { |
+ snprintf(path_, sizeof path_, "/data/local/tmp/temp-XXXXXX"); |
+ if (!mktemp(path_)) |
+ Panic("Could not create temporary directory name: %s\n", strerror(errno)); |
+ if (mkdir(path_, 0700) < 0) |
+ Panic("Could not create temporary directory %s: %s\n", strerror(errno)); |
+ } |
+ |
+ ~TempDirectory() { |
+ // Remove any file in this directory. |
+ DIR* d = opendir(path_); |
+ if (!d) |
+ Panic("Could not open directory %s: %s\n", strerror(errno)); |
+ |
+ struct dirent* entry; |
+ while ((entry = readdir(d)) != NULL) { |
+ if (!strcmp(entry->d_name, ".") || !strcmp(entry->d_name, "..")) |
+ continue; |
+ String file_path; |
+ file_path.Format("%s/%s", path_, entry->d_name); |
+ if (unlink(file_path.c_str()) < 0) |
+ Panic("Could not remove %s: %s\n", file_path.c_str(), strerror(errno)); |
+ } |
+ closedir(d); |
+ |
+ if (rmdir(path_) < 0) |
+ Panic("Could not remove dir %s: %s\n", path_, strerror(errno)); |
+ } |
+ |
+ const char* path() const { return path_; } |
+ |
+ private: |
+ char path_[PATH_MAX]; |
+}; |
+ |
+// Scoped FILE* class. Always closed on destruction. |
+class ScopedFILE { |
+ public: |
+ ScopedFILE() : file_(NULL) {} |
+ |
+ ~ScopedFILE() { |
+ if (file_) { |
+ fclose(file_); |
+ file_ = NULL; |
+ } |
+ } |
+ |
+ void Open(const char* path, const char* mode) { |
+ file_ = fopen(path, mode); |
+ if (!file_) |
+ Panic("Could not open file for reading: %s: %s\n", path, strerror(errno)); |
+ } |
+ |
+ FILE* file() const { return file_; } |
+ |
+ private: |
+ FILE* file_; |
+}; |
+ |
+// Retrieve current executable path as a String. |
+inline String GetCurrentExecutable() { |
+ String path; |
+ path.Resize(PATH_MAX); |
+ ssize_t ret = |
+ TEMP_FAILURE_RETRY(readlink("/proc/self/exe", path.ptr(), path.size())); |
+ if (ret < 0) |
+ Panic("Could not read /proc/self/exe: %s\n", strerror(errno)); |
+ |
+ return path; |
+} |
+ |
+// Retrieve current executable directory as a String. |
+inline String GetCurrentExecutableDirectory() { |
+ String path = GetCurrentExecutable(); |
+ // Find basename. |
+ const char* p = reinterpret_cast<const char*>(strrchr(path.c_str(), '/')); |
+ if (p == NULL) |
+ Panic("Current executable does not have directory root?: %s\n", |
+ path.c_str()); |
+ |
+ path.Resize(p - path.c_str()); |
+ return path; |
+} |
+ |
+// Copy a file named |src_file_name| in directory |src_file_dir| into |
+// a file named |dst_file_name| in directory |dst_file_dir|. Panics on error. |
+inline void CopyFile(const char* src_file_name, |
+ const char* src_file_dir, |
+ const char* dst_file_name, |
+ const char* dst_file_dir) { |
+ String src_path; |
+ src_path.Format("%s/%s", src_file_dir, src_file_name); |
+ |
+ ScopedFILE src_file; |
+ src_file.Open(src_path.c_str(), "rb"); |
+ |
+ String dst_path; |
+ dst_path.Format("%s/%s", dst_file_dir, dst_file_name); |
+ ScopedFILE dst_file; |
+ dst_file.Open(dst_path.c_str(), "wb"); |
+ |
+ char buffer[8192]; |
+ for (;;) { |
+ size_t read = fread(buffer, 1, sizeof buffer, src_file.file()); |
+ if (read > 0) { |
+ size_t written = fwrite(buffer, 1, read, dst_file.file()); |
+ if (written != read) |
+ Panic("Wrote %d bytes instead of %d into %s\n", |
+ written, |
+ read, |
+ dst_path.c_str()); |
+ } |
+ if (read < sizeof buffer) |
+ break; |
+ } |
+} |
+ |
+// Send a file descriptor |fd| through |socket|. |
+// Return 0 on success, -1/errno on failure. |
+inline int SendFd(int socket, int fd) { |
+ struct iovec iov; |
+ |
+ char buffer[1]; |
+ buffer[0] = 0; |
+ |
+ iov.iov_base = buffer; |
+ iov.iov_len = 1; |
+ |
+ struct msghdr msg; |
+ struct cmsghdr* cmsg; |
+ char cms[CMSG_SPACE(sizeof(int))]; |
+ |
+ ::memset(&msg, 0, sizeof(msg)); |
+ msg.msg_iov = &iov; |
+ msg.msg_iovlen = 1; |
+ msg.msg_control = reinterpret_cast<caddr_t>(cms); |
+ msg.msg_controllen = CMSG_LEN(sizeof(int)); |
+ |
+ cmsg = CMSG_FIRSTHDR(&msg); |
+ cmsg->cmsg_len = CMSG_LEN(sizeof(int)); |
+ cmsg->cmsg_level = SOL_SOCKET; |
+ cmsg->cmsg_type = SCM_RIGHTS; |
+ ::memcpy(CMSG_DATA(cmsg), &fd, sizeof(int)); |
+ |
+ int ret = sendmsg(socket, &msg, 0); |
+ if (ret < 0) |
+ return -1; |
+ |
+ if (ret != iov.iov_len) { |
+ errno = EIO; |
+ return -1; |
+ } |
+ |
+ return 0; |
+} |
+ |
+inline int ReceiveFd(int socket, int* fd) { |
+ char buffer[1]; |
+ struct iovec iov; |
+ |
+ iov.iov_base = buffer; |
+ iov.iov_len = 1; |
+ |
+ struct msghdr msg; |
+ struct cmsghdr* cmsg; |
+ char cms[CMSG_SPACE(sizeof(int))]; |
+ |
+ ::memset(&msg, 0, sizeof msg); |
+ msg.msg_name = 0; |
+ msg.msg_namelen = 0; |
+ msg.msg_iov = &iov; |
+ msg.msg_iovlen = 1; |
+ |
+ msg.msg_control = reinterpret_cast<caddr_t>(cms); |
+ msg.msg_controllen = sizeof(cms); |
+ |
+ int ret = recvmsg(socket, &msg, 0); |
+ if (ret < 0) |
+ return -1; |
+ if (ret == 0) { |
+ errno = EIO; |
+ return -1; |
+ } |
+ |
+ cmsg = CMSG_FIRSTHDR(&msg); |
+ ::memcpy(fd, CMSG_DATA(cmsg), sizeof(int)); |
+ return 0; |
+} |
+ |
+// Check that there are exactly |expected_count| memory mappings in |
+// /proc/self/maps that point to a RELRO ashmem region. |
+inline void CheckRelroMaps(int expected_count) { |
+ printf("Checking for %d RELROs in /proc/self/maps\n", expected_count); |
+ |
+ FILE* file = fopen("/proc/self/maps", "rb"); |
+ if (!file) |
+ Panic("Could not open /proc/self/maps (pid %d): %s\n", |
+ getpid(), |
+ strerror(errno)); |
+ |
+ char line[512]; |
+ int count_relros = 0; |
+ printf("proc/%d/maps:\n", getpid()); |
+ while (fgets(line, sizeof line, file)) { |
+ if (strstr(line, "with_relro")) { |
+ // The supported library names are "lib<name>_with_relro.so". |
+ printf("%s", line); |
+ if (strstr(line, "/dev/ashmem/RELRO:")) { |
+ count_relros++; |
+ // Check that they are read-only mappings. |
+ if (!strstr(line, " r--")) |
+ Panic("Shared RELRO mapping is not readonly!\n"); |
+ // Check that they can't be remapped read-write. |
+ unsigned vma_start, vma_end; |
+ if (sscanf(line, "%x-%x", &vma_start, &vma_end) != 2) |
+ Panic("Could not parse VM address range!\n"); |
+ int ret = ::mprotect( |
+ (void*)vma_start, vma_end - vma_start, PROT_READ | PROT_WRITE); |
+ if (ret == 0) |
+ Panic("Could remap shared RELRO as writable, should not happen!\n"); |
+ |
+ if (errno != EACCES) |
+ Panic("remapping shared RELRO to writable failed with: %s\n", |
+ strerror(errno)); |
+ } |
+ } |
+ } |
+ fclose(file); |
+ |
+ if (count_relros != expected_count) |
+ Panic( |
+ "Invalid shared RELRO sections in /proc/self/maps: %d" |
+ " (expected %d)\n", |
+ count_relros, |
+ expected_count); |
+ |
+ printf("RELRO count check ok!\n"); |
+} |
+ |
+struct RelroInfo { |
+ size_t start; |
+ size_t size; |
+ int fd; |
+}; |
+ |
+struct RelroLibrary { |
+ const char* name; |
+ crazy_library_t* library; |
+ RelroInfo relro; |
+ |
+ void Init(const char* name, crazy_context_t* context) { |
+ printf("Loading %s\n", name); |
+ this->name = name; |
+ if (!crazy_library_open(&this->library, name, context)) { |
+ Panic("Could not open %s: %s\n", name, crazy_context_get_error(context)); |
+ } |
+ } |
+ |
+ void Close() { crazy_library_close(this->library); } |
+ |
+ void CreateSharedRelro(crazy_context_t* context, size_t load_address) { |
+ if (!crazy_library_create_shared_relro(this->library, |
+ context, |
+ load_address, |
+ &this->relro.start, |
+ &this->relro.size, |
+ &this->relro.fd)) { |
+ Panic("Could not create shared RELRO for %s: %s", |
+ this->name, |
+ crazy_context_get_error(context)); |
+ } |
+ |
+ printf("Parent %s relro info relro_start=%p relro_size=%p relro_fd=%d\n", |
+ this->name, |
+ (void*)this->relro.start, |
+ (void*)this->relro.size, |
+ this->relro.fd); |
+ } |
+ |
+ void EnableSharedRelro(crazy_context_t* context) { |
+ CreateSharedRelro(context, 0); |
+ UseSharedRelro(context); |
+ } |
+ |
+ void SendRelroInfo(int fd) { |
+ if (SendFd(fd, this->relro.fd) < 0) { |
+ Panic("Could not send %s RELRO fd: %s", this->name, strerror(errno)); |
+ } |
+ |
+ int ret = |
+ TEMP_FAILURE_RETRY(::write(fd, &this->relro, sizeof(this->relro))); |
+ if (ret != static_cast<int>(sizeof(this->relro))) { |
+ Panic("Parent could not send %s RELRO info: %s", |
+ this->name, |
+ strerror(errno)); |
+ } |
+ } |
+ |
+ void ReceiveRelroInfo(int fd) { |
+ // Receive relro information from parent. |
+ int relro_fd = -1; |
+ if (ReceiveFd(fd, &relro_fd) < 0) { |
+ Panic("Could not receive %s relro descriptor from parent", this->name); |
+ } |
+ |
+ printf("Child received %s relro fd %d\n", this->name, relro_fd); |
+ |
+ int ret = TEMP_FAILURE_RETRY(::read(fd, &this->relro, sizeof(this->relro))); |
+ if (ret != static_cast<int>(sizeof(this->relro))) { |
+ Panic("Could not receive %s relro information from parent", this->name); |
+ } |
+ |
+ this->relro.fd = relro_fd; |
+ printf("Child received %s relro start=%p size=%p\n", |
+ this->name, |
+ (void*)this->relro.start, |
+ (void*)this->relro.size); |
+ } |
+ |
+ void UseSharedRelro(crazy_context_t* context) { |
+ if (!crazy_library_use_shared_relro(this->library, |
+ context, |
+ this->relro.start, |
+ this->relro.size, |
+ this->relro.fd)) { |
+ Panic("Could not use %s shared RELRO: %s\n", |
+ this->name, |
+ crazy_context_get_error(context)); |
+ } |
+ } |
+}; |
+ |
+} // namespace |
+ |
+#endif // TEST_UTIL_H |