manifest.cc
72 lines
| 2.0 KiB
| text/x-c
|
CppLexer
Augie Fackler
|
r40089 | #include <Python.h> | ||
#include <assert.h> | ||||
#include <stdlib.h> | ||||
#include <unistd.h> | ||||
Augie Fackler
|
r45519 | #include "FuzzedDataProvider.h" | ||
Augie Fackler
|
r41049 | #include "pyutil.h" | ||
Augie Fackler
|
r40089 | #include <string> | ||
extern "C" { | ||||
Augie Fackler
|
r44311 | static PYCODETYPE *code; | ||
Augie Fackler
|
r40409 | |||
Augie Fackler
|
r40089 | extern "C" int LLVMFuzzerInitialize(int *argc, char ***argv) | ||
{ | ||||
Augie Fackler
|
r41049 | contrib::initpy(*argv[0]); | ||
Augie Fackler
|
r44311 | code = (PYCODETYPE *)Py_CompileString(R"py( | ||
Augie Fackler
|
r40089 | try: | ||
Augie Fackler
|
r44311 | lm = parsers.lazymanifest(mdata) | ||
Augie Fackler
|
r40089 | # iterate the whole thing, which causes the code to fully parse | ||
# every line in the manifest | ||||
Augie Fackler
|
r41342 | for e, _, _ in lm.iterentries(): | ||
# also exercise __getitem__ et al | ||||
lm[e] | ||||
e in lm | ||||
(e + 'nope') in lm | ||||
Augie Fackler
|
r45519 | lm[b'xyzzy'] = (b'\0' * nlen, 'x') | ||
Augie Fackler
|
r40089 | # do an insert, text should change | ||
assert lm.text() != mdata, "insert should change text and didn't: %r %r" % (lm.text(), mdata) | ||||
Augie Fackler
|
r41342 | cloned = lm.filtercopy(lambda x: x != 'xyzzy') | ||
assert cloned.text() == mdata, 'cloned text should equal mdata' | ||||
cloned.diff(lm) | ||||
Augie Fackler
|
r40089 | del lm[b'xyzzy'] | ||
Augie Fackler
|
r41342 | cloned.diff(lm) | ||
Augie Fackler
|
r40089 | # should be back to the same | ||
assert lm.text() == mdata, "delete should have restored text but didn't: %r %r" % (lm.text(), mdata) | ||||
except Exception as e: | ||||
pass | ||||
# uncomment this print if you're editing this Python code | ||||
# to debug failures. | ||||
# print e | ||||
)py", | ||||
Augie Fackler
|
r44311 | "fuzzer", Py_file_input); | ||
Augie Fackler
|
r40409 | return 0; | ||
} | ||||
int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) | ||||
{ | ||||
Augie Fackler
|
r41341 | // Don't allow fuzzer inputs larger than 100k, since we'll just bog | ||
// down and not accomplish much. | ||||
if (Size > 100000) { | ||||
return 0; | ||||
} | ||||
Augie Fackler
|
r45519 | FuzzedDataProvider provider(Data, Size); | ||
Py_ssize_t nodelength = provider.ConsumeBool() ? 20 : 32; | ||||
PyObject *nlen = PyLong_FromSsize_t(nodelength); | ||||
Augie Fackler
|
r40409 | PyObject *mtext = | ||
PyBytes_FromStringAndSize((const char *)Data, (Py_ssize_t)Size); | ||||
PyObject *locals = PyDict_New(); | ||||
PyDict_SetItemString(locals, "mdata", mtext); | ||||
Augie Fackler
|
r45519 | PyDict_SetItemString(locals, "nlen", nlen); | ||
Augie Fackler
|
r41049 | PyObject *res = PyEval_EvalCode(code, contrib::pyglobals(), locals); | ||
Yuya Nishihara
|
r40136 | if (!res) { | ||
PyErr_Print(); | ||||
} | ||||
Py_XDECREF(res); | ||||
Augie Fackler
|
r40089 | Py_DECREF(locals); | ||
Py_DECREF(mtext); | ||||
return 0; // Non-zero return values are reserved for future use. | ||||
} | ||||
} | ||||