blob: 2269d05da8741e73c3165e18f16694fd913f6811 [file] [log] [blame]
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001
2#include "Python.h"
3#include "structmember.h"
4
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00005/* _functools module written and maintained
Raymond Hettinger9c323f82005-02-28 19:39:44 +00006 by Hye-Shik Chang <perky@FreeBSD.org>
7 with adaptations by Raymond Hettinger <python@rcn.com>
Thomas Wouters4d70c3d2006-06-08 14:42:34 +00008 Copyright (c) 2004, 2005, 2006 Python Software Foundation.
Raymond Hettinger9c323f82005-02-28 19:39:44 +00009 All rights reserved.
10*/
11
12/* partial object **********************************************************/
13
14typedef struct {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000015 PyObject_HEAD
16 PyObject *fn;
17 PyObject *args;
18 PyObject *kw;
19 PyObject *dict;
20 PyObject *weakreflist; /* List of weak references */
Raymond Hettinger9c323f82005-02-28 19:39:44 +000021} partialobject;
22
23static PyTypeObject partial_type;
24
25static PyObject *
26partial_new(PyTypeObject *type, PyObject *args, PyObject *kw)
27{
Alexander Belopolskye49af342015-03-01 15:08:17 -050028 PyObject *func, *pargs, *nargs, *pkw;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000029 partialobject *pto;
Raymond Hettinger9c323f82005-02-28 19:39:44 +000030
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000031 if (PyTuple_GET_SIZE(args) < 1) {
32 PyErr_SetString(PyExc_TypeError,
33 "type 'partial' takes at least one argument");
34 return NULL;
35 }
Raymond Hettinger9c323f82005-02-28 19:39:44 +000036
Serhiy Storchaka38741282016-02-02 18:45:17 +020037 pargs = pkw = NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000038 func = PyTuple_GET_ITEM(args, 0);
Alexander Belopolskye49af342015-03-01 15:08:17 -050039 if (Py_TYPE(func) == &partial_type && type == &partial_type) {
40 partialobject *part = (partialobject *)func;
41 if (part->dict == NULL) {
42 pargs = part->args;
43 pkw = part->kw;
44 func = part->fn;
Serhiy Storchaka38741282016-02-02 18:45:17 +020045 assert(PyTuple_Check(pargs));
46 assert(PyDict_Check(pkw));
Alexander Belopolskye49af342015-03-01 15:08:17 -050047 }
48 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000049 if (!PyCallable_Check(func)) {
50 PyErr_SetString(PyExc_TypeError,
51 "the first argument must be callable");
52 return NULL;
53 }
Raymond Hettinger9c323f82005-02-28 19:39:44 +000054
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000055 /* create partialobject structure */
56 pto = (partialobject *)type->tp_alloc(type, 0);
57 if (pto == NULL)
58 return NULL;
Raymond Hettinger9c323f82005-02-28 19:39:44 +000059
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000060 pto->fn = func;
61 Py_INCREF(func);
Alexander Belopolskye49af342015-03-01 15:08:17 -050062
63 nargs = PyTuple_GetSlice(args, 1, PY_SSIZE_T_MAX);
64 if (nargs == NULL) {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000065 Py_DECREF(pto);
66 return NULL;
67 }
Serhiy Storchaka38741282016-02-02 18:45:17 +020068 if (pargs == NULL || PyTuple_GET_SIZE(pargs) == 0) {
Alexander Belopolskye49af342015-03-01 15:08:17 -050069 pto->args = nargs;
70 Py_INCREF(nargs);
71 }
72 else if (PyTuple_GET_SIZE(nargs) == 0) {
73 pto->args = pargs;
74 Py_INCREF(pargs);
75 }
76 else {
77 pto->args = PySequence_Concat(pargs, nargs);
78 if (pto->args == NULL) {
Serhiy Storchaka38741282016-02-02 18:45:17 +020079 Py_DECREF(nargs);
Alexander Belopolskye49af342015-03-01 15:08:17 -050080 Py_DECREF(pto);
81 return NULL;
82 }
Serhiy Storchaka38741282016-02-02 18:45:17 +020083 assert(PyTuple_Check(pto->args));
Alexander Belopolskye49af342015-03-01 15:08:17 -050084 }
85 Py_DECREF(nargs);
86
Serhiy Storchaka38741282016-02-02 18:45:17 +020087 if (pkw == NULL || PyDict_Size(pkw) == 0) {
88 if (kw == NULL) {
89 pto->kw = PyDict_New();
Alexander Belopolskye49af342015-03-01 15:08:17 -050090 }
91 else {
Serhiy Storchaka38741282016-02-02 18:45:17 +020092 Py_INCREF(kw);
93 pto->kw = kw;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000094 }
Alexander Belopolskye49af342015-03-01 15:08:17 -050095 }
96 else {
Serhiy Storchaka38741282016-02-02 18:45:17 +020097 pto->kw = PyDict_Copy(pkw);
98 if (kw != NULL && pto->kw != NULL) {
99 if (PyDict_Merge(pto->kw, kw, 1) != 0) {
Benjamin Petersondae2ef12015-05-09 00:29:08 -0400100 Py_DECREF(pto);
101 return NULL;
102 }
103 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000104 }
Serhiy Storchaka38741282016-02-02 18:45:17 +0200105 if (pto->kw == NULL) {
106 Py_DECREF(pto);
107 return NULL;
108 }
Raymond Hettingerc8b6d1b2005-03-08 06:14:50 +0000109
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000110 return (PyObject *)pto;
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000111}
112
113static void
114partial_dealloc(partialobject *pto)
115{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000116 PyObject_GC_UnTrack(pto);
117 if (pto->weakreflist != NULL)
118 PyObject_ClearWeakRefs((PyObject *) pto);
119 Py_XDECREF(pto->fn);
120 Py_XDECREF(pto->args);
121 Py_XDECREF(pto->kw);
122 Py_XDECREF(pto->dict);
123 Py_TYPE(pto)->tp_free(pto);
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000124}
125
126static PyObject *
127partial_call(partialobject *pto, PyObject *args, PyObject *kw)
128{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000129 PyObject *ret;
Serhiy Storchaka38741282016-02-02 18:45:17 +0200130 PyObject *argappl, *kwappl;
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200131 PyObject **stack;
132 Py_ssize_t nargs;
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000133
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000134 assert (PyCallable_Check(pto->fn));
135 assert (PyTuple_Check(pto->args));
Serhiy Storchaka38741282016-02-02 18:45:17 +0200136 assert (PyDict_Check(pto->kw));
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000137
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000138 if (PyTuple_GET_SIZE(pto->args) == 0) {
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200139 stack = &PyTuple_GET_ITEM(args, 0);
140 nargs = PyTuple_GET_SIZE(args);
141 argappl = NULL;
142 }
143 else if (PyTuple_GET_SIZE(args) == 0) {
144 stack = &PyTuple_GET_ITEM(pto->args, 0);
145 nargs = PyTuple_GET_SIZE(pto->args);
146 argappl = NULL;
147 }
148 else {
149 stack = NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000150 argappl = PySequence_Concat(pto->args, args);
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200151 if (argappl == NULL) {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000152 return NULL;
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200153 }
154
Serhiy Storchaka38741282016-02-02 18:45:17 +0200155 assert(PyTuple_Check(argappl));
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000156 }
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000157
Serhiy Storchaka38741282016-02-02 18:45:17 +0200158 if (PyDict_Size(pto->kw) == 0) {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000159 kwappl = kw;
Serhiy Storchaka38741282016-02-02 18:45:17 +0200160 Py_XINCREF(kwappl);
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200161 }
162 else {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000163 kwappl = PyDict_Copy(pto->kw);
164 if (kwappl == NULL) {
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200165 Py_XDECREF(argappl);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000166 return NULL;
167 }
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200168
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000169 if (kw != NULL) {
170 if (PyDict_Merge(kwappl, kw, 1) != 0) {
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200171 Py_XDECREF(argappl);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000172 Py_DECREF(kwappl);
173 return NULL;
174 }
175 }
176 }
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000177
Victor Stinnerf4d28d42016-08-23 16:22:35 +0200178 if (stack) {
179 ret = _PyObject_FastCallDict(pto->fn, stack, nargs, kwappl);
180 }
181 else {
182 ret = PyObject_Call(pto->fn, argappl, kwappl);
183 Py_DECREF(argappl);
184 }
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000185 Py_XDECREF(kwappl);
186 return ret;
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000187}
188
189static int
190partial_traverse(partialobject *pto, visitproc visit, void *arg)
191{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000192 Py_VISIT(pto->fn);
193 Py_VISIT(pto->args);
194 Py_VISIT(pto->kw);
195 Py_VISIT(pto->dict);
196 return 0;
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000197}
198
199PyDoc_STRVAR(partial_doc,
200"partial(func, *args, **keywords) - new function with partial application\n\
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000201 of the given arguments and keywords.\n");
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000202
203#define OFF(x) offsetof(partialobject, x)
204static PyMemberDef partial_memberlist[] = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000205 {"func", T_OBJECT, OFF(fn), READONLY,
206 "function object to use in future partial calls"},
207 {"args", T_OBJECT, OFF(args), READONLY,
208 "tuple of arguments to future partial calls"},
209 {"keywords", T_OBJECT, OFF(kw), READONLY,
210 "dictionary of keyword arguments to future partial calls"},
211 {NULL} /* Sentinel */
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000212};
213
Georg Brandlc2fb6c72006-02-21 17:49:57 +0000214static PyGetSetDef partial_getsetlist[] = {
Benjamin Peterson23d7f122012-02-19 20:02:57 -0500215 {"__dict__", PyObject_GenericGetDict, PyObject_GenericSetDict},
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000216 {NULL} /* Sentinel */
Raymond Hettingerc8b6d1b2005-03-08 06:14:50 +0000217};
218
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000219static PyObject *
220partial_repr(partialobject *pto)
221{
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300222 PyObject *result = NULL;
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000223 PyObject *arglist;
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000224 Py_ssize_t i, n;
Serhiy Storchaka38741282016-02-02 18:45:17 +0200225 PyObject *key, *value;
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300226 int status;
227
228 status = Py_ReprEnter((PyObject *)pto);
229 if (status != 0) {
230 if (status < 0)
231 return NULL;
Nick Coghlan457fc9a2016-09-10 20:00:02 +1000232 return PyUnicode_FromString("...");
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300233 }
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000234
235 arglist = PyUnicode_FromString("");
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300236 if (arglist == NULL)
237 goto done;
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000238 /* Pack positional arguments */
239 assert (PyTuple_Check(pto->args));
240 n = PyTuple_GET_SIZE(pto->args);
241 for (i = 0; i < n; i++) {
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300242 Py_SETREF(arglist, PyUnicode_FromFormat("%U, %R", arglist,
243 PyTuple_GET_ITEM(pto->args, i)));
244 if (arglist == NULL)
245 goto done;
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000246 }
247 /* Pack keyword arguments */
Serhiy Storchaka38741282016-02-02 18:45:17 +0200248 assert (PyDict_Check(pto->kw));
249 for (i = 0; PyDict_Next(pto->kw, &i, &key, &value);) {
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300250 Py_SETREF(arglist, PyUnicode_FromFormat("%U, %U=%R", arglist,
251 key, value));
252 if (arglist == NULL)
253 goto done;
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000254 }
255 result = PyUnicode_FromFormat("%s(%R%U)", Py_TYPE(pto)->tp_name,
256 pto->fn, arglist);
257 Py_DECREF(arglist);
Serhiy Storchaka179f9602016-06-12 11:44:06 +0300258
259 done:
260 Py_ReprLeave((PyObject *)pto);
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000261 return result;
262}
263
Jack Diederiche0cbd692009-04-01 04:27:09 +0000264/* Pickle strategy:
265 __reduce__ by itself doesn't support getting kwargs in the unpickle
266 operation so we define a __setstate__ that replaces all the information
267 about the partial. If we only replaced part of it someone would use
Ezio Melotti13925002011-03-16 11:05:33 +0200268 it as a hook to do strange things.
Jack Diederiche0cbd692009-04-01 04:27:09 +0000269 */
270
Antoine Pitrou69f71142009-05-24 21:25:49 +0000271static PyObject *
Jack Diederiche0cbd692009-04-01 04:27:09 +0000272partial_reduce(partialobject *pto, PyObject *unused)
273{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000274 return Py_BuildValue("O(O)(OOOO)", Py_TYPE(pto), pto->fn, pto->fn,
275 pto->args, pto->kw,
276 pto->dict ? pto->dict : Py_None);
Jack Diederiche0cbd692009-04-01 04:27:09 +0000277}
278
Antoine Pitrou69f71142009-05-24 21:25:49 +0000279static PyObject *
Serhiy Storchaka19c4e0d2013-02-04 12:47:24 +0200280partial_setstate(partialobject *pto, PyObject *state)
Jack Diederiche0cbd692009-04-01 04:27:09 +0000281{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000282 PyObject *fn, *fnargs, *kw, *dict;
Serhiy Storchaka38741282016-02-02 18:45:17 +0200283
284 if (!PyTuple_Check(state) ||
285 !PyArg_ParseTuple(state, "OOOO", &fn, &fnargs, &kw, &dict) ||
286 !PyCallable_Check(fn) ||
287 !PyTuple_Check(fnargs) ||
288 (kw != Py_None && !PyDict_Check(kw)))
289 {
290 PyErr_SetString(PyExc_TypeError, "invalid partial state");
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000291 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000292 }
Serhiy Storchaka38741282016-02-02 18:45:17 +0200293
294 if(!PyTuple_CheckExact(fnargs))
295 fnargs = PySequence_Tuple(fnargs);
296 else
297 Py_INCREF(fnargs);
298 if (fnargs == NULL)
299 return NULL;
300
301 if (kw == Py_None)
302 kw = PyDict_New();
303 else if(!PyDict_CheckExact(kw))
304 kw = PyDict_Copy(kw);
305 else
306 Py_INCREF(kw);
307 if (kw == NULL) {
308 Py_DECREF(fnargs);
309 return NULL;
310 }
311
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000312 Py_INCREF(fn);
Serhiy Storchaka38741282016-02-02 18:45:17 +0200313 if (dict == Py_None)
314 dict = NULL;
315 else
316 Py_INCREF(dict);
317
Serhiy Storchaka864b63c2016-04-11 09:53:37 +0300318 Py_SETREF(pto->fn, fn);
319 Py_SETREF(pto->args, fnargs);
320 Py_SETREF(pto->kw, kw);
Serhiy Storchaka48842712016-04-06 09:45:48 +0300321 Py_XSETREF(pto->dict, dict);
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000322 Py_RETURN_NONE;
Jack Diederiche0cbd692009-04-01 04:27:09 +0000323}
324
325static PyMethodDef partial_methods[] = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000326 {"__reduce__", (PyCFunction)partial_reduce, METH_NOARGS},
Serhiy Storchaka19c4e0d2013-02-04 12:47:24 +0200327 {"__setstate__", (PyCFunction)partial_setstate, METH_O},
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000328 {NULL, NULL} /* sentinel */
Jack Diederiche0cbd692009-04-01 04:27:09 +0000329};
330
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000331static PyTypeObject partial_type = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000332 PyVarObject_HEAD_INIT(NULL, 0)
333 "functools.partial", /* tp_name */
334 sizeof(partialobject), /* tp_basicsize */
335 0, /* tp_itemsize */
336 /* methods */
337 (destructor)partial_dealloc, /* tp_dealloc */
338 0, /* tp_print */
339 0, /* tp_getattr */
340 0, /* tp_setattr */
341 0, /* tp_reserved */
Alexander Belopolsky41e422a2010-12-01 20:05:49 +0000342 (reprfunc)partial_repr, /* tp_repr */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000343 0, /* tp_as_number */
344 0, /* tp_as_sequence */
345 0, /* tp_as_mapping */
346 0, /* tp_hash */
347 (ternaryfunc)partial_call, /* tp_call */
348 0, /* tp_str */
349 PyObject_GenericGetAttr, /* tp_getattro */
350 PyObject_GenericSetAttr, /* tp_setattro */
351 0, /* tp_as_buffer */
352 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
353 Py_TPFLAGS_BASETYPE, /* tp_flags */
354 partial_doc, /* tp_doc */
355 (traverseproc)partial_traverse, /* tp_traverse */
356 0, /* tp_clear */
357 0, /* tp_richcompare */
358 offsetof(partialobject, weakreflist), /* tp_weaklistoffset */
359 0, /* tp_iter */
360 0, /* tp_iternext */
361 partial_methods, /* tp_methods */
362 partial_memberlist, /* tp_members */
363 partial_getsetlist, /* tp_getset */
364 0, /* tp_base */
365 0, /* tp_dict */
366 0, /* tp_descr_get */
367 0, /* tp_descr_set */
368 offsetof(partialobject, dict), /* tp_dictoffset */
369 0, /* tp_init */
370 0, /* tp_alloc */
371 partial_new, /* tp_new */
372 PyObject_GC_Del, /* tp_free */
Raymond Hettinger9c323f82005-02-28 19:39:44 +0000373};
374
375
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700376/* cmp_to_key ***************************************************************/
377
378typedef struct {
Victor Stinner446c8d52011-04-05 12:21:35 +0200379 PyObject_HEAD
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700380 PyObject *cmp;
381 PyObject *object;
382} keyobject;
383
384static void
385keyobject_dealloc(keyobject *ko)
386{
387 Py_DECREF(ko->cmp);
388 Py_XDECREF(ko->object);
389 PyObject_FREE(ko);
390}
391
392static int
393keyobject_traverse(keyobject *ko, visitproc visit, void *arg)
394{
395 Py_VISIT(ko->cmp);
396 if (ko->object)
397 Py_VISIT(ko->object);
398 return 0;
399}
400
Benjamin Peterson3bd97292011-04-05 17:25:14 -0500401static int
402keyobject_clear(keyobject *ko)
403{
404 Py_CLEAR(ko->cmp);
405 if (ko->object)
406 Py_CLEAR(ko->object);
407 return 0;
408}
409
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700410static PyMemberDef keyobject_members[] = {
411 {"obj", T_OBJECT,
412 offsetof(keyobject, object), 0,
413 PyDoc_STR("Value wrapped by a key function.")},
414 {NULL}
415};
416
417static PyObject *
Raymond Hettingera5632862011-04-09 12:57:00 -0700418keyobject_call(keyobject *ko, PyObject *args, PyObject *kwds);
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700419
420static PyObject *
421keyobject_richcompare(PyObject *ko, PyObject *other, int op);
422
423static PyTypeObject keyobject_type = {
424 PyVarObject_HEAD_INIT(&PyType_Type, 0)
425 "functools.KeyWrapper", /* tp_name */
426 sizeof(keyobject), /* tp_basicsize */
427 0, /* tp_itemsize */
428 /* methods */
429 (destructor)keyobject_dealloc, /* tp_dealloc */
430 0, /* tp_print */
431 0, /* tp_getattr */
432 0, /* tp_setattr */
433 0, /* tp_reserved */
434 0, /* tp_repr */
435 0, /* tp_as_number */
436 0, /* tp_as_sequence */
437 0, /* tp_as_mapping */
438 0, /* tp_hash */
439 (ternaryfunc)keyobject_call, /* tp_call */
440 0, /* tp_str */
441 PyObject_GenericGetAttr, /* tp_getattro */
442 0, /* tp_setattro */
443 0, /* tp_as_buffer */
444 Py_TPFLAGS_DEFAULT, /* tp_flags */
445 0, /* tp_doc */
446 (traverseproc)keyobject_traverse, /* tp_traverse */
Benjamin Peterson3bd97292011-04-05 17:25:14 -0500447 (inquiry)keyobject_clear, /* tp_clear */
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700448 keyobject_richcompare, /* tp_richcompare */
449 0, /* tp_weaklistoffset */
450 0, /* tp_iter */
451 0, /* tp_iternext */
452 0, /* tp_methods */
453 keyobject_members, /* tp_members */
454 0, /* tp_getset */
455};
456
457static PyObject *
458keyobject_call(keyobject *ko, PyObject *args, PyObject *kwds)
459{
460 PyObject *object;
461 keyobject *result;
462 static char *kwargs[] = {"obj", NULL};
463
464 if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:K", kwargs, &object))
465 return NULL;
466 result = PyObject_New(keyobject, &keyobject_type);
467 if (!result)
468 return NULL;
469 Py_INCREF(ko->cmp);
470 result->cmp = ko->cmp;
471 Py_INCREF(object);
472 result->object = object;
473 return (PyObject *)result;
474}
475
476static PyObject *
477keyobject_richcompare(PyObject *ko, PyObject *other, int op)
478{
479 PyObject *res;
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700480 PyObject *x;
481 PyObject *y;
482 PyObject *compare;
483 PyObject *answer;
484 static PyObject *zero;
Victor Stinnerf7a4c482016-08-19 18:52:35 +0200485 PyObject* stack[2];
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700486
487 if (zero == NULL) {
488 zero = PyLong_FromLong(0);
489 if (!zero)
490 return NULL;
491 }
492
493 if (Py_TYPE(other) != &keyobject_type){
494 PyErr_Format(PyExc_TypeError, "other argument must be K instance");
495 return NULL;
496 }
497 compare = ((keyobject *) ko)->cmp;
498 assert(compare != NULL);
499 x = ((keyobject *) ko)->object;
500 y = ((keyobject *) other)->object;
501 if (!x || !y){
502 PyErr_Format(PyExc_AttributeError, "object");
503 return NULL;
504 }
505
506 /* Call the user's comparison function and translate the 3-way
507 * result into true or false (or error).
508 */
Victor Stinnerf7a4c482016-08-19 18:52:35 +0200509 stack[0] = x;
510 stack[1] = y;
Victor Stinner559bb6a2016-08-22 22:48:54 +0200511 res = _PyObject_FastCall(compare, stack, 2);
Victor Stinnerf7a4c482016-08-19 18:52:35 +0200512 if (res == NULL) {
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700513 return NULL;
Victor Stinnerf7a4c482016-08-19 18:52:35 +0200514 }
515
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700516 answer = PyObject_RichCompare(res, zero, op);
517 Py_DECREF(res);
518 return answer;
519}
520
521static PyObject *
Victor Stinner446c8d52011-04-05 12:21:35 +0200522functools_cmp_to_key(PyObject *self, PyObject *args, PyObject *kwds)
523{
524 PyObject *cmp;
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700525 static char *kwargs[] = {"mycmp", NULL};
Victor Stinner446c8d52011-04-05 12:21:35 +0200526 keyobject *object;
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700527
528 if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:cmp_to_key", kwargs, &cmp))
529 return NULL;
Victor Stinner446c8d52011-04-05 12:21:35 +0200530 object = PyObject_New(keyobject, &keyobject_type);
Raymond Hettinger7ab9e222011-04-05 02:33:54 -0700531 if (!object)
532 return NULL;
533 Py_INCREF(cmp);
534 object->cmp = cmp;
535 object->object = NULL;
536 return (PyObject *)object;
537}
538
539PyDoc_STRVAR(functools_cmp_to_key_doc,
540"Convert a cmp= function into a key= function.");
541
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000542/* reduce (used to be a builtin) ********************************************/
543
544static PyObject *
545functools_reduce(PyObject *self, PyObject *args)
546{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000547 PyObject *seq, *func, *result = NULL, *it;
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000548
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000549 if (!PyArg_UnpackTuple(args, "reduce", 2, 3, &func, &seq, &result))
550 return NULL;
551 if (result != NULL)
552 Py_INCREF(result);
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000553
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000554 it = PyObject_GetIter(seq);
555 if (it == NULL) {
Alexander Belopolskye29e6bf2010-08-16 18:55:46 +0000556 if (PyErr_ExceptionMatches(PyExc_TypeError))
557 PyErr_SetString(PyExc_TypeError,
558 "reduce() arg 2 must support iteration");
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000559 Py_XDECREF(result);
560 return NULL;
561 }
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000562
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000563 if ((args = PyTuple_New(2)) == NULL)
564 goto Fail;
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000565
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000566 for (;;) {
567 PyObject *op2;
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000568
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000569 if (args->ob_refcnt > 1) {
570 Py_DECREF(args);
571 if ((args = PyTuple_New(2)) == NULL)
572 goto Fail;
573 }
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000574
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000575 op2 = PyIter_Next(it);
576 if (op2 == NULL) {
577 if (PyErr_Occurred())
578 goto Fail;
579 break;
580 }
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000581
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000582 if (result == NULL)
583 result = op2;
584 else {
585 PyTuple_SetItem(args, 0, result);
586 PyTuple_SetItem(args, 1, op2);
587 if ((result = PyEval_CallObject(func, args)) == NULL)
588 goto Fail;
589 }
590 }
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000591
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000592 Py_DECREF(args);
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000593
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000594 if (result == NULL)
595 PyErr_SetString(PyExc_TypeError,
596 "reduce() of empty sequence with no initial value");
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000597
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000598 Py_DECREF(it);
599 return result;
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000600
601Fail:
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000602 Py_XDECREF(args);
603 Py_XDECREF(result);
604 Py_DECREF(it);
605 return NULL;
Guido van Rossum0919a1a2006-08-26 20:49:04 +0000606}
607
608PyDoc_STRVAR(functools_reduce_doc,
609"reduce(function, sequence[, initial]) -> value\n\
610\n\
611Apply a function of two arguments cumulatively to the items of a sequence,\n\
612from left to right, so as to reduce the sequence to a single value.\n\
613For example, reduce(lambda x, y: x+y, [1, 2, 3, 4, 5]) calculates\n\
614((((1+2)+3)+4)+5). If initial is present, it is placed before the items\n\
615of the sequence in the calculation, and serves as a default when the\n\
616sequence is empty.");
617
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300618/* lru_cache object **********************************************************/
619
620/* this object is used delimit args and keywords in the cache keys */
621static PyObject *kwd_mark = NULL;
622
623struct lru_list_elem;
624struct lru_cache_object;
625
626typedef struct lru_list_elem {
627 PyObject_HEAD
628 struct lru_list_elem *prev, *next; /* borrowed links */
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300629 Py_hash_t hash;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300630 PyObject *key, *result;
631} lru_list_elem;
632
633static void
634lru_list_elem_dealloc(lru_list_elem *link)
635{
636 _PyObject_GC_UNTRACK(link);
637 Py_XDECREF(link->key);
638 Py_XDECREF(link->result);
639 PyObject_GC_Del(link);
640}
641
642static int
643lru_list_elem_traverse(lru_list_elem *link, visitproc visit, void *arg)
644{
645 Py_VISIT(link->key);
646 Py_VISIT(link->result);
647 return 0;
648}
649
650static int
651lru_list_elem_clear(lru_list_elem *link)
652{
653 Py_CLEAR(link->key);
654 Py_CLEAR(link->result);
655 return 0;
656}
657
658static PyTypeObject lru_list_elem_type = {
659 PyVarObject_HEAD_INIT(&PyType_Type, 0)
660 "functools._lru_list_elem", /* tp_name */
661 sizeof(lru_list_elem), /* tp_basicsize */
662 0, /* tp_itemsize */
663 /* methods */
664 (destructor)lru_list_elem_dealloc, /* tp_dealloc */
665 0, /* tp_print */
666 0, /* tp_getattr */
667 0, /* tp_setattr */
668 0, /* tp_reserved */
669 0, /* tp_repr */
670 0, /* tp_as_number */
671 0, /* tp_as_sequence */
672 0, /* tp_as_mapping */
673 0, /* tp_hash */
674 0, /* tp_call */
675 0, /* tp_str */
676 0, /* tp_getattro */
677 0, /* tp_setattro */
678 0, /* tp_as_buffer */
679 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */
680 0, /* tp_doc */
681 (traverseproc)lru_list_elem_traverse, /* tp_traverse */
682 (inquiry)lru_list_elem_clear, /* tp_clear */
683};
684
685
686typedef PyObject *(*lru_cache_ternaryfunc)(struct lru_cache_object *, PyObject *, PyObject *);
687
688typedef struct lru_cache_object {
689 lru_list_elem root; /* includes PyObject_HEAD */
690 Py_ssize_t maxsize;
691 PyObject *maxsize_O;
692 PyObject *func;
693 lru_cache_ternaryfunc wrapper;
694 PyObject *cache;
695 PyObject *cache_info_type;
696 Py_ssize_t misses, hits;
697 int typed;
698 PyObject *dict;
699 int full;
700} lru_cache_object;
701
702static PyTypeObject lru_cache_type;
703
704static PyObject *
705lru_cache_make_key(PyObject *args, PyObject *kwds, int typed)
706{
707 PyObject *key, *sorted_items;
708 Py_ssize_t key_size, pos, key_pos;
709
710 /* short path, key will match args anyway, which is a tuple */
711 if (!typed && !kwds) {
712 Py_INCREF(args);
713 return args;
714 }
715
716 if (kwds && PyDict_Size(kwds) > 0) {
717 sorted_items = PyDict_Items(kwds);
718 if (!sorted_items)
719 return NULL;
720 if (PyList_Sort(sorted_items) < 0) {
721 Py_DECREF(sorted_items);
722 return NULL;
723 }
724 } else
725 sorted_items = NULL;
726
727 key_size = PyTuple_GET_SIZE(args);
728 if (sorted_items)
729 key_size += PyList_GET_SIZE(sorted_items);
730 if (typed)
731 key_size *= 2;
732 if (sorted_items)
733 key_size++;
734
735 key = PyTuple_New(key_size);
736 if (key == NULL)
737 goto done;
738
739 key_pos = 0;
740 for (pos = 0; pos < PyTuple_GET_SIZE(args); ++pos) {
741 PyObject *item = PyTuple_GET_ITEM(args, pos);
742 Py_INCREF(item);
743 PyTuple_SET_ITEM(key, key_pos++, item);
744 }
745 if (sorted_items) {
746 Py_INCREF(kwd_mark);
747 PyTuple_SET_ITEM(key, key_pos++, kwd_mark);
748 for (pos = 0; pos < PyList_GET_SIZE(sorted_items); ++pos) {
749 PyObject *item = PyList_GET_ITEM(sorted_items, pos);
750 Py_INCREF(item);
751 PyTuple_SET_ITEM(key, key_pos++, item);
752 }
753 }
754 if (typed) {
755 for (pos = 0; pos < PyTuple_GET_SIZE(args); ++pos) {
756 PyObject *item = (PyObject *)Py_TYPE(PyTuple_GET_ITEM(args, pos));
757 Py_INCREF(item);
758 PyTuple_SET_ITEM(key, key_pos++, item);
759 }
760 if (sorted_items) {
761 for (pos = 0; pos < PyList_GET_SIZE(sorted_items); ++pos) {
762 PyObject *tp_items = PyList_GET_ITEM(sorted_items, pos);
763 PyObject *item = (PyObject *)Py_TYPE(PyTuple_GET_ITEM(tp_items, 1));
764 Py_INCREF(item);
765 PyTuple_SET_ITEM(key, key_pos++, item);
766 }
767 }
768 }
769 assert(key_pos == key_size);
770
771done:
772 if (sorted_items)
773 Py_DECREF(sorted_items);
774 return key;
775}
776
777static PyObject *
778uncached_lru_cache_wrapper(lru_cache_object *self, PyObject *args, PyObject *kwds)
779{
780 PyObject *result = PyObject_Call(self->func, args, kwds);
781 if (!result)
782 return NULL;
783 self->misses++;
784 return result;
785}
786
787static PyObject *
788infinite_lru_cache_wrapper(lru_cache_object *self, PyObject *args, PyObject *kwds)
789{
790 PyObject *result;
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300791 Py_hash_t hash;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300792 PyObject *key = lru_cache_make_key(args, kwds, self->typed);
793 if (!key)
794 return NULL;
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300795 hash = PyObject_Hash(key);
Yury Selivanov46a02db2016-11-09 18:55:45 -0500796 if (hash == -1) {
797 Py_DECREF(key);
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300798 return NULL;
Yury Selivanov46a02db2016-11-09 18:55:45 -0500799 }
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300800 result = _PyDict_GetItem_KnownHash(self->cache, key, hash);
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300801 if (result) {
802 Py_INCREF(result);
803 self->hits++;
804 Py_DECREF(key);
805 return result;
806 }
807 if (PyErr_Occurred()) {
808 Py_DECREF(key);
809 return NULL;
810 }
811 result = PyObject_Call(self->func, args, kwds);
812 if (!result) {
813 Py_DECREF(key);
814 return NULL;
815 }
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300816 if (_PyDict_SetItem_KnownHash(self->cache, key, result, hash) < 0) {
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300817 Py_DECREF(result);
818 Py_DECREF(key);
819 return NULL;
820 }
821 Py_DECREF(key);
822 self->misses++;
823 return result;
824}
825
826static void
827lru_cache_extricate_link(lru_list_elem *link)
828{
829 link->prev->next = link->next;
830 link->next->prev = link->prev;
831}
832
833static void
834lru_cache_append_link(lru_cache_object *self, lru_list_elem *link)
835{
836 lru_list_elem *root = &self->root;
837 lru_list_elem *last = root->prev;
838 last->next = root->prev = link;
839 link->prev = last;
840 link->next = root;
841}
842
843static PyObject *
844bounded_lru_cache_wrapper(lru_cache_object *self, PyObject *args, PyObject *kwds)
845{
846 lru_list_elem *link;
847 PyObject *key, *result;
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300848 Py_hash_t hash;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300849
850 key = lru_cache_make_key(args, kwds, self->typed);
851 if (!key)
852 return NULL;
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300853 hash = PyObject_Hash(key);
Yury Selivanov46a02db2016-11-09 18:55:45 -0500854 if (hash == -1) {
855 Py_DECREF(key);
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300856 return NULL;
Yury Selivanov46a02db2016-11-09 18:55:45 -0500857 }
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300858 link = (lru_list_elem *)_PyDict_GetItem_KnownHash(self->cache, key, hash);
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300859 if (link) {
860 lru_cache_extricate_link(link);
861 lru_cache_append_link(self, link);
862 self->hits++;
863 result = link->result;
864 Py_INCREF(result);
865 Py_DECREF(key);
866 return result;
867 }
868 if (PyErr_Occurred()) {
869 Py_DECREF(key);
870 return NULL;
871 }
872 result = PyObject_Call(self->func, args, kwds);
873 if (!result) {
874 Py_DECREF(key);
875 return NULL;
876 }
877 if (self->full && self->root.next != &self->root) {
878 /* Use the oldest item to store the new key and result. */
879 PyObject *oldkey, *oldresult;
880 /* Extricate the oldest item. */
881 link = self->root.next;
882 lru_cache_extricate_link(link);
883 /* Remove it from the cache.
884 The cache dict holds one reference to the link,
885 and the linked list holds yet one reference to it. */
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300886 if (_PyDict_DelItem_KnownHash(self->cache, link->key,
887 link->hash) < 0) {
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300888 lru_cache_append_link(self, link);
889 Py_DECREF(key);
890 Py_DECREF(result);
891 return NULL;
892 }
893 /* Keep a reference to the old key and old result to
894 prevent their ref counts from going to zero during the
895 update. That will prevent potentially arbitrary object
896 clean-up code (i.e. __del__) from running while we're
897 still adjusting the links. */
898 oldkey = link->key;
899 oldresult = link->result;
900
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300901 link->hash = hash;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300902 link->key = key;
903 link->result = result;
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300904 if (_PyDict_SetItem_KnownHash(self->cache, key, (PyObject *)link,
905 hash) < 0) {
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300906 Py_DECREF(link);
907 Py_DECREF(oldkey);
908 Py_DECREF(oldresult);
909 return NULL;
910 }
911 lru_cache_append_link(self, link);
912 Py_INCREF(result); /* for return */
913 Py_DECREF(oldkey);
914 Py_DECREF(oldresult);
915 } else {
916 /* Put result in a new link at the front of the queue. */
917 link = (lru_list_elem *)PyObject_GC_New(lru_list_elem,
918 &lru_list_elem_type);
919 if (link == NULL) {
920 Py_DECREF(key);
921 Py_DECREF(result);
922 return NULL;
923 }
924
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300925 link->hash = hash;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300926 link->key = key;
927 link->result = result;
928 _PyObject_GC_TRACK(link);
Serhiy Storchakab9d98d52015-10-02 12:47:11 +0300929 if (_PyDict_SetItem_KnownHash(self->cache, key, (PyObject *)link,
930 hash) < 0) {
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300931 Py_DECREF(link);
932 return NULL;
933 }
934 lru_cache_append_link(self, link);
935 Py_INCREF(result); /* for return */
936 self->full = (PyDict_Size(self->cache) >= self->maxsize);
937 }
938 self->misses++;
939 return result;
940}
941
942static PyObject *
943lru_cache_new(PyTypeObject *type, PyObject *args, PyObject *kw)
944{
Serhiy Storchaka374164c2015-07-25 12:10:21 +0300945 PyObject *func, *maxsize_O, *cache_info_type, *cachedict;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300946 int typed;
947 lru_cache_object *obj;
948 Py_ssize_t maxsize;
949 PyObject *(*wrapper)(lru_cache_object *, PyObject *, PyObject *);
950 static char *keywords[] = {"user_function", "maxsize", "typed",
951 "cache_info_type", NULL};
952
953 if (!PyArg_ParseTupleAndKeywords(args, kw, "OOpO:lru_cache", keywords,
954 &func, &maxsize_O, &typed,
955 &cache_info_type)) {
956 return NULL;
957 }
958
959 if (!PyCallable_Check(func)) {
960 PyErr_SetString(PyExc_TypeError,
961 "the first argument must be callable");
962 return NULL;
963 }
964
965 /* select the caching function, and make/inc maxsize_O */
966 if (maxsize_O == Py_None) {
967 wrapper = infinite_lru_cache_wrapper;
968 /* use this only to initialize lru_cache_object attribute maxsize */
969 maxsize = -1;
970 } else if (PyIndex_Check(maxsize_O)) {
971 maxsize = PyNumber_AsSsize_t(maxsize_O, PyExc_OverflowError);
972 if (maxsize == -1 && PyErr_Occurred())
973 return NULL;
974 if (maxsize == 0)
975 wrapper = uncached_lru_cache_wrapper;
976 else
977 wrapper = bounded_lru_cache_wrapper;
978 } else {
979 PyErr_SetString(PyExc_TypeError, "maxsize should be integer or None");
980 return NULL;
981 }
982
Serhiy Storchaka374164c2015-07-25 12:10:21 +0300983 if (!(cachedict = PyDict_New()))
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300984 return NULL;
985
Serhiy Storchaka374164c2015-07-25 12:10:21 +0300986 obj = (lru_cache_object *)type->tp_alloc(type, 0);
987 if (obj == NULL) {
988 Py_DECREF(cachedict);
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300989 return NULL;
990 }
991
Serhiy Storchaka374164c2015-07-25 12:10:21 +0300992 obj->cache = cachedict;
Serhiy Storchaka46c56112015-05-24 21:53:49 +0300993 obj->root.prev = &obj->root;
994 obj->root.next = &obj->root;
995 obj->maxsize = maxsize;
996 Py_INCREF(maxsize_O);
997 obj->maxsize_O = maxsize_O;
998 Py_INCREF(func);
999 obj->func = func;
1000 obj->wrapper = wrapper;
1001 obj->misses = obj->hits = 0;
1002 obj->typed = typed;
1003 Py_INCREF(cache_info_type);
1004 obj->cache_info_type = cache_info_type;
1005
1006 return (PyObject *)obj;
1007}
1008
1009static lru_list_elem *
1010lru_cache_unlink_list(lru_cache_object *self)
1011{
1012 lru_list_elem *root = &self->root;
1013 lru_list_elem *link = root->next;
1014 if (link == root)
1015 return NULL;
1016 root->prev->next = NULL;
1017 root->next = root->prev = root;
1018 return link;
1019}
1020
1021static void
1022lru_cache_clear_list(lru_list_elem *link)
1023{
1024 while (link != NULL) {
1025 lru_list_elem *next = link->next;
1026 Py_DECREF(link);
1027 link = next;
1028 }
1029}
1030
1031static void
1032lru_cache_dealloc(lru_cache_object *obj)
1033{
1034 lru_list_elem *list = lru_cache_unlink_list(obj);
1035 Py_XDECREF(obj->maxsize_O);
1036 Py_XDECREF(obj->func);
1037 Py_XDECREF(obj->cache);
1038 Py_XDECREF(obj->dict);
1039 Py_XDECREF(obj->cache_info_type);
1040 lru_cache_clear_list(list);
1041 Py_TYPE(obj)->tp_free(obj);
1042}
1043
1044static PyObject *
1045lru_cache_call(lru_cache_object *self, PyObject *args, PyObject *kwds)
1046{
1047 return self->wrapper(self, args, kwds);
1048}
1049
1050static PyObject *
Serhiy Storchakae7070f02015-06-08 11:19:24 +03001051lru_cache_descr_get(PyObject *self, PyObject *obj, PyObject *type)
1052{
1053 if (obj == Py_None || obj == NULL) {
1054 Py_INCREF(self);
1055 return self;
1056 }
1057 return PyMethod_New(self, obj);
1058}
1059
1060static PyObject *
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001061lru_cache_cache_info(lru_cache_object *self, PyObject *unused)
1062{
1063 return PyObject_CallFunction(self->cache_info_type, "nnOn",
1064 self->hits, self->misses, self->maxsize_O,
1065 PyDict_Size(self->cache));
1066}
1067
1068static PyObject *
1069lru_cache_cache_clear(lru_cache_object *self, PyObject *unused)
1070{
1071 lru_list_elem *list = lru_cache_unlink_list(self);
1072 self->hits = self->misses = 0;
1073 self->full = 0;
1074 PyDict_Clear(self->cache);
1075 lru_cache_clear_list(list);
1076 Py_RETURN_NONE;
1077}
1078
Serhiy Storchaka45120f22015-10-24 09:49:56 +03001079static PyObject *
1080lru_cache_reduce(PyObject *self, PyObject *unused)
1081{
1082 return PyObject_GetAttrString(self, "__qualname__");
1083}
1084
Serhiy Storchakae4d65e32015-12-28 23:58:07 +02001085static PyObject *
1086lru_cache_copy(PyObject *self, PyObject *unused)
1087{
1088 Py_INCREF(self);
1089 return self;
1090}
1091
1092static PyObject *
1093lru_cache_deepcopy(PyObject *self, PyObject *unused)
1094{
1095 Py_INCREF(self);
1096 return self;
1097}
1098
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001099static int
1100lru_cache_tp_traverse(lru_cache_object *self, visitproc visit, void *arg)
1101{
1102 lru_list_elem *link = self->root.next;
1103 while (link != &self->root) {
1104 lru_list_elem *next = link->next;
1105 Py_VISIT(link);
1106 link = next;
1107 }
1108 Py_VISIT(self->maxsize_O);
1109 Py_VISIT(self->func);
1110 Py_VISIT(self->cache);
1111 Py_VISIT(self->cache_info_type);
1112 Py_VISIT(self->dict);
1113 return 0;
1114}
1115
1116static int
1117lru_cache_tp_clear(lru_cache_object *self)
1118{
1119 lru_list_elem *list = lru_cache_unlink_list(self);
1120 Py_CLEAR(self->maxsize_O);
1121 Py_CLEAR(self->func);
1122 Py_CLEAR(self->cache);
1123 Py_CLEAR(self->cache_info_type);
1124 Py_CLEAR(self->dict);
1125 lru_cache_clear_list(list);
1126 return 0;
1127}
1128
1129
1130PyDoc_STRVAR(lru_cache_doc,
1131"Create a cached callable that wraps another function.\n\
1132\n\
1133user_function: the function being cached\n\
1134\n\
1135maxsize: 0 for no caching\n\
1136 None for unlimited cache size\n\
1137 n for a bounded cache\n\
1138\n\
1139typed: False cache f(3) and f(3.0) as identical calls\n\
1140 True cache f(3) and f(3.0) as distinct calls\n\
1141\n\
1142cache_info_type: namedtuple class with the fields:\n\
1143 hits misses currsize maxsize\n"
1144);
1145
1146static PyMethodDef lru_cache_methods[] = {
1147 {"cache_info", (PyCFunction)lru_cache_cache_info, METH_NOARGS},
1148 {"cache_clear", (PyCFunction)lru_cache_cache_clear, METH_NOARGS},
Serhiy Storchaka45120f22015-10-24 09:49:56 +03001149 {"__reduce__", (PyCFunction)lru_cache_reduce, METH_NOARGS},
Serhiy Storchakae4d65e32015-12-28 23:58:07 +02001150 {"__copy__", (PyCFunction)lru_cache_copy, METH_VARARGS},
1151 {"__deepcopy__", (PyCFunction)lru_cache_deepcopy, METH_VARARGS},
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001152 {NULL}
1153};
1154
1155static PyGetSetDef lru_cache_getsetlist[] = {
1156 {"__dict__", PyObject_GenericGetDict, PyObject_GenericSetDict},
1157 {NULL}
1158};
1159
1160static PyTypeObject lru_cache_type = {
1161 PyVarObject_HEAD_INIT(NULL, 0)
1162 "functools._lru_cache_wrapper", /* tp_name */
1163 sizeof(lru_cache_object), /* tp_basicsize */
1164 0, /* tp_itemsize */
1165 /* methods */
1166 (destructor)lru_cache_dealloc, /* tp_dealloc */
1167 0, /* tp_print */
1168 0, /* tp_getattr */
1169 0, /* tp_setattr */
1170 0, /* tp_reserved */
1171 0, /* tp_repr */
1172 0, /* tp_as_number */
1173 0, /* tp_as_sequence */
1174 0, /* tp_as_mapping */
1175 0, /* tp_hash */
1176 (ternaryfunc)lru_cache_call, /* tp_call */
1177 0, /* tp_str */
1178 0, /* tp_getattro */
1179 0, /* tp_setattro */
1180 0, /* tp_as_buffer */
1181 Py_TPFLAGS_DEFAULT|Py_TPFLAGS_BASETYPE|Py_TPFLAGS_HAVE_GC,
1182 /* tp_flags */
1183 lru_cache_doc, /* tp_doc */
1184 (traverseproc)lru_cache_tp_traverse,/* tp_traverse */
1185 (inquiry)lru_cache_tp_clear, /* tp_clear */
1186 0, /* tp_richcompare */
1187 0, /* tp_weaklistoffset */
1188 0, /* tp_iter */
1189 0, /* tp_iternext */
1190 lru_cache_methods, /* tp_methods */
1191 0, /* tp_members */
1192 lru_cache_getsetlist, /* tp_getset */
1193 0, /* tp_base */
1194 0, /* tp_dict */
Serhiy Storchakae7070f02015-06-08 11:19:24 +03001195 lru_cache_descr_get, /* tp_descr_get */
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001196 0, /* tp_descr_set */
1197 offsetof(lru_cache_object, dict), /* tp_dictoffset */
1198 0, /* tp_init */
1199 0, /* tp_alloc */
1200 lru_cache_new, /* tp_new */
1201};
1202
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001203/* module level code ********************************************************/
1204
1205PyDoc_STRVAR(module_doc,
Thomas Wouters4d70c3d2006-06-08 14:42:34 +00001206"Tools that operate on functions.");
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001207
1208static PyMethodDef module_methods[] = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001209 {"reduce", functools_reduce, METH_VARARGS, functools_reduce_doc},
Victor Stinner446c8d52011-04-05 12:21:35 +02001210 {"cmp_to_key", (PyCFunction)functools_cmp_to_key,
1211 METH_VARARGS | METH_KEYWORDS, functools_cmp_to_key_doc},
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001212 {NULL, NULL} /* sentinel */
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001213};
1214
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001215static void
1216module_free(void *m)
1217{
1218 Py_CLEAR(kwd_mark);
1219}
Martin v. Löwis1a214512008-06-11 05:26:20 +00001220
1221static struct PyModuleDef _functoolsmodule = {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001222 PyModuleDef_HEAD_INIT,
1223 "_functools",
1224 module_doc,
1225 -1,
1226 module_methods,
1227 NULL,
1228 NULL,
1229 NULL,
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001230 module_free,
Martin v. Löwis1a214512008-06-11 05:26:20 +00001231};
1232
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001233PyMODINIT_FUNC
Martin v. Löwis1a214512008-06-11 05:26:20 +00001234PyInit__functools(void)
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001235{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001236 int i;
1237 PyObject *m;
1238 char *name;
1239 PyTypeObject *typelist[] = {
1240 &partial_type,
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001241 &lru_cache_type,
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001242 NULL
1243 };
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001244
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001245 m = PyModule_Create(&_functoolsmodule);
1246 if (m == NULL)
1247 return NULL;
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001248
Serhiy Storchaka46c56112015-05-24 21:53:49 +03001249 kwd_mark = PyObject_CallObject((PyObject *)&PyBaseObject_Type, NULL);
1250 if (!kwd_mark) {
1251 Py_DECREF(m);
1252 return NULL;
1253 }
1254
Antoine Pitrouf95a1b32010-05-09 15:52:27 +00001255 for (i=0 ; typelist[i] != NULL ; i++) {
1256 if (PyType_Ready(typelist[i]) < 0) {
1257 Py_DECREF(m);
1258 return NULL;
1259 }
1260 name = strchr(typelist[i]->tp_name, '.');
1261 assert (name != NULL);
1262 Py_INCREF(typelist[i]);
1263 PyModule_AddObject(m, name+1, (PyObject *)typelist[i]);
1264 }
1265 return m;
Raymond Hettinger9c323f82005-02-28 19:39:44 +00001266}