structseq.c 13.6 KB
Newer Older
1 2 3 4 5 6
/* Implementation helper: a struct that looks like a tuple.  See timemodule
   and posixmodule for example uses. */

#include "Python.h"
#include "structmember.h"

7 8 9
static const char visible_length_key[] = "n_sequence_fields";
static const char real_length_key[] = "n_fields";
static const char unnamed_fields_key[] = "n_unnamed_fields";
10

11
/* Fields with this name have only a field index, not a field name.
12 13
   They are only allowed for indices < n_visible_fields. */
char *PyStructSequence_UnnamedField = "unnamed field";
14 15 16
_Py_IDENTIFIER(n_sequence_fields);
_Py_IDENTIFIER(n_fields);
_Py_IDENTIFIER(n_unnamed_fields);
17

18
#define VISIBLE_SIZE(op) Py_SIZE(op)
19
#define VISIBLE_SIZE_TP(tp) PyLong_AsSsize_t( \
20
                      _PyDict_GetItemId((tp)->tp_dict, &PyId_n_sequence_fields))
21

22
#define REAL_SIZE_TP(tp) PyLong_AsSsize_t( \
23
                      _PyDict_GetItemId((tp)->tp_dict, &PyId_n_fields))
24
#define REAL_SIZE(op) REAL_SIZE_TP(Py_TYPE(op))
25

26
#define UNNAMED_FIELDS_TP(tp) PyLong_AsSsize_t( \
27
                      _PyDict_GetItemId((tp)->tp_dict, &PyId_n_unnamed_fields))
28
#define UNNAMED_FIELDS(op) UNNAMED_FIELDS_TP(Py_TYPE(op))
29

30 31 32 33

PyObject *
PyStructSequence_New(PyTypeObject *type)
{
34
    PyStructSequence *obj;
35
    Py_ssize_t size = REAL_SIZE_TP(type), i;
36

37
    obj = PyObject_GC_NewVar(PyStructSequence, type, size);
38 39
    if (obj == NULL)
        return NULL;
40 41 42
    /* Hack the size of the variable object, so invisible fields don't appear
     to Python code. */
    Py_SIZE(obj) = VISIBLE_SIZE_TP(type);
43 44
    for (i = 0; i < size; i++)
        obj->ob_item[i] = NULL;
45

46
    return (PyObject*)obj;
47 48
}

49 50 51 52 53 54 55 56 57 58 59 60
void
PyStructSequence_SetItem(PyObject* op, Py_ssize_t i, PyObject* v)
{
    PyStructSequence_SET_ITEM(op, i, v);
}

PyObject*
PyStructSequence_GetItem(PyObject* op, Py_ssize_t i)
{
    return PyStructSequence_GET_ITEM(op, i);
}

61 62 63 64
static void
structseq_dealloc(PyStructSequence *obj)
{
    Py_ssize_t i, size;
65

66 67 68 69 70 71 72
    size = REAL_SIZE(obj);
    for (i = 0; i < size; ++i) {
        Py_XDECREF(obj->ob_item[i]);
    }
    PyObject_GC_Del(obj);
}

73 74 75 76 77 78 79 80 81 82 83 84 85 86
/*[clinic input]
class structseq "PyStructSequence *" "NULL"
[clinic start generated code]*/
/*[clinic end generated code: output=da39a3ee5e6b4b0d input=9d781c6922c77752]*/

#include "clinic/structseq.c.h"

/*[clinic input]
@classmethod
structseq.__new__ as structseq_new
    sequence as arg: object
    dict: object = NULL
[clinic start generated code]*/

