mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 19:34:08 +00:00 
			
		
		
		
	The `PyDict_SetDefaultRef` function is similar to `PyDict_SetDefault`, but returns a strong reference through the optional `**result` pointer instead of a borrowed reference. Co-authored-by: Petr Viktorin <encukou@gmail.com>
		
			
				
	
	
		
			484 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			484 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include "parts.h"
 | 
						|
#include "util.h"
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_check(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyLong_FromLong(PyDict_Check(obj));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_checkexact(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyLong_FromLong(PyDict_CheckExact(obj));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_new(PyObject *self, PyObject *Py_UNUSED(ignored))
 | 
						|
{
 | 
						|
    return PyDict_New();
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dictproxy_new(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyDictProxy_New(obj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_clear(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    PyDict_Clear(obj);
 | 
						|
    Py_RETURN_NONE;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_copy(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyDict_Copy(obj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_contains(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *obj, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &obj, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(obj);
 | 
						|
    NULLABLE(key);
 | 
						|
    RETURN_INT(PyDict_Contains(obj, key));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_containsstring(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *obj;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &obj, &key, &size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(obj);
 | 
						|
    RETURN_INT(PyDict_ContainsString(obj, key));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_size(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    RETURN_SIZE(PyDict_Size(obj));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_getitem(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(key);
 | 
						|
    PyObject *value = PyDict_GetItem(mapping, key);
 | 
						|
    if (value == NULL) {
 | 
						|
        if (PyErr_Occurred()) {
 | 
						|
            return NULL;
 | 
						|
        }
 | 
						|
        return Py_NewRef(PyExc_KeyError);
 | 
						|
    }
 | 
						|
    return Py_NewRef(value);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_getitemstring(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &mapping, &key, &size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    PyObject *value = PyDict_GetItemString(mapping, key);
 | 
						|
    if (value == NULL) {
 | 
						|
        if (PyErr_Occurred()) {
 | 
						|
            return NULL;
 | 
						|
        }
 | 
						|
        return Py_NewRef(PyExc_KeyError);
 | 
						|
    }
 | 
						|
    return Py_NewRef(value);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_getitemwitherror(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(key);
 | 
						|
    PyObject *value = PyDict_GetItemWithError(mapping, key);
 | 
						|
    if (value == NULL) {
 | 
						|
        if (PyErr_Occurred()) {
 | 
						|
            return NULL;
 | 
						|
        }
 | 
						|
        return Py_NewRef(PyExc_KeyError);
 | 
						|
    }
 | 
						|
    return Py_NewRef(value);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_getitemref(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *obj, *attr_name, *value = UNINITIALIZED_PTR;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &obj, &attr_name)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(obj);
 | 
						|
    NULLABLE(attr_name);
 | 
						|
 | 
						|
    switch (PyDict_GetItemRef(obj, attr_name, &value)) {
 | 
						|
        case -1:
 | 
						|
            assert(value == NULL);
 | 
						|
            return NULL;
 | 
						|
        case 0:
 | 
						|
            assert(value == NULL);
 | 
						|
            return Py_NewRef(PyExc_KeyError);
 | 
						|
        case 1:
 | 
						|
            return value;
 | 
						|
        default:
 | 
						|
            Py_FatalError("PyMapping_GetItemRef() returned invalid code");
 | 
						|
            Py_UNREACHABLE();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_getitemstringref(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *obj, *value = UNINITIALIZED_PTR;
 | 
						|
    const char *attr_name;
 | 
						|
    Py_ssize_t size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &obj, &attr_name, &size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(obj);
 | 
						|
 | 
						|
    switch (PyDict_GetItemStringRef(obj, attr_name, &value)) {
 | 
						|
        case -1:
 | 
						|
            assert(value == NULL);
 | 
						|
            return NULL;
 | 
						|
        case 0:
 | 
						|
            assert(value == NULL);
 | 
						|
            return Py_NewRef(PyExc_KeyError);
 | 
						|
        case 1:
 | 
						|
            return value;
 | 
						|
        default:
 | 
						|
            Py_FatalError("PyDict_GetItemStringRef() returned invalid code");
 | 
						|
            Py_UNREACHABLE();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_setitem(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key, *value;
 | 
						|
    if (!PyArg_ParseTuple(args, "OOO", &mapping, &key, &value)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(key);
 | 
						|
    NULLABLE(value);
 | 
						|
    RETURN_INT(PyDict_SetItem(mapping, key, value));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_setitemstring(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *value;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#O", &mapping, &key, &size, &value)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(value);
 | 
						|
    RETURN_INT(PyDict_SetItemString(mapping, key, value));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_setdefault(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key, *defaultobj;
 | 
						|
    if (!PyArg_ParseTuple(args, "OOO", &mapping, &key, &defaultobj)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(key);
 | 
						|
    NULLABLE(defaultobj);
 | 
						|
    return PyDict_SetDefault(mapping, key, defaultobj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_setdefaultref(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *obj, *key, *default_value, *result = UNINITIALIZED_PTR;
 | 
						|
    if (!PyArg_ParseTuple(args, "OOO", &obj, &key, &default_value)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(obj);
 | 
						|
    NULLABLE(key);
 | 
						|
    NULLABLE(default_value);
 | 
						|
    switch (PyDict_SetDefaultRef(obj, key, default_value, &result)) {
 | 
						|
        case -1:
 | 
						|
            assert(result == NULL);
 | 
						|
            return NULL;
 | 
						|
        case 0:
 | 
						|
            assert(result == default_value);
 | 
						|
            return result;
 | 
						|
        case 1:
 | 
						|
            return result;
 | 
						|
        default:
 | 
						|
            Py_FatalError("PyDict_SetDefaultRef() returned invalid code");
 | 
						|
            Py_UNREACHABLE();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_delitem(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &mapping, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(key);
 | 
						|
    RETURN_INT(PyDict_DelItem(mapping, key));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_delitemstring(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &mapping, &key, &size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    RETURN_INT(PyDict_DelItemString(mapping, key));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_keys(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyDict_Keys(obj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_values(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyDict_Values(obj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_items(PyObject *self, PyObject *obj)
 | 
						|
{
 | 
						|
    NULLABLE(obj);
 | 
						|
    return PyDict_Items(obj);
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_next(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *key = UNINITIALIZED_PTR, *value = UNINITIALIZED_PTR;
 | 
						|
    Py_ssize_t pos;
 | 
						|
    if (!PyArg_ParseTuple(args, "On", &mapping, &pos)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    int rc = PyDict_Next(mapping, &pos, &key, &value);
 | 
						|
    if (rc != 0) {
 | 
						|
        return Py_BuildValue("inOO", rc, pos, key, value);
 | 
						|
    }
 | 
						|
    assert(key == UNINITIALIZED_PTR);
 | 
						|
    assert(value == UNINITIALIZED_PTR);
 | 
						|
    if (PyErr_Occurred()) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    Py_RETURN_NONE;
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_merge(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *mapping2;
 | 
						|
    int override;
 | 
						|
    if (!PyArg_ParseTuple(args, "OOi", &mapping, &mapping2, &override)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(mapping2);
 | 
						|
    RETURN_INT(PyDict_Merge(mapping, mapping2, override));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_update(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *mapping2;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &mapping, &mapping2)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(mapping2);
 | 
						|
    RETURN_INT(PyDict_Update(mapping, mapping2));
 | 
						|
}
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_mergefromseq2(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *mapping, *seq;
 | 
						|
    int override;
 | 
						|
    if (!PyArg_ParseTuple(args, "OOi", &mapping, &seq, &override)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(mapping);
 | 
						|
    NULLABLE(seq);
 | 
						|
    RETURN_INT(PyDict_MergeFromSeq2(mapping, seq, override));
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_pop(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    // Test PyDict_Pop(dict, key, &value)
 | 
						|
    PyObject *dict, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &dict, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(dict);
 | 
						|
    NULLABLE(key);
 | 
						|
    PyObject *result = UNINITIALIZED_PTR;
 | 
						|
    int res = PyDict_Pop(dict, key,  &result);
 | 
						|
    if (res < 0) {
 | 
						|
        assert(result == NULL);
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    if (res == 0) {
 | 
						|
        assert(result == NULL);
 | 
						|
        result = Py_NewRef(Py_None);
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        assert(result != NULL);
 | 
						|
    }
 | 
						|
    return Py_BuildValue("iN", res, result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_pop_null(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    // Test PyDict_Pop(dict, key, NULL)
 | 
						|
    PyObject *dict, *key;
 | 
						|
    if (!PyArg_ParseTuple(args, "OO", &dict, &key)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(dict);
 | 
						|
    NULLABLE(key);
 | 
						|
    RETURN_INT(PyDict_Pop(dict, key,  NULL));
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_popstring(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *dict;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t key_size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &dict, &key, &key_size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(dict);
 | 
						|
    PyObject *result = UNINITIALIZED_PTR;
 | 
						|
    int res = PyDict_PopString(dict, key,  &result);
 | 
						|
    if (res < 0) {
 | 
						|
        assert(result == NULL);
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    if (res == 0) {
 | 
						|
        assert(result == NULL);
 | 
						|
        result = Py_NewRef(Py_None);
 | 
						|
    }
 | 
						|
    else {
 | 
						|
        assert(result != NULL);
 | 
						|
    }
 | 
						|
    return Py_BuildValue("iN", res, result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyObject *
 | 
						|
dict_popstring_null(PyObject *self, PyObject *args)
 | 
						|
{
 | 
						|
    PyObject *dict;
 | 
						|
    const char *key;
 | 
						|
    Py_ssize_t key_size;
 | 
						|
    if (!PyArg_ParseTuple(args, "Oz#", &dict, &key, &key_size)) {
 | 
						|
        return NULL;
 | 
						|
    }
 | 
						|
    NULLABLE(dict);
 | 
						|
    RETURN_INT(PyDict_PopString(dict, key,  NULL));
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static PyMethodDef test_methods[] = {
 | 
						|
    {"dict_check", dict_check, METH_O},
 | 
						|
    {"dict_checkexact", dict_checkexact, METH_O},
 | 
						|
    {"dict_new", dict_new, METH_NOARGS},
 | 
						|
    {"dictproxy_new", dictproxy_new, METH_O},
 | 
						|
    {"dict_clear", dict_clear, METH_O},
 | 
						|
    {"dict_copy", dict_copy, METH_O},
 | 
						|
    {"dict_size", dict_size, METH_O},
 | 
						|
    {"dict_getitem", dict_getitem, METH_VARARGS},
 | 
						|
    {"dict_getitemwitherror", dict_getitemwitherror, METH_VARARGS},
 | 
						|
    {"dict_getitemstring", dict_getitemstring, METH_VARARGS},
 | 
						|
    {"dict_getitemref", dict_getitemref, METH_VARARGS},
 | 
						|
    {"dict_getitemstringref", dict_getitemstringref, METH_VARARGS},
 | 
						|
    {"dict_contains", dict_contains, METH_VARARGS},
 | 
						|
    {"dict_containsstring", dict_containsstring, METH_VARARGS},
 | 
						|
    {"dict_setitem", dict_setitem, METH_VARARGS},
 | 
						|
    {"dict_setitemstring", dict_setitemstring, METH_VARARGS},
 | 
						|
    {"dict_delitem", dict_delitem, METH_VARARGS},
 | 
						|
    {"dict_delitemstring", dict_delitemstring, METH_VARARGS},
 | 
						|
    {"dict_setdefault", dict_setdefault, METH_VARARGS},
 | 
						|
    {"dict_setdefaultref", dict_setdefaultref, METH_VARARGS},
 | 
						|
    {"dict_keys", dict_keys, METH_O},
 | 
						|
    {"dict_values", dict_values, METH_O},
 | 
						|
    {"dict_items", dict_items, METH_O},
 | 
						|
    {"dict_next", dict_next, METH_VARARGS},
 | 
						|
    {"dict_merge", dict_merge, METH_VARARGS},
 | 
						|
    {"dict_update", dict_update, METH_VARARGS},
 | 
						|
    {"dict_mergefromseq2", dict_mergefromseq2, METH_VARARGS},
 | 
						|
    {"dict_pop", dict_pop, METH_VARARGS},
 | 
						|
    {"dict_pop_null", dict_pop_null, METH_VARARGS},
 | 
						|
    {"dict_popstring", dict_popstring, METH_VARARGS},
 | 
						|
    {"dict_popstring_null", dict_popstring_null, METH_VARARGS},
 | 
						|
    {NULL},
 | 
						|
};
 | 
						|
 | 
						|
int
 | 
						|
_PyTestCapi_Init_Dict(PyObject *m)
 | 
						|
{
 | 
						|
    if (PyModule_AddFunctions(m, test_methods) < 0) {
 | 
						|
        return -1;
 | 
						|
    }
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 |