Cisco Blogs


Cisco Blog > Security

Injecting the Python Interpreter Via GDB

Recently I was evaluating the security of an application sandbox and I needed a way to inject some kind of interface into the sandboxed application in order to explore the possibilities available from that context. The main objective was to be able to easily explore file and system call access to determine what was allowed/denied. I decided the most suitable interface I could use for this exploration would be the Python interactive shell.

The first step I needed to take was to get the Python library (libpython) loaded into the address space of the target application. The easiest way that I could think to do this was to utilize the call command in the Gnu Debugger (GDB). GDB’s call command performs a debugee procedure call by injecting a new thread into the debugee and controlling the startup state. Since GDB already performs the necessary steps, I could take advantage of this by issuing the command:

call (int)dlopen("/usr/lib/libpython2.7.dylib")

This causes GDB to create a new thread and call the dynamic linker function dlopen() to load a shared library. Because I was testing this on Mac OS X, I specified the default path to the libpython dynamic library. On other UNIX based platforms this path would be to an .so library. On Windows libpython is a DLL .

This same functionality can be achieved on Windows using the LoadLibrary() function instead of dlopen(). Windbg also supports debugee procedure calls with the .call command. Obviously for the call to dlopen() to succeed, the platform you are using must have debug symbols for the libraries you are calling into. Otherwise, GDB does not know what address the function is at. Depending on the situation, it might be useful to use the dynamic linkers configuration variables to force the load of a debug build of the necessary libraries.

Now that I had the Python interpreter embedded into the address space of the target application, the next step was to get my own Python code executing. Again this could be easily achieved from GDB using the call command. When you are embedding Python in any application the first step is to call Py_Initialize(). The Py_Initialize() function creates the building modules in the Python namespace, as well as setting up the appropriate module search paths. The call to this function from GDB looks exactly like you would expect:

call (int)Py_Initialize()

With the Python namespace established, the only thing left to do is to execute a Python string. One easy method of doing this is to use the PyRun_SimpleString() function. This function takes a character string argument and compiles and executes it as Python code.

As you can see from the example below, we can verify that PyRun_SimpleString() has been successful by calling os.getpid() in python and comparing it with the process ID (pid) from a bash shell debugee ($$).

[nearchib@needle:~]$ gdb /bin/bash
GNU gdb 6.3.50-20050815 (Apple version gdb-1820) (Sat Jun 16 02:40:11 UTC 2012)
Copyright 2004 Free Software Foundation, Inc.

 (gdb) r
Starting program: /bin/bash
-[nearchib@needle:~]$
Program received signal SIGINT, Interrupt.
0x00007fff9189fffa in read ()
(gdb) call (int)dlopen("/usr/lib/libpython2.7.dylib")
Reading symbols for shared libraries . done
$1 = 1123568
(gdb) call (int)Py_Initialize()
$2 = 1985151592
 (gdb) call (int)PyRun_SimpleString("import os\nprint os.getpid()")
39776
$3 = 0
(gdb) c
Continuing.
5
-[nearchib@needle:~]$ echo $$
39776

While this worked great for a console-based application where we are directly in control of stdin/stdout, it’s less than ideal for a GUI-based application or a server. I decided the solution was to create a Python bindshell similar to what would be used in an exploitation scenario.

For the most part the code was fairly straightforward, although there were a few ugly hacks involved. I have included the entire code listing in the appendix to this post below; however, I will briefly run through how I went about this.

Basically, I used some pretty standard socket server code that utilizes the socket library. Calling bind()/listen() and accept() to bind a socket to a port, listen for a client to connect, and generate a new handle for the newly connected client.

