2024-01-05 14:49:04 +01:00
|
|
|
# Sourced from https://forum.nim-lang.org/t/9255#60617
|
|
|
|
|
|
|
|
import posix
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
type Instr {.union.} = object
|
|
|
|
bytes: array[8, byte]
|
|
|
|
value: uint64
|
2024-01-05 14:49:04 +01:00
|
|
|
|
2024-05-13 17:25:44 +02:00
|
|
|
proc mockImpl*(target, replacement: pointer) =
|
2024-01-05 14:49:04 +01:00
|
|
|
# YOLO who needs alignment
|
|
|
|
#doAssert (cast[ByteAddress](target) and ByteAddress(0x07)) == 0
|
|
|
|
var page = cast[pointer](cast[ByteAddress](target) and (not 0xfff))
|
|
|
|
doAssert mprotect(page, 4096, PROT_WRITE or PROT_EXEC) == 0
|
|
|
|
let rel = cast[ByteAddress](replacement) - cast[ByteAddress](target) - 5
|
2024-03-16 00:08:47 +01:00
|
|
|
var instr = Instr(
|
|
|
|
bytes: [
|
|
|
|
0xe9.byte,
|
|
|
|
(rel shr 0).byte,
|
|
|
|
(rel shr 8).byte,
|
|
|
|
(rel shr 16).byte,
|
|
|
|
(rel shr 24).byte,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
]
|
|
|
|
)
|
2024-01-05 14:49:04 +01:00
|
|
|
cast[ptr uint64](target)[] = instr.value
|
|
|
|
doAssert mprotect(page, 4096, PROT_EXEC) == 0
|
|
|
|
|
|
|
|
# Note: Requires manual cleanup
|
|
|
|
# Usage Example:
|
|
|
|
# proc helloWorld(): string =
|
|
|
|
# "Hello, World!"
|
|
|
|
#
|
|
|
|
# echo helloWorld() # "Hello, World!"
|
|
|
|
#
|
|
|
|
# let backup = helloWorld
|
|
|
|
# mock(helloWorld):
|
|
|
|
# proc mockedHellWorld(): string =
|
|
|
|
# "Mocked Hello, World!"
|
|
|
|
# mockedMigrate
|
|
|
|
#
|
|
|
|
# echo helloWorld() # "Mocked Hello, World!"
|
|
|
|
#
|
|
|
|
# helloWorld = backup # Restore the original function
|
|
|
|
template mock*(target, replacement: untyped): untyped =
|
|
|
|
mockImpl(cast[pointer](target), cast[pointer](replacement))
|