aboutsummaryrefslogtreecommitdiff
path: root/scripts/Python/modules/readline/readline.cpp
blob: d66ccf4b6b7dc7979ef75e3d4c7769ea902c7a85 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
// NOTE: Since Python may define some pre-processor definitions which affect the
// standard headers on some systems, you must include Python.h before any
// standard headers are included.
#include "Python.h"

#include <stdio.h>

#ifndef LLDB_DISABLE_LIBEDIT
#include <editline/readline.h>
#endif

// Simple implementation of the Python readline module using libedit.
// In the event that libedit is excluded from the build, this turns
// back into a null implementation that blocks the module from pulling
// in the GNU readline shared lib, which causes linkage confusion when
// both readline and libedit's readline compatibility symbols collide.
//
// Currently it only installs a PyOS_ReadlineFunctionPointer, without
// implementing any of the readline module methods. This is meant to
// work around LLVM pr18841 to avoid seg faults in the stock Python
// readline.so linked against GNU readline.

static struct PyMethodDef moduleMethods[] =
{
    {nullptr, nullptr, 0, nullptr}
};

#ifndef LLDB_DISABLE_LIBEDIT
PyDoc_STRVAR(
    moduleDocumentation,
    "Simple readline module implementation based on libedit.");
#else
PyDoc_STRVAR(
    moduleDocumentation,
    "Stub module meant to avoid linking GNU readline.");
#endif

#ifndef LLDB_DISABLE_LIBEDIT
static char*
simple_readline(FILE *stdin, FILE *stdout, char *prompt)
{
    rl_instream = stdin;
    rl_outstream = stdout;
    char* line = readline(prompt);
    if (!line)
    {
        char* ret = (char*)PyMem_Malloc(1);
        if (ret != NULL)
            *ret = '\0';
        return ret;
    }
    if (*line)
        add_history(line);
    int n = strlen(line);
    char* ret = (char*)PyMem_Malloc(n + 2);
    strncpy(ret, line, n);
    free(line);
    ret[n] = '\n';
    ret[n+1] = '\0';
    return ret;
}
#endif

PyMODINIT_FUNC
initreadline(void)
{
#ifndef LLDB_DISABLE_LIBEDIT
    PyOS_ReadlineFunctionPointer = simple_readline;
#endif
    Py_InitModule4(
        "readline",
        moduleMethods,
        moduleDocumentation,
        static_cast<PyObject *>(NULL),
        PYTHON_API_VERSION);
}