python - printf 如何损坏字符串?

标签 python c assembly reverse-engineering

我正在尝试对我没有源代码且受反调试器保护的函数进行逆向工程。

无论如何,我对调用程序集中的 PyRun_ConsoleString 函数感兴趣:

CPU Disasm
Address   Hex dump          Command                                                             Comments
200DAB20  /$  68 D8961A20   PUSH OFFSET 201A96D8                                                ; ASCII "__main__"
200DAB25  |.  FF15 B0331720 CALL DWORD PTR DS:[<&vampire_python21.PyImport_AddModule>]
200DAB2B  |.  83C4 04       ADD ESP,4
200DAB2E  |.  85C0          TEST EAX,EAX
200DAB30  |.  74 4C         JE SHORT 200DAB7E
200DAB32  |.  50            PUSH EAX
200DAB33  |.  FF15 8C331720 CALL DWORD PTR DS:[<&vampire_python21.PyModule_GetDict>]
200DAB39  |.  50            PUSH EAX
200DAB3A  |.  50            PUSH EAX
200DAB3B  |.  8B4424 10     MOV EAX,DWORD PTR SS:[ESP+10]
200DAB3F  |.  68 00010000   PUSH 100
200DAB44  |.  50            PUSH EAX
200DAB45  |.  FF15 90331720 CALL DWORD PTR DS:[<&vampire_python21.PyRun_ConsoleString>]
200DAB4B  |.  83C4 14       ADD ESP,14
200DAB4E  |.  85C0          TEST EAX,EAX
200DAB50  |.  75 08         JNE SHORT 200DAB5A
200DAB52  |.  FF15 94331720 CALL DWORD PTR DS:[<&vampire_python21.PyErr_Print>]
200DAB58  |.- EB 1E         JMP SHORT <JMP.&vampire_python21.Py_FlushConsoleOutput>             ; Jump to vampire_python21.Py_FlushConsoleOutput
200DAB5A  |>  FF08          DEC DWORD PTR DS:[EAX]
200DAB5C  |.  75 0A         JNE SHORT 200DAB68
200DAB5E  |.  8B48 04       MOV ECX,DWORD PTR DS:[EAX+4]
200DAB61  |.  50            PUSH EAX
200DAB62  |.  FF51 18       CALL DWORD PTR DS:[ECX+18]
200DAB65  |.  83C4 04       ADD ESP,4
200DAB68  |>  FF15 98331720 CALL DWORD PTR DS:[<&vampire_python21.Py_FlushLine>]
200DAB6E  |.  85C0          TEST EAX,EAX
200DAB70  |.- 74 06         JE SHORT <JMP.&vampire_python21.Py_FlushConsoleOutput>              ; Jump to vampire_python21.Py_FlushConsoleOutput
200DAB72  |.  FF15 9C331720 CALL DWORD PTR DS:[<&vampire_python21.PyErr_Clear>]
200DAB78  |>- FF25 BC331720 JMP DWORD PTR DS:[<&vampire_python21.Py_FlushConsoleOutput>]
200DAB7E  \>  C3            RETN 

所以我认为很明显签名会是 int (const char*, int/它总是 256 并且与这些 python 方法中的常量匹配/, PyObject */*getDict 的返回 */, PyObject *)

但是,每当我尝试访问(或处理)假定的 pyObjects 为 pyobject 时,我就会崩溃,即用

打印它们

printf("%s\n", PyString_AsString(PyObject_Str(pyobj)));

所以我想打印字符串(我知道它是一个字符串,因为它来自一个文件)。

__declspec (dllexport) int PyRun_ConsoleString(const char *str, int typeOfExpression, PyObject * globals, PyObject * locals){

    printf("%s\n", str);
    fflush(stdout);
    return 0;
}

printf("%s\n",str) 打印预期的字符串 'qwerty "what"' 好的,这就是文件中的内容

printf("%s 1\n", str) 打印' 1 rty "what"' ?!?

如果您能协助查找返回值类型,我们将不胜感激。该程序集是:

CPU Disasm
Address   Hex dump          Command                                 Comments
1E153160  /$  8B4424 10     MOV EAX,DWORD PTR SS:[ARG.4]
1E153164  |.  8B4C24 0C     MOV ECX,DWORD PTR SS:[ARG.3]
1E153168  |.  8B5424 08     MOV EDX,DWORD PTR SS:[ARG.2]
1E15316C  |.  6A 00         PUSH 0
1E15316E  |.  50            PUSH EAX
1E15316F  |.  8B4424 0C     MOV EAX,DWORD PTR SS:[ARG.1]
1E153173  |.  51            PUSH ECX
1E153174  |.  68 B825191E   PUSH OFFSET 1E1925B8                    ; ASCII "<string>"
1E153179  |.  52            PUSH EDX                                ; /Arg2 => [ARG.2]
1E15317A  |.  50            PUSH EAX                                ; |Arg1 => [ARG.1]
1E15317B  |.  E8 D0030000   CALL PyParser_SimpleParseConsoleString  ; \vampire_python21_backup.PyParser_SimpleParseConsoleString
1E153180  |.  83C4 08       ADD ESP,8
1E153183  |.  50            PUSH EAX
1E153184  |.  E8 37010000   CALL 1E1532C0
1E153189  |.  83C4 14       ADD ESP,14
1E15318C  \.  C3            RETN

虽然我怀疑这个值来自一个子函数。

最佳答案

嗯,不错的拼图。 总的来说: printf("%s 1\n", str) 打印' 1 rty "what"' ?!? ... 在我看来就像您在控制台上看到的,对吧?)是:

qwerty“什么”

后跟回车(无换行),被覆盖:

'1'

然后\n

这表明您提供的字符串(来自文件?)包含终止返回字符。

关于python - printf 如何损坏字符串?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5229500/

相关文章:

c - 如何让我的代码免于进入无限循环?

C 中比较器函数的正确实现未按预期工作

assembly - ARM 组件 Raspberry-Pi : Converting a string to Upper case

windows - x64下的线程信息 block

python - 在 Plone 附加组件中包含 Python 脚本

python - Unicode解码错误: 'utf-8' codec can't decode byte 0xd5 in position 3362: invalid continuation byte

c - 链表错误,指针

delphi - 如何在 64 位代码中实现高效的 32 位 DivMod

python - Google Python API 电子表格行/列计数

python - 缩进错误 :at second char. lower()?