Once the socket was set up, the next step was to create an instance of the code module and utilize the InteractiveInterpreter method to provide a nice Python shell. Typical invocation of this method will simply read from stdin and write to stdout; however, a different approach was used since I needed to bind this to a socket. The InteractiveInterpreter method accepts an argument readfunc, which allows us to specify a function for performing a read of the user input rather than just using raw_input (the default). I then created a function that reads from the socket (after converting it to a file using the makefile() method) using the readline() method. Doing this caused my Python bindshell to half work. Input was read from the socket; however, the output from the executing code would still go to the client. To work around this I needed to redirect writes to stdout through the socket. Because my interpreter was in the address space of another program that I was inspecting, I did not want to just dup2 stdin, as doing this would redirect the debugee’s output through the socket too. To work around this I had to modify the stdin object in the python namespace to write to the socket. Unfortunately this was not a simple case of swapping the stdout object with the socket object because the socket object does not have a write() method, which is what is used by the code module to perform output functionality. At first I tried using the filehandle that I created with the makefile() method of the socket; however, this did not end up sending the output to the client. To combat this I created my own class (wsocket) that wrapped the socket object and provided a write() method that passed the data through to the send() method of the socket. I then replaced the stdout object in the Python namespace with the wsocket instance. This caused the bindshell to work as expected, as you can see below (using the netcat (nc) command to connect to the shell on port 55555):

-[nearchib@needle:~/code/pyject]$ ./pybindshell.py 

