Merge pull request #8054 from hjelmn/kill_the_never_going_to_work_patcher_linux_component_to_prevent_future_confusion_as_to_its_effectiveness
patcher: remove the linux component
Этот коммит содержится в:
Коммит
920315611e
@ -1,49 +0,0 @@
|
|||||||
#
|
|
||||||
# Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
||||||
# University Research and Technology
|
|
||||||
# Corporation. All rights reserved.
|
|
||||||
# Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
||||||
# of Tennessee Research Foundation. All rights
|
|
||||||
# reserved.
|
|
||||||
# Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
||||||
# University of Stuttgart. All rights reserved.
|
|
||||||
# Copyright (c) 2004-2005 The Regents of the University of California.
|
|
||||||
# All rights reserved.
|
|
||||||
# Copyright (c) 2009-2010 Cisco Systems, Inc. All rights reserved.
|
|
||||||
# Copyright (c) 2015 Research Organization for Information Science
|
|
||||||
# and Technology (RIST). All rights reserved.
|
|
||||||
# Copyright (c) 2016 Los Alamos National Security, LLC. All rights
|
|
||||||
# reserved.
|
|
||||||
# Copyright (c) 2017 IBM Corporation. All rights reserved.
|
|
||||||
# $COPYRIGHT$
|
|
||||||
#
|
|
||||||
# Additional copyrights may follow
|
|
||||||
#
|
|
||||||
# $HEADER$
|
|
||||||
#
|
|
||||||
|
|
||||||
if MCA_BUILD_opal_patcher_linux_DSO
|
|
||||||
component_noinst =
|
|
||||||
component_install = mca_patcher_linux.la
|
|
||||||
else
|
|
||||||
component_noinst = libmca_patcher_linux.la
|
|
||||||
component_install =
|
|
||||||
endif
|
|
||||||
|
|
||||||
linux_SOURCES = \
|
|
||||||
patcher_linux.h \
|
|
||||||
patcher_linux_module.c \
|
|
||||||
patcher_linux_component.c
|
|
||||||
|
|
||||||
mcacomponentdir = $(opallibdir)
|
|
||||||
mcacomponent_LTLIBRARIES = $(component_install)
|
|
||||||
mca_patcher_linux_la_SOURCES = $(linux_SOURCES)
|
|
||||||
nodist_mca_patcher_linux_la_SOURCES = $(linux_nodist_SOURCES)
|
|
||||||
mca_patcher_linux_la_LDFLAGS = -module -avoid-version
|
|
||||||
mca_patcher_linux_la_LIBADD = $(top_builddir)/opal/lib@OPAL_LIB_PREFIX@open-pal.la
|
|
||||||
|
|
||||||
noinst_LTLIBRARIES = $(component_noinst)
|
|
||||||
libmca_patcher_linux_la_SOURCES = $(linux_SOURCES)
|
|
||||||
nodist_libmca_patcher_linux_la_SOURCES = $(linux_nodist_SOURCES)
|
|
||||||
libmca_patcher_linux_la_LIBADD = $(patcher_linux_LIBS)
|
|
||||||
libmca_patcher_linux_la_LDFLAGS = -module -avoid-version
|
|
@ -1,57 +0,0 @@
|
|||||||
# -*- shell-script -*-
|
|
||||||
#
|
|
||||||
# Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
|
|
||||||
# University Research and Technology
|
|
||||||
# Corporation. All rights reserved.
|
|
||||||
# Copyright (c) 2004-2005 The University of Tennessee and The University
|
|
||||||
# of Tennessee Research Foundation. All rights
|
|
||||||
# reserved.
|
|
||||||
# Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
|
|
||||||
# University of Stuttgart. All rights reserved.
|
|
||||||
# Copyright (c) 2004-2005 The Regents of the University of California.
|
|
||||||
# All rights reserved.
|
|
||||||
# Copyright (c) 2008-2010 Cisco Systems, Inc. All rights reserved.
|
|
||||||
# Copyright (c) 2015 Research Organization for Information Science
|
|
||||||
# and Technology (RIST). All rights reserved.
|
|
||||||
# Copyright (c) 2016 Los Alamos National Security, LLC. All rights
|
|
||||||
# reserved.
|
|
||||||
# $COPYRIGHT$
|
|
||||||
#
|
|
||||||
# Additional copyrights may follow
|
|
||||||
#
|
|
||||||
# $HEADER$
|
|
||||||
#
|
|
||||||
|
|
||||||
# MCA_patcher_linux_CONFIG(action-if-can-compile,
|
|
||||||
# [action-if-cant-compile])
|
|
||||||
# ------------------------------------------------
|
|
||||||
AC_DEFUN([MCA_opal_patcher_linux_CONFIG],[
|
|
||||||
AC_CONFIG_FILES([opal/mca/patcher/linux/Makefile])
|
|
||||||
|
|
||||||
OPAL_VAR_SCOPE_PUSH([opal_patcher_linux_CPPFLAGS_save])
|
|
||||||
|
|
||||||
opal_patcher_linux_happy=no
|
|
||||||
if test $OPAL_ENABLE_DLOPEN_SUPPORT = 1 ; then
|
|
||||||
# Only enable on Linux for now. In the future this component might
|
|
||||||
# be modified to work on FreeBSD.
|
|
||||||
case $host in
|
|
||||||
*-linux*)
|
|
||||||
opal_patcher_linux_happy=yes;
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
if test $opal_patcher_linux_happy = yes ; then
|
|
||||||
OPAL_CHECK_PACKAGE([patcher_linux], [dlfcn.h], [dl], [dl_iterate_phdr], [], [], [],
|
|
||||||
[],[opal_patcher_linux_happy=no])
|
|
||||||
AC_CHECK_HEADERS([elf.h],[],[opal_patcher_linux_happy=no])
|
|
||||||
AC_CHECK_HEADERS([sys/auxv.h])
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# this component can not be used until we can determine a way to hook munmap, etc inside
|
|
||||||
# glibc. this is needed to catch munmap called by free
|
|
||||||
opal_patcher_linux_happy=no
|
|
||||||
|
|
||||||
AS_IF([test $opal_patcher_linux_happy = yes], [$1], [$2])
|
|
||||||
OPAL_VAR_SCOPE_POP
|
|
||||||
])
|
|
@ -1,45 +0,0 @@
|
|||||||
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */
|
|
||||||
/*
|
|
||||||
* Copyright (c) 2016 Los Alamos National Security, LLC. All rights
|
|
||||||
* reserved.
|
|
||||||
* $COPYRIGHT$
|
|
||||||
*
|
|
||||||
* Additional copyrights may follow
|
|
||||||
*
|
|
||||||
* $HEADER$
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if !defined(OPAL_PATCHER_LINUX_H)
|
|
||||||
#define OPAL_PATCHER_LINUX_H
|
|
||||||
|
|
||||||
#include "opal_config.h"
|
|
||||||
|
|
||||||
#include "opal/mca/patcher/base/base.h"
|
|
||||||
#include "opal/mca/patcher/patcher.h"
|
|
||||||
|
|
||||||
#include "opal/class/opal_list.h"
|
|
||||||
#include "opal/mca/threads/mutex.h"
|
|
||||||
|
|
||||||
struct mca_patcher_linux_patch_got_t {
|
|
||||||
opal_list_item_t super;
|
|
||||||
void **got_entry;
|
|
||||||
void *got_orig;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct mca_patcher_linux_patch_got_t mca_patcher_linux_patch_got_t;
|
|
||||||
|
|
||||||
OBJ_CLASS_DECLARATION(mca_patcher_linux_patch_got_t);
|
|
||||||
|
|
||||||
struct mca_patcher_linux_patch_t {
|
|
||||||
mca_patcher_base_patch_t super;
|
|
||||||
opal_list_t patch_got_list;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef struct mca_patcher_linux_patch_t mca_patcher_linux_patch_t;
|
|
||||||
|
|
||||||
OBJ_CLASS_DECLARATION(mca_patcher_linux_patch_t);
|
|
||||||
|
|
||||||
extern mca_patcher_base_module_t mca_patcher_linux_module;
|
|
||||||
extern mca_patcher_base_component_t mca_patcher_linux_component;
|
|
||||||
|
|
||||||
#endif /* !defined(OPAL_PATCHER_LINUX_H) */
|
|
@ -1,43 +0,0 @@
|
|||||||
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */
|
|
||||||
/*
|
|
||||||
* Copyright (c) 2016 Los Alamos National Security, LLC. All rights
|
|
||||||
* reserved.
|
|
||||||
* $COPYRIGHT$
|
|
||||||
*
|
|
||||||
* Additional copyrights may follow
|
|
||||||
*
|
|
||||||
* $HEADER$
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "patcher_linux.h"
|
|
||||||
|
|
||||||
static int mca_patcher_linux_priority;
|
|
||||||
|
|
||||||
static int mca_patcher_linux_register (void)
|
|
||||||
{
|
|
||||||
mca_patcher_linux_priority = 13;
|
|
||||||
mca_base_component_var_register (&mca_patcher_linux_component.patcherc_version,
|
|
||||||
"priority", "Priority of the linux binary patcher component",
|
|
||||||
MCA_BASE_VAR_TYPE_INT, NULL, 0, 0, OPAL_INFO_LVL_5,
|
|
||||||
MCA_BASE_VAR_SCOPE_CONSTANT, &mca_patcher_linux_priority);
|
|
||||||
|
|
||||||
return OPAL_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_query (mca_base_module_t **module, int *priority)
|
|
||||||
{
|
|
||||||
*module = &mca_patcher_linux_module.super;
|
|
||||||
*priority = mca_patcher_linux_priority;
|
|
||||||
return OPAL_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
mca_patcher_base_component_t mca_patcher_linux_component = {
|
|
||||||
.patcherc_version = {
|
|
||||||
OPAL_PATCHER_BASE_VERSION_1_0_0,
|
|
||||||
.mca_component_name = "linux",
|
|
||||||
MCA_BASE_MAKE_VERSION(component, OPAL_MAJOR_VERSION, OPAL_MINOR_VERSION,
|
|
||||||
OPAL_RELEASE_VERSION),
|
|
||||||
.mca_query_component = mca_patcher_linux_query,
|
|
||||||
.mca_register_component_params = mca_patcher_linux_register,
|
|
||||||
},
|
|
||||||
};
|
|
@ -1,397 +0,0 @@
|
|||||||
/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil -*- */
|
|
||||||
/*
|
|
||||||
* Copyright (C) Mellanox Technologies Ltd. 2001-2015. ALL RIGHTS RESERVED.
|
|
||||||
* Copyright (c) 2016 Los Alamos National Security, LLC. All rights
|
|
||||||
* reserved.
|
|
||||||
* $COPYRIGHT$
|
|
||||||
*
|
|
||||||
* Additional copyrights may follow
|
|
||||||
*
|
|
||||||
* $HEADER$
|
|
||||||
*/
|
|
||||||
/*
|
|
||||||
* Copied from OpenUCX
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "patcher_linux.h"
|
|
||||||
|
|
||||||
#include "opal/mca/patcher/base/base.h"
|
|
||||||
|
|
||||||
#include "opal/constants.h"
|
|
||||||
#include "opal/util/sys_limits.h"
|
|
||||||
#include "opal/util/output.h"
|
|
||||||
#include "opal/prefetch.h"
|
|
||||||
|
|
||||||
#if defined(HAVE_SYS_AUXV_H)
|
|
||||||
#include <sys/auxv.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <elf.h>
|
|
||||||
|
|
||||||
#include <sys/mman.h>
|
|
||||||
#include <pthread.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <dlfcn.h>
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <link.h>
|
|
||||||
|
|
||||||
static void *mca_patcher_linux_dlopen(const char *filename, int flag);
|
|
||||||
|
|
||||||
typedef struct mca_patcher_linux_dl_iter_context {
|
|
||||||
mca_patcher_linux_patch_t *patch;
|
|
||||||
bool remove;
|
|
||||||
int status;
|
|
||||||
} mca_patcher_linux_dl_iter_context_t;
|
|
||||||
|
|
||||||
OBJ_CLASS_INSTANCE(mca_patcher_linux_patch_got_t, opal_list_item_t, NULL, NULL);
|
|
||||||
|
|
||||||
static void mca_patcher_linux_patch_construct (mca_patcher_linux_patch_t *patch)
|
|
||||||
{
|
|
||||||
OBJ_CONSTRUCT(&patch->patch_got_list, opal_list_t);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void mca_patcher_linux_patch_destruct (mca_patcher_linux_patch_t *patch)
|
|
||||||
{
|
|
||||||
OPAL_LIST_DESTRUCT(&patch->patch_got_list);
|
|
||||||
}
|
|
||||||
|
|
||||||
OBJ_CLASS_INSTANCE(mca_patcher_linux_patch_t, mca_patcher_base_patch_t, mca_patcher_linux_patch_construct,
|
|
||||||
mca_patcher_linux_patch_destruct);
|
|
||||||
|
|
||||||
/* List of patches to be applied to additional libraries */
|
|
||||||
static void *(*orig_dlopen) (const char *, int);
|
|
||||||
|
|
||||||
static const ElfW(Phdr) *
|
|
||||||
mca_patcher_linux_get_phdr_dynamic(const ElfW(Phdr) *phdr, uint16_t phnum, int phent)
|
|
||||||
{
|
|
||||||
for (uint16_t i = 0 ; i < phnum ; ++i, phdr = (ElfW(Phdr)*)((intptr_t) phdr + phent)) {
|
|
||||||
if (phdr->p_type == PT_DYNAMIC) {
|
|
||||||
return phdr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void *mca_patcher_linux_get_dynentry(ElfW(Addr) base, const ElfW(Phdr) *pdyn, ElfW(Sxword) type)
|
|
||||||
{
|
|
||||||
for (ElfW(Dyn) *dyn = (ElfW(Dyn)*)(base + pdyn->p_vaddr); dyn->d_tag; ++dyn) {
|
|
||||||
if (dyn->d_tag == type) {
|
|
||||||
return (void *) (uintptr_t) dyn->d_un.d_val;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void * mca_patcher_linux_get_got_entry (ElfW(Addr) base, const ElfW(Phdr) *phdr, int16_t phnum,
|
|
||||||
int phent, const char *symbol)
|
|
||||||
{
|
|
||||||
const ElfW(Phdr) *dphdr;
|
|
||||||
void *jmprel, *strtab;
|
|
||||||
ElfW(Sym) *symtab;
|
|
||||||
size_t pltrelsz;
|
|
||||||
|
|
||||||
dphdr = mca_patcher_linux_get_phdr_dynamic (phdr, phnum, phent);
|
|
||||||
|
|
||||||
jmprel = mca_patcher_linux_get_dynentry (base, dphdr, DT_JMPREL);
|
|
||||||
symtab = (ElfW(Sym) *) mca_patcher_linux_get_dynentry (base, dphdr, DT_SYMTAB);
|
|
||||||
strtab = mca_patcher_linux_get_dynentry (base, dphdr, DT_STRTAB);
|
|
||||||
pltrelsz = (size_t) (uintptr_t) mca_patcher_linux_get_dynentry (base, dphdr, DT_PLTRELSZ);
|
|
||||||
|
|
||||||
for (ElfW(Rela) *reloc = jmprel; (intptr_t) reloc < (intptr_t) jmprel + pltrelsz; ++reloc) {
|
|
||||||
#if SIZEOF_VOID_P == 8
|
|
||||||
uint32_t relsymidx = ELF64_R_SYM(reloc->r_info);
|
|
||||||
#else
|
|
||||||
uint32_t relsymidx = ELF32_R_SYM(reloc->r_info);
|
|
||||||
#endif
|
|
||||||
char *elf_sym = (char *) strtab + symtab[relsymidx].st_name;
|
|
||||||
|
|
||||||
if (0 == strcmp (symbol, elf_sym)) {
|
|
||||||
return (void *)(base + reloc->r_offset);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_get_aux_phent (void)
|
|
||||||
{
|
|
||||||
#if !defined(HAVE_SYS_AUXV_H)
|
|
||||||
#define MCA_PATCHER_LINUX_AUXV_BUF_LEN 16
|
|
||||||
static const char *proc_auxv_filename = "/proc/self/auxv";
|
|
||||||
static int phent = 0;
|
|
||||||
ElfW(auxv_t) buffer[MCA_PATCHER_LINUX_AUXV_BUF_LEN];
|
|
||||||
unsigned count;
|
|
||||||
ssize_t nread;
|
|
||||||
int fd;
|
|
||||||
|
|
||||||
/* Can avoid lock here - worst case we'll read the file more than once */
|
|
||||||
if (phent == 0) {
|
|
||||||
fd = open(proc_auxv_filename, O_RDONLY);
|
|
||||||
if (fd < 0) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"failed to open '%s' for reading: %s", proc_auxv_filename,
|
|
||||||
strerror (errno));
|
|
||||||
return OPAL_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Use small buffer on the stack, avoid using malloc() */
|
|
||||||
do {
|
|
||||||
nread = read(fd, buffer, sizeof(buffer));
|
|
||||||
if (nread < 0) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"failed to read %" PRIsize_t " bytes from %s (ret=%ld): %s", sizeof (buffer),
|
|
||||||
proc_auxv_filename, nread, strerror (errno));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
count = nread / sizeof(buffer[0]);
|
|
||||||
for (unsigned i = 0 ; i < count && AT_NULL != buffer[i].a_type ; ++i) {
|
|
||||||
if (AT_PHENT == buffer[i].a_type) {
|
|
||||||
phent = buffer[i].a_un.a_val;
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"read phent from %s: %d", proc_auxv_filename, phent);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while ((count > 0) && (phent == 0));
|
|
||||||
|
|
||||||
close(fd);
|
|
||||||
}
|
|
||||||
|
|
||||||
return phent;
|
|
||||||
#else
|
|
||||||
return getauxval (AT_PHENT);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
mca_patcher_linux_modify_got (ElfW(Addr) base, const ElfW(Phdr) *phdr, const char *phname,
|
|
||||||
int16_t phnum, int phent, mca_patcher_linux_dl_iter_context_t *ctx)
|
|
||||||
{
|
|
||||||
long page_size = opal_getpagesize ();
|
|
||||||
void **entry, *page;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
entry = mca_patcher_linux_get_got_entry (base, phdr, phnum, phent, ctx->patch->super.patch_symbol);
|
|
||||||
if (entry == NULL) {
|
|
||||||
return OPAL_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
page = (void *)((intptr_t)entry & ~(page_size - 1));
|
|
||||||
ret = mprotect(page, page_size, PROT_READ|PROT_WRITE);
|
|
||||||
if (ret < 0) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"failed to modify GOT page %p to rw: %s", page, strerror (errno));
|
|
||||||
return OPAL_ERR_NOT_SUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!ctx->remove) {
|
|
||||||
if (*entry != (void *) ctx->patch->super.patch_value) {
|
|
||||||
mca_patcher_linux_patch_got_t *patch_got = OBJ_NEW(mca_patcher_linux_patch_got_t);
|
|
||||||
if (NULL == patch_got) {
|
|
||||||
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
||||||
}
|
|
||||||
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_TRACE, opal_patcher_base_framework.framework_output,
|
|
||||||
"patch %p (%s): modifying got entry %p. original value %p. new value %p\n", (void *)ctx->patch,
|
|
||||||
ctx->patch->super.patch_symbol, (void *) entry, *entry, (void *) ctx->patch->super.patch_value);
|
|
||||||
|
|
||||||
patch_got->got_entry = entry;
|
|
||||||
patch_got->got_orig = *entry;
|
|
||||||
|
|
||||||
opal_list_append (&ctx->patch->patch_got_list, &patch_got->super);
|
|
||||||
|
|
||||||
*entry = (void *) ctx->patch->super.patch_value;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/* find the appropriate entry and restore the original value */
|
|
||||||
mca_patcher_linux_patch_got_t *patch_got;
|
|
||||||
OPAL_LIST_FOREACH_REV(patch_got, &ctx->patch->patch_got_list, mca_patcher_linux_patch_got_t) {
|
|
||||||
if (patch_got->got_entry == entry) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_TRACE, opal_patcher_base_framework.framework_output,
|
|
||||||
"restoring got entry %p with original value %p\n", (void *) entry, patch_got->got_orig);
|
|
||||||
if (*entry == (void *) ctx->patch->super.patch_value) {
|
|
||||||
*entry = patch_got->got_orig;
|
|
||||||
}
|
|
||||||
opal_list_remove_item (&ctx->patch->patch_got_list, &patch_got->super);
|
|
||||||
OBJ_RELEASE(patch_got);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return OPAL_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_phdr_iterator(struct dl_phdr_info *info, size_t size, void *data)
|
|
||||||
{
|
|
||||||
mca_patcher_linux_dl_iter_context_t *ctx = data;
|
|
||||||
int phent;
|
|
||||||
|
|
||||||
phent = mca_patcher_linux_get_aux_phent();
|
|
||||||
if (phent <= 0) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"failed to read phent size");
|
|
||||||
ctx->status = OPAL_ERR_NOT_SUPPORTED;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx->status = mca_patcher_linux_modify_got (info->dlpi_addr, info->dlpi_phdr,
|
|
||||||
info->dlpi_name, info->dlpi_phnum,
|
|
||||||
phent, ctx);
|
|
||||||
if (ctx->status == OPAL_SUCCESS) {
|
|
||||||
return 0; /* continue iteration and patch all objects */
|
|
||||||
} else {
|
|
||||||
return -1; /* stop iteration if got a real error */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* called with lock held */
|
|
||||||
static int mca_patcher_linux_apply_patch (mca_patcher_linux_patch_t *patch)
|
|
||||||
{
|
|
||||||
mca_patcher_linux_dl_iter_context_t ctx = {
|
|
||||||
.patch = patch,
|
|
||||||
.remove = false,
|
|
||||||
.status = OPAL_SUCCESS,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Avoid locks here because we don't modify ELF data structures.
|
|
||||||
* Worst case the same symbol will be written more than once.
|
|
||||||
*/
|
|
||||||
(void) dl_iterate_phdr(mca_patcher_linux_phdr_iterator, &ctx);
|
|
||||||
|
|
||||||
return ctx.status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_remove_patch (mca_patcher_linux_patch_t *patch)
|
|
||||||
{
|
|
||||||
mca_patcher_linux_dl_iter_context_t ctx = {
|
|
||||||
.patch = patch,
|
|
||||||
.remove = true,
|
|
||||||
.status = OPAL_SUCCESS,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Avoid locks here because we don't modify ELF data structures.
|
|
||||||
* Worst case the same symbol will be written more than once.
|
|
||||||
*/
|
|
||||||
(void) dl_iterate_phdr(mca_patcher_linux_phdr_iterator, &ctx);
|
|
||||||
|
|
||||||
return ctx.status;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void *mca_patcher_linux_dlopen(const char *filename, int flag)
|
|
||||||
{
|
|
||||||
OPAL_PATCHER_BEGIN;
|
|
||||||
mca_patcher_linux_patch_t *patch;
|
|
||||||
void *handle;
|
|
||||||
|
|
||||||
handle = orig_dlopen (filename, flag);
|
|
||||||
if (handle != NULL) {
|
|
||||||
/*
|
|
||||||
* Every time a new object is loaded, we must update its relocations
|
|
||||||
* with our list of patches (including dlopen itself). This code is less
|
|
||||||
* efficient and will modify all existing objects every time, but good
|
|
||||||
* enough.
|
|
||||||
*/
|
|
||||||
opal_mutex_lock (&mca_patcher_linux_module.patch_list_mutex);
|
|
||||||
OPAL_LIST_FOREACH(patch, &mca_patcher_linux_module.patch_list, mca_patcher_linux_patch_t) {
|
|
||||||
if (!patch->super.patch_data_size) {
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_INFO, opal_patcher_base_framework.framework_output,
|
|
||||||
"in dlopen(), re-applying '%s' to %p", patch->super.patch_symbol, (void *) patch->super.patch_value);
|
|
||||||
/* ignore hook binary patches */
|
|
||||||
mca_patcher_linux_apply_patch (patch);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
opal_mutex_unlock (&mca_patcher_linux_module.patch_list_mutex);
|
|
||||||
}
|
|
||||||
|
|
||||||
OPAL_PATCHER_END;
|
|
||||||
return handle;
|
|
||||||
}
|
|
||||||
|
|
||||||
static intptr_t mca_patcher_linux_get_orig (const char *symbol, void *replacement)
|
|
||||||
{
|
|
||||||
const char *error;
|
|
||||||
void *func_ptr;
|
|
||||||
|
|
||||||
func_ptr = dlsym(RTLD_DEFAULT, symbol);
|
|
||||||
if (func_ptr == replacement) {
|
|
||||||
(void)dlerror();
|
|
||||||
func_ptr = dlsym(RTLD_NEXT, symbol);
|
|
||||||
if (func_ptr == NULL) {
|
|
||||||
error = dlerror();
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_ERROR, opal_patcher_base_framework.framework_output,
|
|
||||||
"could not find address of original %s(): %s", symbol, error ? error : "Unknown error");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
opal_output_verbose (MCA_BASE_VERBOSE_INFO, opal_patcher_base_framework.framework_output,
|
|
||||||
"original %s() is at %p", symbol, func_ptr);
|
|
||||||
|
|
||||||
return (intptr_t) func_ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_patch_symbol (const char *symbol_name, uintptr_t replacement, uintptr_t *orig)
|
|
||||||
{
|
|
||||||
mca_patcher_linux_patch_t *patch = OBJ_NEW(mca_patcher_linux_patch_t);
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
if (OPAL_UNLIKELY(NULL == patch)) {
|
|
||||||
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
||||||
}
|
|
||||||
|
|
||||||
patch->super.patch_symbol = strdup (symbol_name);
|
|
||||||
if (NULL == patch->super.patch_symbol) {
|
|
||||||
OBJ_RELEASE(patch);
|
|
||||||
return OPAL_ERR_OUT_OF_RESOURCE;
|
|
||||||
}
|
|
||||||
|
|
||||||
patch->super.patch_value = mca_patcher_base_addr_text (replacement);
|
|
||||||
patch->super.patch_restore = (mca_patcher_base_restore_fn_t) mca_patcher_linux_remove_patch;
|
|
||||||
|
|
||||||
/* Take lock first to handle a possible race where dlopen() is called
|
|
||||||
* from another thread and we may end up not patching it.
|
|
||||||
*/
|
|
||||||
opal_mutex_lock (&mca_patcher_linux_module.patch_list_mutex);
|
|
||||||
do {
|
|
||||||
rc = mca_patcher_base_patch_hook (&mca_patcher_linux_module, patch->super.patch_value);
|
|
||||||
if (OPAL_SUCCESS != rc) {
|
|
||||||
OBJ_RELEASE(patch);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = mca_patcher_linux_apply_patch (patch);
|
|
||||||
if (OPAL_SUCCESS != rc) {
|
|
||||||
OBJ_RELEASE(patch);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
*orig = mca_patcher_linux_get_orig (patch->super.patch_symbol, (void *) replacement);
|
|
||||||
|
|
||||||
opal_list_append (&mca_patcher_linux_module.patch_list, &patch->super.super);
|
|
||||||
} while (0);
|
|
||||||
opal_mutex_unlock (&mca_patcher_linux_module.patch_list_mutex);
|
|
||||||
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* called with lock held */
|
|
||||||
static int mca_patcher_linux_install_dlopen (void)
|
|
||||||
{
|
|
||||||
return mca_patcher_linux_patch_symbol ("dlopen", (uintptr_t) mca_patcher_linux_dlopen,
|
|
||||||
(uintptr_t *) &orig_dlopen);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int mca_patcher_linux_init (void)
|
|
||||||
{
|
|
||||||
return mca_patcher_linux_install_dlopen ();
|
|
||||||
}
|
|
||||||
|
|
||||||
mca_patcher_base_module_t mca_patcher_linux_module = {
|
|
||||||
.patch_init = mca_patcher_linux_init,
|
|
||||||
.patch_symbol = mca_patcher_linux_patch_symbol,
|
|
||||||
};
|
|
Загрузка…
x
Ссылка в новой задаче
Block a user