87
static PyObject *
88 89
structseq_new_impl(PyTypeObject *type, PyObject *arg, PyObject *dict)
/*[clinic end generated code: output=baa082e788b171da input=9b44810243907377]*/
90
{
91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
    PyObject *ob;
    PyStructSequence *res = NULL;
    Py_ssize_t len, min_len, max_len, i, n_unnamed_fields;

    arg = PySequence_Fast(arg, "constructor requires a sequence");

    if (!arg) {
        return NULL;
    }

    if (dict && !PyDict_Check(dict)) {
        PyErr_Format(PyExc_TypeError,
                     "%.500s() takes a dict as second arg, if any",
                     type->tp_name);
        Py_DECREF(arg);
        return NULL;
    }

    len = PySequence_Fast_GET_SIZE(arg);
    min_len = VISIBLE_SIZE_TP(type);
    max_len = REAL_SIZE_TP(type);
    n_unnamed_fields = UNNAMED_FIELDS_TP(type);

    if (min_len != max_len) {
        if (len < min_len) {
            PyErr_Format(PyExc_TypeError,
Antoine Pitrou's avatar
Antoine Pitrou committed
117 118 119 120
                "%.500s() takes an at least %zd-sequence (%zd-sequence given)",
                type->tp_name, min_len, len);
            Py_DECREF(arg);
            return NULL;
121 122 123 124
        }

        if (len > max_len) {
            PyErr_Format(PyExc_TypeError,
Antoine Pitrou's avatar
Antoine Pitrou committed
125 126 127 128
                "%.500s() takes an at most %zd-sequence (%zd-sequence given)",
                type->tp_name, max_len, len);
            Py_DECREF(arg);
            return NULL;
129 130 131 132 133
        }
    }
    else {
        if (len != min_len) {
            PyErr_Format(PyExc_TypeError,
Antoine Pitrou's avatar
Antoine Pitrou committed
134 135 136 137
                         "%.500s() takes a %zd-sequence (%zd-sequence given)",
                         type->tp_name, min_len, len);
            Py_DECREF(arg);
            return NULL;
138 139 140 141 142
        }
    }

    res = (PyStructSequence*) PyStructSequence_New(type);
    if (res == NULL) {
143
        Py_DECREF(arg);
144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
        return NULL;
    }
    for (i = 0; i < len; ++i) {
        PyObject *v = PySequence_Fast_GET_ITEM(arg, i);
        Py_INCREF(v);
        res->ob_item[i] = v;
    }
    for (; i < max_len; ++i) {
        if (dict && (ob = PyDict_GetItemString(
            dict, type->tp_members[i-n_unnamed_fields].name))) {
        }
        else {
            ob = Py_None;
        }
        Py_INCREF(ob);
        res->ob_item[i] = ob;
    }

    Py_DECREF(arg);
    return (PyObject*) res;
164 165 166 167 168 169
}


static PyObject *
structseq_repr(PyStructSequence *obj)
{
170
    /* buffer and type size were chosen well considered. */
171 172 173
#define REPR_BUFFER_SIZE 512
#define TYPE_MAXSIZE 100

174
    PyTypeObject *typ = Py_TYPE(obj);
175 176
    Py_ssize_t i;
    int removelast = 0;
177 178 179 180 181 182 183 184
    Py_ssize_t len;
    char buf[REPR_BUFFER_SIZE];
    char *endofbuf, *pbuf = buf;

    /* pointer to end of writeable buffer; safes space for "...)\0" */
    endofbuf= &buf[REPR_BUFFER_SIZE-5];

    /* "typename(", limited to  TYPE_MAXSIZE */
185 186 187
    len = strlen(typ->tp_name);
    len = Py_MIN(len, TYPE_MAXSIZE);
    memcpy(pbuf, typ->tp_name, len);
188 189 190 191 192
    pbuf += len;
    *pbuf++ = '(';

    for (i=0; i < VISIBLE_SIZE(obj); i++) {
        PyObject *val, *repr;
193
        const char *cname, *crepr;
194 195

        cname = typ->tp_members[i].name;
196 197 198
        if (cname == NULL) {
            PyErr_Format(PyExc_SystemError, "In structseq_repr(), member %d name is NULL"
                         " for type %.500s", i, typ->tp_name);
199
            return NULL;
200
        }
201
        val = PyStructSequence_GET_ITEM(obj, i);
202
        repr = PyObject_Repr(val);
203
        if (repr == NULL)
204
            return NULL;
205
        crepr = PyUnicode_AsUTF8(repr);
206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239
        if (crepr == NULL) {
            Py_DECREF(repr);
            return NULL;
        }

        /* + 3: keep space for "=" and ", " */
        len = strlen(cname) + strlen(crepr) + 3;
        if ((pbuf+len) <= endofbuf) {
            strcpy(pbuf, cname);
            pbuf += strlen(cname);
            *pbuf++ = '=';
            strcpy(pbuf, crepr);
            pbuf += strlen(crepr);
            *pbuf++ = ',';
            *pbuf++ = ' ';
            removelast = 1;
            Py_DECREF(repr);
        }
        else {
            strcpy(pbuf, "...");
            pbuf += 3;
            removelast = 0;
            Py_DECREF(repr);
            break;
        }
    }
    if (removelast) {
        /* overwrite last ", " */
        pbuf-=2;
    }
    *pbuf++ = ')';
    *pbuf = '\0';

    return PyUnicode_FromString(buf);
240 241
}

