1
2 /* Use this file as a template to start implementing a module that
3 also declares object types. All occurrences of 'Xxo' should be changed
4 to something reasonable for your objects. After that, all other
5 occurrences of 'xx' should be changed to something reasonable for your
6 module. If your module is named foo your sourcefile should be named
7 foomodule.c.
8
9 You will probably want to delete all references to 'x_attr' and add
10 your own types of attributes instead. Maybe you want to name your
11 local variables other than 'self'. If your object type is needed in
12 other files, you'll have to create a file "foobarobject.h"; see
13 floatobject.h for an example. */
14
15 /* Xxo objects */
16
17 #include "Python.h"
18
19 static PyObject *ErrorObject;
20
21 typedef struct {
22 PyObject_HEAD
23 PyObject *x_attr; /* Attributes dictionary */
24 } XxoObject;
25
26 static PyTypeObject Xxo_Type;
27
28 #define XxoObject_Check(v) (Py_TYPE(v) == &Xxo_Type)
29
30 static XxoObject *
newXxoObject(PyObject * arg)31 newXxoObject(PyObject *arg)
32 {
33 XxoObject *self;
34 self = PyObject_New(XxoObject, &Xxo_Type);
35 if (self == NULL)
36 return NULL;
37 self->x_attr = NULL;
38 return self;
39 }
40
41 /* Xxo methods */
42
43 static void
Xxo_dealloc(XxoObject * self)44 Xxo_dealloc(XxoObject *self)
45 {
46 Py_XDECREF(self->x_attr);
47 PyObject_Del(self);
48 }
49
50 static PyObject *
Xxo_demo(XxoObject * self,PyObject * args)51 Xxo_demo(XxoObject *self, PyObject *args)
52 {
53 if (!PyArg_ParseTuple(args, ":demo"))
54 return NULL;
55 Py_INCREF(Py_None);
56 return Py_None;
57 }
58
59 static PyMethodDef Xxo_methods[] = {
60 {"demo", (PyCFunction)Xxo_demo, METH_VARARGS,
61 PyDoc_STR("demo() -> None")},
62 {NULL, NULL} /* sentinel */
63 };
64
65 static PyObject *
Xxo_getattro(XxoObject * self,PyObject * name)66 Xxo_getattro(XxoObject *self, PyObject *name)
67 {
68 if (self->x_attr != NULL) {
69 PyObject *v = PyDict_GetItemWithError(self->x_attr, name);
70 if (v != NULL) {
71 Py_INCREF(v);
72 return v;
73 }
74 else if (PyErr_Occurred()) {
75 return NULL;
76 }
77 }
78 return PyObject_GenericGetAttr((PyObject *)self, name);
79 }
80
81 static int
Xxo_setattr(XxoObject * self,const char * name,PyObject * v)82 Xxo_setattr(XxoObject *self, const char *name, PyObject *v)
83 {
84 if (self->x_attr == NULL) {
85 self->x_attr = PyDict_New();
86 if (self->x_attr == NULL)
87 return -1;
88 }
89 if (v == NULL) {
90 int rv = PyDict_DelItemString(self->x_attr, name);
91 if (rv < 0 && PyErr_ExceptionMatches(PyExc_KeyError))
92 PyErr_SetString(PyExc_AttributeError,
93 "delete non-existing Xxo attribute");
94 return rv;
95 }
96 else
97 return PyDict_SetItemString(self->x_attr, name, v);
98 }
99
100 static PyTypeObject Xxo_Type = {
101 /* The ob_type field must be initialized in the module init function
102 * to be portable to Windows without using C++. */
103 PyVarObject_HEAD_INIT(NULL, 0)
104 "xxmodule.Xxo", /*tp_name*/
105 sizeof(XxoObject), /*tp_basicsize*/
106 0, /*tp_itemsize*/
107 /* methods */
108 (destructor)Xxo_dealloc, /*tp_dealloc*/
109 0, /*tp_vectorcall_offset*/
110 (getattrfunc)0, /*tp_getattr*/
111 (setattrfunc)Xxo_setattr, /*tp_setattr*/
112 0, /*tp_as_async*/
113 0, /*tp_repr*/
114 0, /*tp_as_number*/
115 0, /*tp_as_sequence*/
116 0, /*tp_as_mapping*/
117 0, /*tp_hash*/
118 0, /*tp_call*/
119 0, /*tp_str*/
120 (getattrofunc)Xxo_getattro, /*tp_getattro*/
121 0, /*tp_setattro*/
122 0, /*tp_as_buffer*/
123 Py_TPFLAGS_DEFAULT, /*tp_flags*/
124 0, /*tp_doc*/
125 0, /*tp_traverse*/
126 0, /*tp_clear*/
127 0, /*tp_richcompare*/
128 0, /*tp_weaklistoffset*/
129 0, /*tp_iter*/
130 0, /*tp_iternext*/
131 Xxo_methods, /*tp_methods*/
132 0, /*tp_members*/
133 0, /*tp_getset*/
134 0, /*tp_base*/
135 0, /*tp_dict*/
136 0, /*tp_descr_get*/
137 0, /*tp_descr_set*/
138 0, /*tp_dictoffset*/
139 0, /*tp_init*/
140 0, /*tp_alloc*/
141 0, /*tp_new*/
142 0, /*tp_free*/
143 0, /*tp_is_gc*/
144 };
145 /* --------------------------------------------------------------------- */
146
147 /* Function of two integers returning integer */
148
149 PyDoc_STRVAR(xx_foo_doc,
150 "foo(i,j)\n\
151 \n\
152 Return the sum of i and j.");
153
154 static PyObject *
xx_foo(PyObject * self,PyObject * args)155 xx_foo(PyObject *self, PyObject *args)
156 {
157 long i, j;
158 long res;
159 if (!PyArg_ParseTuple(args, "ll:foo", &i, &j))
160 return NULL;
161 res = i+j; /* XXX Do something here */
162 return PyLong_FromLong(res);
163 }
164
165
166 /* Function of no arguments returning new Xxo object */
167
168 static PyObject *
xx_new(PyObject * self,PyObject * args)169 xx_new(PyObject *self, PyObject *args)
170 {
171 XxoObject *rv;
172
173 if (!PyArg_ParseTuple(args, ":new"))
174 return NULL;
175 rv = newXxoObject(args);
176 if (rv == NULL)
177 return NULL;
178 return (PyObject *)rv;
179 }
180
181 /* Example with subtle bug from extensions manual ("Thin Ice"). */
182
183 static PyObject *
xx_bug(PyObject * self,PyObject * args)184 xx_bug(PyObject *self, PyObject *args)
185 {
186 PyObject *list, *item;
187
188 if (!PyArg_ParseTuple(args, "O:bug", &list))
189 return NULL;
190
191 item = PyList_GetItem(list, 0);
192 /* Py_INCREF(item); */
193 PyList_SetItem(list, 1, PyLong_FromLong(0L));
194 PyObject_Print(item, stdout, 0);
195 printf("\n");
196 /* Py_DECREF(item); */
197
198 Py_INCREF(Py_None);
199 return Py_None;
200 }
201
202 /* Test bad format character */
203
204 static PyObject *
xx_roj(PyObject * self,PyObject * args)205 xx_roj(PyObject *self, PyObject *args)
206 {
207 PyObject *a;
208 long b;
209 if (!PyArg_ParseTuple(args, "O#:roj", &a, &b))
210 return NULL;
211 Py_INCREF(Py_None);
212 return Py_None;
213 }
214
215
216 /* ---------- */
217
218 static PyTypeObject Str_Type = {
219 /* The ob_type field must be initialized in the module init function
220 * to be portable to Windows without using C++. */
221 PyVarObject_HEAD_INIT(NULL, 0)
222 "xxmodule.Str", /*tp_name*/
223 0, /*tp_basicsize*/
224 0, /*tp_itemsize*/
225 /* methods */
226 0, /*tp_dealloc*/
227 0, /*tp_vectorcall_offset*/
228 0, /*tp_getattr*/
229 0, /*tp_setattr*/
230 0, /*tp_as_async*/
231 0, /*tp_repr*/
232 0, /*tp_as_number*/
233 0, /*tp_as_sequence*/
234 0, /*tp_as_mapping*/
235 0, /*tp_hash*/
236 0, /*tp_call*/
237 0, /*tp_str*/
238 0, /*tp_getattro*/
239 0, /*tp_setattro*/
240 0, /*tp_as_buffer*/
241 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/
242 0, /*tp_doc*/
243 0, /*tp_traverse*/
244 0, /*tp_clear*/
245 0, /*tp_richcompare*/
246 0, /*tp_weaklistoffset*/
247 0, /*tp_iter*/
248 0, /*tp_iternext*/
249 0, /*tp_methods*/
250 0, /*tp_members*/
251 0, /*tp_getset*/
252 0, /* see PyInit_xx */ /*tp_base*/
253 0, /*tp_dict*/
254 0, /*tp_descr_get*/
255 0, /*tp_descr_set*/
256 0, /*tp_dictoffset*/
257 0, /*tp_init*/
258 0, /*tp_alloc*/
259 0, /*tp_new*/
260 0, /*tp_free*/
261 0, /*tp_is_gc*/
262 };
263
264 /* ---------- */
265
266 static PyObject *
null_richcompare(PyObject * self,PyObject * other,int op)267 null_richcompare(PyObject *self, PyObject *other, int op)
268 {
269 Py_INCREF(Py_NotImplemented);
270 return Py_NotImplemented;
271 }
272
273 static PyTypeObject Null_Type = {
274 /* The ob_type field must be initialized in the module init function
275 * to be portable to Windows without using C++. */
276 PyVarObject_HEAD_INIT(NULL, 0)
277 "xxmodule.Null", /*tp_name*/
278 0, /*tp_basicsize*/
279 0, /*tp_itemsize*/
280 /* methods */
281 0, /*tp_dealloc*/
282 0, /*tp_vectorcall_offset*/
283 0, /*tp_getattr*/
284 0, /*tp_setattr*/
285 0, /*tp_as_async*/
286 0, /*tp_repr*/
287 0, /*tp_as_number*/
288 0, /*tp_as_sequence*/
289 0, /*tp_as_mapping*/
290 0, /*tp_hash*/
291 0, /*tp_call*/
292 0, /*tp_str*/
293 0, /*tp_getattro*/
294 0, /*tp_setattro*/
295 0, /*tp_as_buffer*/
296 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/
297 0, /*tp_doc*/
298 0, /*tp_traverse*/
299 0, /*tp_clear*/
300 null_richcompare, /*tp_richcompare*/
301 0, /*tp_weaklistoffset*/
302 0, /*tp_iter*/
303 0, /*tp_iternext*/
304 0, /*tp_methods*/
305 0, /*tp_members*/
306 0, /*tp_getset*/
307 0, /* see PyInit_xx */ /*tp_base*/
308 0, /*tp_dict*/
309 0, /*tp_descr_get*/
310 0, /*tp_descr_set*/
311 0, /*tp_dictoffset*/
312 0, /*tp_init*/
313 0, /*tp_alloc*/
314 PyType_GenericNew, /*tp_new*/
315 0, /*tp_free*/
316 0, /*tp_is_gc*/
317 };
318
319
320 /* ---------- */
321
322
323 /* List of functions defined in the module */
324
325 static PyMethodDef xx_methods[] = {
326 {"roj", xx_roj, METH_VARARGS,
327 PyDoc_STR("roj(a,b) -> None")},
328 {"foo", xx_foo, METH_VARARGS,
329 xx_foo_doc},
330 {"new", xx_new, METH_VARARGS,
331 PyDoc_STR("new() -> new Xx object")},
332 {"bug", xx_bug, METH_VARARGS,
333 PyDoc_STR("bug(o) -> None")},
334 {NULL, NULL} /* sentinel */
335 };
336
337 PyDoc_STRVAR(module_doc,
338 "This is a template module just for instruction.");
339
340
341 static int
xx_exec(PyObject * m)342 xx_exec(PyObject *m)
343 {
344 /* Slot initialization is subject to the rules of initializing globals.
345 C99 requires the initializers to be "address constants". Function
346 designators like 'PyType_GenericNew', with implicit conversion to
347 a pointer, are valid C99 address constants.
348
349 However, the unary '&' operator applied to a non-static variable
350 like 'PyBaseObject_Type' is not required to produce an address
351 constant. Compilers may support this (gcc does), MSVC does not.
352
353 Both compilers are strictly standard conforming in this particular
354 behavior.
355 */
356 Null_Type.tp_base = &PyBaseObject_Type;
357 Str_Type.tp_base = &PyUnicode_Type;
358
359 /* Finalize the type object including setting type of the new type
360 * object; doing it here is required for portability, too. */
361 if (PyType_Ready(&Xxo_Type) < 0)
362 goto fail;
363
364 /* Add some symbolic constants to the module */
365 if (ErrorObject == NULL) {
366 ErrorObject = PyErr_NewException("xx.error", NULL, NULL);
367 if (ErrorObject == NULL)
368 goto fail;
369 }
370 Py_INCREF(ErrorObject);
371 PyModule_AddObject(m, "error", ErrorObject);
372
373 /* Add Str */
374 if (PyType_Ready(&Str_Type) < 0)
375 goto fail;
376 PyModule_AddObject(m, "Str", (PyObject *)&Str_Type);
377
378 /* Add Null */
379 if (PyType_Ready(&Null_Type) < 0)
380 goto fail;
381 PyModule_AddObject(m, "Null", (PyObject *)&Null_Type);
382 return 0;
383 fail:
384 Py_XDECREF(m);
385 return -1;
386 }
387
388 static struct PyModuleDef_Slot xx_slots[] = {
389 {Py_mod_exec, xx_exec},
390 {0, NULL},
391 };
392
393 static struct PyModuleDef xxmodule = {
394 PyModuleDef_HEAD_INIT,
395 "xx",
396 module_doc,
397 0,
398 xx_methods,
399 xx_slots,
400 NULL,
401 NULL,
402 NULL
403 };
404
405 /* Export function for the module (*must* be called PyInit_xx) */
406
407 PyMODINIT_FUNC
PyInit_xx(void)408 PyInit_xx(void)
409 {
410 return PyModuleDef_Init(&xxmodule);
411 }
412