Monkey Patching in Go
Many people think that monkey patching is something that is restricted to dynamic languages like Ruby and Python. That is not true however, as computers are just dumb machines and we can always make them do what we want! Let’s look at how Go functions work and how we can modify them at runtime. This article will use a lot of Intel assembly syntax, so I’m assuming you can read it already or are using a reference while reading.
If you’re not interested in how it works and you just want to do monkey patching, then you can find the library here.
Let’s look at what the following code produces when disassembled:
Samples should be built with
go build -gcflags=-l to disable inlining. For this article I assume your architecture is 64-bits and that you’re using a unix-based operating system like Mac OSX or a Linux variant.
When compiled and looked at through Hopper, the above code will produce this assembly code:
I will be referring to the addresses of the various instructions displayed on the left side of the screen.
Our code starts in procedure
main.main, where instructions
0x2026 set up the stack. You can read more about that here, I will be ignoring that code for the rest of the article.
0x202a is the call to function
main.a at line
0x2000 which simply moves
0x1 onto the stack and returns. Lines
0x2037 then pass that value on to
Simple enough! Now let’s take a look at how function values are implemented in Go.
How function values work in Go
Consider the following code:
What I’m doing on line 11 is assigning
f, which means that doing
f() will now call
a. Then I use the unsafe Go package to directly read out the value stored in
f. If you come from a C background you might expect
f to simply be a function pointer to
a and thus this code to print out
0x2000 (the location of
main.a as we saw above). When I run this on my machine I get
0x102c38, which is an address not even close to our code! When disassembled, this is what happens on line 11 above:
This references something called
main.a.f, and when we look at that location, we see this:
main.a.f is at
0x102c38 and contains
0x2000, which is the location of
main.a. It seems
f isn’t a pointer to a function, but a pointer to a pointer to a function. Let’s modify the code to compensate for that.
This will now print
0x2000, as expected. We can find a clue as to why this is implemented as it is here. Go function values can contain extra information, which is how closures and bound instance methods are implemented.
Let’s look at how calling a function value works. I’ll change the code to call
f after assigning it.
When we disassemble this we get the following:
main.a.f gets loaded into
rdx, then whatever
rdx points at gets loaded into
rbx, which then gets called. The address of the function value always gets loaded into
rdx, which the code being called can use to load any extra information it might need. This extra information is a pointer to the instance for a bound instance method and the closure for an anonymous function. I advise you to take out a disassembler and dive deeper if you want to know more!
Let’s use our newly gained knowledge to implement monkeypatching in Go.
Replacing a function at runtime
What we want to achieve is to have the following code print out
Now how do we implement
replace? We need to modify function
a to jump to
b’s code instead of executing its own body. Essentialy, we need to replace it with this, which loads the function value of
rdx and then jumps to the location pointed to by
I’ve put the corresponding machine code that those lines generate when assembled next to it (you can easily play around with assembly using an online assembler like this). Writing a function that will generate this code is now straightforward, and looks like this:
We now have everything we need to replace
a’s function body with a jump to
b! The following code attempts to copy the machine code directly to the location of the function body.
Running this code does not work however, and will result in a segmentation fault. This is because the loaded binary is not writable by default. We can use the
mprotect syscall to disable this protection, and this final version of the code does exactly that, resulting in function
a being replaced by function
b, and ‘2’ being printed.
Wrapping it up in a nice library
I took the above code and put it in an easy to use library. It supports 32 bit, reversing patches, and patching instance methods. I wrote a couple of examples and put those in the README.
Where there’s a will there’s a way! It’s possible for a program to modify itself at runtime, which allows us to implement cool tricks like monkey patching.
I hope you got something useful out of this blogpost, I know I had fun making it!