Michael W. Hudson's avatar
Michael W. Hudson committed
242
static PyObject *
243
structseq_reduce(PyStructSequence* self, PyObject *Py_UNUSED(ignored))
Michael W. Hudson's avatar
Michael W. Hudson committed
244
{
245 246
    PyObject* tup = NULL;
    PyObject* dict = NULL;
247
    PyObject* result;
248
    Py_ssize_t n_fields, n_visible_fields, n_unnamed_fields, i;
249 250 251 252 253

    n_fields = REAL_SIZE(self);
    n_visible_fields = VISIBLE_SIZE(self);
    n_unnamed_fields = UNNAMED_FIELDS(self);
    tup = PyTuple_New(n_visible_fields);
254 255
    if (!tup)
        goto error;
256 257

    dict = PyDict_New();
258 259
    if (!dict)
        goto error;
260 261 262 263 264 265 266

    for (i = 0; i < n_visible_fields; i++) {
        Py_INCREF(self->ob_item[i]);
        PyTuple_SET_ITEM(tup, i, self->ob_item[i]);
    }

    for (; i < n_fields; i++) {
267
        const char *n = Py_TYPE(self)->tp_members[i-n_unnamed_fields].name;
268 269
        if (PyDict_SetItemString(dict, n, self->ob_item[i]) < 0)
            goto error;
270 271 272 273 274 275 276 277
    }

    result = Py_BuildValue("(O(OO))", Py_TYPE(self), tup, dict);

    Py_DECREF(tup);
    Py_DECREF(dict);

    return result;
278 279 280 281 282

error:
    Py_XDECREF(tup);
    Py_XDECREF(dict);
    return NULL;
Michael W. Hudson's avatar
Michael W. Hudson committed
283 284 285
}

static PyMethodDef structseq_methods[] = {
286
    {"__reduce__", (PyCFunction)structseq_reduce, METH_NOARGS, NULL},
287
    {NULL, NULL}
Michael W. Hudson's avatar
Michael W. Hudson committed
288 289
};

290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348
static Py_ssize_t
count_members(PyStructSequence_Desc *desc, Py_ssize_t *n_unnamed_members) {
    Py_ssize_t i;

    *n_unnamed_members = 0;
    for (i = 0; desc->fields[i].name != NULL; ++i) {
        if (desc->fields[i].name == PyStructSequence_UnnamedField) {
            (*n_unnamed_members)++;
        }
    }
    return i;
}

static int
initialize_structseq_dict(PyStructSequence_Desc *desc, PyObject* dict,
                          Py_ssize_t n_members, Py_ssize_t n_unnamed_members) {
    PyObject *v;

#define SET_DICT_FROM_SIZE(key, value)                                         \
    do {                                                                       \
        v = PyLong_FromSsize_t(value);                                         \
        if (v == NULL) {                                                       \
            return -1;                                                         \
        }                                                                      \
        if (PyDict_SetItemString(dict, key, v) < 0) {                          \
            Py_DECREF(v);                                                      \
            return -1;                                                         \
        }                                                                      \
        Py_DECREF(v);                                                          \
    } while (0)

    SET_DICT_FROM_SIZE(visible_length_key, desc->n_in_sequence);
    SET_DICT_FROM_SIZE(real_length_key, n_members);
    SET_DICT_FROM_SIZE(unnamed_fields_key, n_unnamed_members);
    return 0;
}

static void
initialize_members(PyStructSequence_Desc *desc, PyMemberDef* members,
                   Py_ssize_t n_members) {
    Py_ssize_t i, k;

    for (i = k = 0; i < n_members; ++i) {
        if (desc->fields[i].name == PyStructSequence_UnnamedField) {
            continue;
        }

        /* The names and docstrings in these MemberDefs are statically */
        /* allocated so it is expected that they'll outlive the MemberDef */
        members[k].name = desc->fields[i].name;
        members[k].type = T_OBJECT;
        members[k].offset = offsetof(PyStructSequence, ob_item)
          + i * sizeof(PyObject*);
        members[k].flags = READONLY;
        members[k].doc = desc->fields[i].doc;
        k++;
    }
    members[k].name = NULL;
}
349

