Add better error reporting for MemoryErrors caused by str->float conversions.
[python.git] / Objects / cellobject.c
blob4e0bcf8833ed6c5260fe23ab70cafbdd501d2e60
1 /* Cell object implementation */
3 #include "Python.h"
5 PyObject *
6 PyCell_New(PyObject *obj)
8 PyCellObject *op;
10 op = (PyCellObject *)PyObject_GC_New(PyCellObject, &PyCell_Type);
11 if (op == NULL)
12 return NULL;
13 op->ob_ref = obj;
14 Py_XINCREF(obj);
16 _PyObject_GC_TRACK(op);
17 return (PyObject *)op;
20 PyObject *
21 PyCell_Get(PyObject *op)
23 if (!PyCell_Check(op)) {
24 PyErr_BadInternalCall();
25 return NULL;
27 Py_XINCREF(((PyCellObject*)op)->ob_ref);
28 return PyCell_GET(op);
31 int
32 PyCell_Set(PyObject *op, PyObject *obj)
34 PyObject* oldobj;
35 if (!PyCell_Check(op)) {
36 PyErr_BadInternalCall();
37 return -1;
39 oldobj = PyCell_GET(op);
40 Py_XINCREF(obj);
41 PyCell_SET(op, obj);
42 Py_XDECREF(oldobj);
43 return 0;
46 static void
47 cell_dealloc(PyCellObject *op)
49 _PyObject_GC_UNTRACK(op);
50 Py_XDECREF(op->ob_ref);
51 PyObject_GC_Del(op);
54 static int
55 cell_compare(PyCellObject *a, PyCellObject *b)
57 /* Py3K warning for comparisons */
58 if (PyErr_WarnPy3k("cell comparisons not supported in 3.x",
59 1) < 0) {
60 return -2;
63 if (a->ob_ref == NULL) {
64 if (b->ob_ref == NULL)
65 return 0;
66 return -1;
67 } else if (b->ob_ref == NULL)
68 return 1;
69 return PyObject_Compare(a->ob_ref, b->ob_ref);
72 static PyObject *
73 cell_repr(PyCellObject *op)
75 if (op->ob_ref == NULL)
76 return PyString_FromFormat("<cell at %p: empty>", op);
78 return PyString_FromFormat("<cell at %p: %.80s object at %p>",
79 op, op->ob_ref->ob_type->tp_name,
80 op->ob_ref);
83 static int
84 cell_traverse(PyCellObject *op, visitproc visit, void *arg)
86 Py_VISIT(op->ob_ref);
87 return 0;
90 static int
91 cell_clear(PyCellObject *op)
93 Py_CLEAR(op->ob_ref);
94 return 0;
97 static PyObject *
98 cell_get_contents(PyCellObject *op, void *closure)
100 if (op->ob_ref == NULL)
102 PyErr_SetString(PyExc_ValueError, "Cell is empty");
103 return NULL;
105 Py_INCREF(op->ob_ref);
106 return op->ob_ref;
109 static PyGetSetDef cell_getsetlist[] = {
110 {"cell_contents", (getter)cell_get_contents, NULL},
111 {NULL} /* sentinel */
114 PyTypeObject PyCell_Type = {
115 PyVarObject_HEAD_INIT(&PyType_Type, 0)
116 "cell",
117 sizeof(PyCellObject),
119 (destructor)cell_dealloc, /* tp_dealloc */
120 0, /* tp_print */
121 0, /* tp_getattr */
122 0, /* tp_setattr */
123 (cmpfunc)cell_compare, /* tp_compare */
124 (reprfunc)cell_repr, /* tp_repr */
125 0, /* tp_as_number */
126 0, /* tp_as_sequence */
127 0, /* tp_as_mapping */
128 0, /* tp_hash */
129 0, /* tp_call */
130 0, /* tp_str */
131 PyObject_GenericGetAttr, /* tp_getattro */
132 0, /* tp_setattro */
133 0, /* tp_as_buffer */
134 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,/* tp_flags */
135 0, /* tp_doc */
136 (traverseproc)cell_traverse, /* tp_traverse */
137 (inquiry)cell_clear, /* tp_clear */
138 0, /* tp_richcompare */
139 0, /* tp_weaklistoffset */
140 0, /* tp_iter */
141 0, /* tp_iternext */
142 0, /* tp_methods */
143 0, /* tp_members */
144 cell_getsetlist, /* tp_getset */