1 /* Cell object implementation */
2
3 #include "Python.h"
4
5 PyObject *
PyCell_New(PyObject * obj)6 PyCell_New(PyObject *obj)
7 {
8 PyCellObject *op;
9
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);
15
16 _PyObject_GC_TRACK(op);
17 return (PyObject *)op;
18 }
19
20 PyObject *
PyCell_Get(PyObject * op)21 PyCell_Get(PyObject *op)
22 {
23 if (!PyCell_Check(op)) {
24 PyErr_BadInternalCall();
25 return NULL;
26 }
27 Py_XINCREF(((PyCellObject*)op)->ob_ref);
28 return PyCell_GET(op);
29 }
30
31 int
PyCell_Set(PyObject * op,PyObject * obj)32 PyCell_Set(PyObject *op, PyObject *obj)
33 {
34 PyObject* oldobj;
35 if (!PyCell_Check(op)) {
36 PyErr_BadInternalCall();
37 return -1;
38 }
39 oldobj = PyCell_GET(op);
40 Py_XINCREF(obj);
41 PyCell_SET(op, obj);
42 Py_XDECREF(oldobj);
43 return 0;
44 }
45
46 static void
cell_dealloc(PyCellObject * op)47 cell_dealloc(PyCellObject *op)
48 {
49 _PyObject_GC_UNTRACK(op);
50 Py_XDECREF(op->ob_ref);
51 PyObject_GC_Del(op);
52 }
53
54 static int
cell_compare(PyCellObject * a,PyCellObject * b)55 cell_compare(PyCellObject *a, PyCellObject *b)
56 {
57 /* Py3K warning for comparisons */
58 if (PyErr_WarnPy3k("cell comparisons not supported in 3.x",
59 1) < 0) {
60 return -2;
61 }
62
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);
70 }
71
72 static PyObject *
cell_repr(PyCellObject * op)73 cell_repr(PyCellObject *op)
74 {
75 if (op->ob_ref == NULL)
76 return PyString_FromFormat("<cell at %p: empty>", op);
77
78 return PyString_FromFormat("<cell at %p: %.80s object at %p>",
79 op, op->ob_ref->ob_type->tp_name,
80 op->ob_ref);
81 }
82
83 static int
cell_traverse(PyCellObject * op,visitproc visit,void * arg)84 cell_traverse(PyCellObject *op, visitproc visit, void *arg)
85 {
86 Py_VISIT(op->ob_ref);
87 return 0;
88 }
89
90 static int
cell_clear(PyCellObject * op)91 cell_clear(PyCellObject *op)
92 {
93 Py_CLEAR(op->ob_ref);
94 return 0;
95 }
96
97 static PyObject *
cell_get_contents(PyCellObject * op,void * closure)98 cell_get_contents(PyCellObject *op, void *closure)
99 {
100 if (op->ob_ref == NULL)
101 {
102 PyErr_SetString(PyExc_ValueError, "Cell is empty");
103 return NULL;
104 }
105 Py_INCREF(op->ob_ref);
106 return op->ob_ref;
107 }
108
109 static PyGetSetDef cell_getsetlist[] = {
110 {"cell_contents", (getter)cell_get_contents, NULL},
111 {NULL} /* sentinel */
112 };
113
114 PyTypeObject PyCell_Type = {
115 PyVarObject_HEAD_INIT(&PyType_Type, 0)
116 "cell",
117 sizeof(PyCellObject),
118 0,
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 */
145 };
146