350 351
int
PyStructSequence_InitType2(PyTypeObject *type, PyStructSequence_Desc *desc)
352
{
353 354
    PyMemberDef *members;
    Py_ssize_t n_members, n_unnamed_members;
355

356
#ifdef Py_TRACE_REFS
357 358 359
    /* if the type object was chained, unchain it first
       before overwriting its storage */
    if (type->ob_base.ob_base._ob_next) {
360
        _Py_ForgetReference((PyObject *)type);
361
    }
362 363
#endif

364 365 366 367 368
    /* PyTypeObject has already been initialized */
    if (Py_REFCNT(type) != 0) {
        PyErr_BadInternalCall();
        return -1;
    }
369 370

    type->tp_name = desc->name;
371 372 373 374
    type->tp_basicsize = sizeof(PyStructSequence) - sizeof(PyObject *);
    type->tp_itemsize = sizeof(PyObject *);
    type->tp_dealloc = (destructor)structseq_dealloc;
    type->tp_repr = (reprfunc)structseq_repr;
375
    type->tp_doc = desc->doc;
376 377 378 379
    type->tp_base = &PyTuple_Type;
    type->tp_methods = structseq_methods;
    type->tp_new = structseq_new;
    type->tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC;
380

381 382
    n_members = count_members(desc, &n_unnamed_members);
    members = PyMem_NEW(PyMemberDef, n_members - n_unnamed_members + 1);
383 384 385 386
    if (members == NULL) {
        PyErr_NoMemory();
        return -1;
    }
387
    initialize_members(desc, members, n_members);
388 389
    type->tp_members = members;

390 391
    if (PyType_Ready(type) < 0) {
        PyMem_FREE(members);
392
        return -1;
393
    }
394 395
    Py_INCREF(type);

396 397 398 399 400 401
    if (initialize_structseq_dict(
            desc, type->tp_dict, n_members, n_unnamed_members) < 0) {
        PyMem_FREE(members);
        Py_DECREF(type);
        return -1;
    }
402 403 404 405 406 407 408 409

    return 0;
}

void
PyStructSequence_InitType(PyTypeObject *type, PyStructSequence_Desc *desc)
{
    (void)PyStructSequence_InitType2(type, desc);
410
}
411

412
PyTypeObject *
413 414
PyStructSequence_NewType(PyStructSequence_Desc *desc)
{
415 416 417 418 419 420 421 422 423 424 425 426
    PyMemberDef *members;
    PyObject *bases;
    PyTypeObject *type;
    PyType_Slot slots[7];
    PyType_Spec spec;
    Py_ssize_t n_members, n_unnamed_members;

    /* Initialize MemberDefs */
    n_members = count_members(desc, &n_unnamed_members);
    members = PyMem_NEW(PyMemberDef, n_members - n_unnamed_members + 1);
    if (members == NULL) {
        PyErr_NoMemory();
427
        return NULL;
428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449
    }
    initialize_members(desc, members, n_members);

    /* Initialize Slots */
    slots[0] = (PyType_Slot){Py_tp_dealloc, (destructor)structseq_dealloc};
    slots[1] = (PyType_Slot){Py_tp_repr, (reprfunc)structseq_repr};
    slots[2] = (PyType_Slot){Py_tp_doc, (void *)desc->doc};
    slots[3] = (PyType_Slot){Py_tp_methods, structseq_methods};
    slots[4] = (PyType_Slot){Py_tp_new, structseq_new};
    slots[5] = (PyType_Slot){Py_tp_members, members};
    slots[6] = (PyType_Slot){0, 0};

    /* Initialize Spec */
    /* The name in this PyType_Spec is statically allocated so it is */
    /* expected that it'll outlive the PyType_Spec */
    spec.name = desc->name;
    spec.basicsize = sizeof(PyStructSequence) - sizeof(PyObject *);
    spec.itemsize = sizeof(PyObject *);
    spec.flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC;
    spec.slots = slots;

    bases = PyTuple_Pack(1, &PyTuple_Type);
450 451 452 453
    if (bases == NULL) {
        PyMem_FREE(members);
        return NULL;
    }
454 455 456 457
    type = (PyTypeObject *)PyType_FromSpecWithBases(&spec, bases);
    Py_DECREF(bases);
    PyMem_FREE(members);
    if (type == NULL) {
458
        return NULL;
459
    }
460 461 462

    if (initialize_structseq_dict(
            desc, type->tp_dict, n_members, n_unnamed_members) < 0) {
463
        Py_DECREF(type);
464 465 466 467
        return NULL;
    }

    return type;
468
}
469 470 471 472 473 474 475 476 477 478

int _PyStructSequence_Init(void)
{
    if (_PyUnicode_FromId(&PyId_n_sequence_fields) == NULL
        || _PyUnicode_FromId(&PyId_n_fields) == NULL
        || _PyUnicode_FromId(&PyId_n_unnamed_fields) == NULL)
        return -1;

    return 0;
}