-[nearchib@needle:~]$ nc localhost 55555
Python 2.7.2 (default, Jun 20 2012, 16:23:33)
[GCC 4.2.1 Compatible Apple Clang 4.0 (tags/Apple/clang-418.0.60)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
>>> a = 1
>>> a
1
>>>

Now that I had a working Python bindshell, I needed to inject it using the GDB commands from earlier. I decided that, rather than implementing a whole series of calls to the Python framework, I would simply encode the entire program into a single line and use PyExc_SimpleString() to execute it.

I made a tiny Python program using the base64 module, which read the whole file in, and b64 encoded it (shown below):

import base64
print base64.b64encode(open("raw","r").read())

In order to decode and execute this at runtime, I needed to use the following code.

"import code\nimport base64\ncode.InteractiveInterpreter().runcode(base64.b64decode('”

This code simply takes the base64 encoded output, decodes it back to ascii text, and passes it to the runcode() method of the InteractiveInterpreter to be executed. With this in mind the final command that I sent to GDB is:

call (int)PyRun_SimpleString("import code\nimport base64\ncode.InteractiveInterpreter().runcode(base64.b64decode(
'aW1wb3J0IHN5cwppbXBvcnQgY29kZQppbXBvcnQgdGhyZWFkCmltcG9ydCBzb2NrZXQgIAppbXBvcnQgb3MgIAoKZ2xvYmFscygpWydzb2NrZmQnX
SA9ICIiCmdsb2JhbHMoKVsnc29jayddID0gIiIKCmRlZiByZWFkX2NtZCh2YXIpOgoJc29ja2ZkID0gZ2xvYmFscygpWydzb2NrZmQnXSAKCWdsb2J
hbHMoKVsnc29jayddLnNlbmQodmFyKSAKCXJldHVybiBzb2NrZmQucmVhZGxpbmUoKQpkZWYgd3JpdGVfY21kKHNlbGYsIGRhdGEpOgoJZ2xvYmFsc
ygpWydzb2NrJ10uc2VuZChkYXRhKQpjbGFzcyB3c29jaygpOgoJZGVmIF9faW5pdF9fKHNlbGYsIHNvY2spOgoJCXNlbGYuX19zb2NrID0gc29jawo
JZGVmIHdyaXRlKHNlbGYsIHRleHQpOgoJCXJldHVybiBzZWxmLl9fc29jay5zZW5kKHRleHQpCmRlZiBiaW5kX2lhcHMoKToKCXBvcnQgPSA1NTU1N
QoJc29ja2ZkID0gc29ja2V0LnNvY2tldCgpCglzb2NrZmQuc2V0c29ja29wdChzb2NrZXQuU09MX1NPQ0tFVCAsIHNvY2tldC5TT19SRVVTRUFERFI
gLCAxKQoJc29ja2ZkLmJpbmQoKCIiLHBvcnQpKSAgCglzb2NrZmQubGlzdGVuKDEpICAKCWNzICwgY2FkZHIgPSBzb2NrZmQuYWNjZXB0KCkgIAoJC
glnbG9iYWxzKClbJ3NvY2tmZCddID0gY3MubWFrZWZpbGUoKQoJZ2xvYmFscygpWydzb2NrJ10gPSBjcwoJc3lzLnN0ZG91dCA9IHdzb2NrKGNzKSA
KCWNvZGUuSW50ZXJhY3RpdmVJbnRlcnByZXRlci53cml0ZSA9IHdyaXRlX2NtZAoJY29kZS5pbnRlcmFjdChsb2NhbD1sb2NhbHMoKSwgcmVhZGZ1b
mM9cmVhZF9jbWQpCnRocmVhZC5zdGFydF9uZXcoYmluZF9pYXBzLCgpKQppbXBvcnQgdGltZQp3aGlsZShUcnVlKToKCXRpbWUuc2xlZXAoNSkK'))")

This worked perfectly and allowed me to easily inject the interpreter; however, it would leave the GDB session blocking and unusable. I decided it would be more useful to have a working GDB session, which would make it possible to explore the address space while using ctypes from the Python shell to create API stubs for executing functionality in the debugee. The problem though, was that if I pressed ^C (control C) to interrupt the execution of the call command, GDB shut down the Python code that was executing. To get around this, I created a thread inside my Python bindshell, and executed the InteractiveInterpreter inside it.

I then issued the following command in GDB:

set unwindonsignal

Set unwinding of the stack if a signal is received while in a function that gdb called in the program being debugged. If set to on, gdb unwinds the stack it created for the call and restores the context to what it was before the call. If set to off (the default), gdb stops in the frame where the signal was received.

The result of this command was that when I interrupted the running command in GDB, the stack unwound and GDB continued as normal. However, because the bindshell was now running in its own thread, it resumed execution when the continue command was given.

At this stage I thought that I was done. However, when I injected my bindshell into the target app I received an error loading modules such as base64 or ctypes from the lib-dynload/ directory. After much searching I discovered an obscure bug report on the Python site, http://bugs.python.org/issue4434. The poster notes that he was provided with a work around:

“I have been given the following workaround: in mylib.c, before PyInitialize() I can call dlopen("libpython2.5.so", RTLD_LAZY | RTLD_GLOBAL);”

After reading this I modified my dlopen call to include the flag value: 9.
This is the result of the operation RTLD_LAZY | RTLD_GLOBAL. This left me with the following:

call (int)dlopen("/usr/lib/libpython.dylib",9)

After I made this modification it solved the above problem and the code worked fine.

To make it more convenient for me to run the code again, I created a GDB command to house the finished statements. The syntax for this is pretty simple. I created the pyject command, which takes no arguments and will simply run the commands we discussed thus far. Running the command results in the Python interpreter being injected and executed, and our Python bindshell binding to port 55555.

There are many uses for this functionality, the sandbox example is just one of them. As I mentioned previously, it’s possible to craft ctypes stubs from Python to call into the APIs of the debugged process. This can be very useful as confirmation of reverse engineering practice, as well as providing a strong basis for exploration of unknown software. There have been several published tools for injecting Python into a process in the past; the reason I worked on this, however, is that it doesn’t require an additional tool to inject and this technique works on multiple platforms.

To utilize the gdbinit, on a unix platform, copy it into your home directory as .gdbinit.

Appendix (Code):

.gdbinit:

define pyject
        echo [+] pyject - Injecting a python interpreter into the debugee.\n
        echo \n
        echo            press ^C to return to gdb...\n
        set unwindonsignal on
#RTLD_LAZY | RTLD_GLOBAL == 9
        call (int)dlopen("/usr/lib/libpython.dylib",9)
        call (int)Py_Initialize()
        call (int)PyRun_SimpleString("import code\nimport base64\ncode.InteractiveInterpreter().runcode(base64.b64decode('aW1wb3J0IHN5cwppbXBvcnQgY29
kZQppbXBvcnQgdGhyZWFkCmltcG9ydCBzb2NrZXQgIAppbXBvcnQgb3MgIAoKZ2xvYmFscygpWydzb2NrZmQnXSA9ICIiCmdsb2JhbHMoKVsnc29jayddID0gIiIKCmRlZiByZWFkX2NtZCh2YXIp
OgoJc29ja2ZkID0gZ2xvYmFscygpWydzb2NrZmQnXSAKCWdsb2JhbHMoKVsnc29jayddLnNlbmQodmFyKSAKCXJldHVybiBzb2NrZmQucmVhZGxpbmUoKQpkZWYgd3JpdGVfY21kKHNlbGYsIGRhd
GEpOgoJZ2xvYmFscygpWydzb2NrJ10uc2VuZChkYXRhKQpjbGFzcyB3c29jaygpOgoJZGVmIF9faW5pdF9fKHNlbGYsIHNvY2spOgoJCXNlbGYuX19zb2NrID0gc29jawoJZGVmIHdyaXRlKHNlbG
YsIHRleHQpOgoJCXJldHVybiBzZWxmLl9fc29jay5zZW5kKHRleHQpCmRlZiBiaW5kX2lhcHMoKToKCXBvcnQgPSA1NTU1NQoJc29ja2ZkID0gc29ja2V0LnNvY2tldCgpCglzb2NrZmQuc2V0c29
ja29wdChzb2NrZXQuU09MX1NPQ0tFVCAsIHNvY2tldC5TT19SRVVTRUFERFIgLCAxKQoJc29ja2ZkLmJpbmQoKCIiLHBvcnQpKSAgCglzb2NrZmQubGlzdGVuKDEpICAKCWNzICwgY2FkZHIgPSBz
b2NrZmQuYWNjZXB0KCkgIAoJCglnbG9iYWxzKClbJ3NvY2tmZCddID0gY3MubWFrZWZpbGUoKQoJZ2xvYmFscygpWydzb2NrJ10gPSBjcwoJc3lzLnN0ZG91dCA9IHdzb2NrKGNzKSAKCWNvZGUuS
W50ZXJhY3RpdmVJbnRlcnByZXRlci53cml0ZSA9IHdyaXRlX2NtZAoJY29kZS5pbnRlcmFjdChsb2NhbD1sb2NhbHMoKSwgcmVhZGZ1bmM9cmVhZF9jbWQpCnRocmVhZC5zdGFydF9uZXcoYmluZF
9pYXBzLCgpKQppbXBvcnQgdGltZQp3aGlsZShUcnVlKToKCXRpbWUuc2xlZXAoNSkK'))")
end
document pyject
pyject -- Inject the python interpreter into the running process. Binds an interactive session to TCP port 55555
end

pybindshell.py:

#!/usr/bin/python

import sys
import code
import thread
import socket
import os  

globals()['sockfd'] = ""
globals()['sock'] = ""

def read_cmd(var):
	sockfd = globals()['sockfd']
	globals()['sock'].send(var) # prompt
	return sockfd.readline()
# end read_cmd

def write_cmd(self, data):
#	print "writing %s\n" % data;
	globals()['sock'].send(data)
# end write_cmd

class wsock():
	def __init__(self, sock):
		self.__sock = sock
	# end __init__

	def write(self, text):
		return self.__sock.send(text)
	# end write
# end class

def bind_iaps():
	port = 55555
	sockfd = socket.socket()
	sockfd.setsockopt(socket.SOL_SOCKET , socket.SO_REUSEADDR , 1)
	sockfd.bind(("",port))
	sockfd.listen(1)
	# got connection
	cs , caddr = sockfd.accept()  

	globals()['sockfd'] = cs.makefile()
	globals()['sock'] = cs
	sys.stdout = wsock(cs) # hack much?
	code.InteractiveInterpreter.write = write_cmd
	code.interact(local=locals(), readfunc=read_cmd)
# end bind_iaps

thread.start_new(bind_iaps,())
import time
while(True):
	time.sleep(5)

Tags: , ,

In an effort to keep conversations fresh, Cisco Blogs closes comments after 60 days. Please visit the Cisco Blogs hub page for the latest content.

2 Comments.


  1. Thank you, that was an excellent article.

       0 likes

  2. October 18, 2012 at 7:23 am

    ...An excellent approach...thanks for sharing it, greatly appreciated...regards from PR.

       0 likes