summaryrefslogtreecommitdiffstats
path: root/Objects
diff options
context:
space:
mode:
authorMark Shannon <mark@hotpy.org>2022-02-01 15:05:18 (GMT)
committerGitHub <noreply@github.com>2022-02-01 15:05:18 (GMT)
commit48be46ec1f3f8010570165daa1da4bf9961f3a83 (patch)
tree03c3f5810599572799b4645477622af83aa30ba2 /Objects
parent913e340a323c7e61ae6e4acbb1312b4342657bec (diff)
downloadcpython-48be46ec1f3f8010570165daa1da4bf9961f3a83.zip
cpython-48be46ec1f3f8010570165daa1da4bf9961f3a83.tar.gz
cpython-48be46ec1f3f8010570165daa1da4bf9961f3a83.tar.bz2
bpo-46072: Add some object layout and allocation stats (GH-31051)
Diffstat (limited to 'Objects')
-rw-r--r--Objects/dictobject.c12
-rw-r--r--Objects/obmalloc.c4
2 files changed, 16 insertions, 0 deletions
diff --git a/Objects/dictobject.c b/Objects/dictobject.c
index 39be189..0ad0f0b 100644
--- a/Objects/dictobject.c
+++ b/Objects/dictobject.c
@@ -114,6 +114,7 @@ As a consequence of this, split keys have a maximum size of 16.
#include "Python.h"
#include "pycore_bitutils.h" // _Py_bit_length
#include "pycore_call.h" // _PyObject_CallNoArgs()
+#include "pycore_code.h" // stats
#include "pycore_dict.h" // PyDictKeysObject
#include "pycore_gc.h" // _PyObject_GC_IS_TRACKED()
#include "pycore_object.h" // _PyObject_GC_TRACK()
@@ -4990,6 +4991,7 @@ _PyObject_InitializeDict(PyObject *obj)
return 0;
}
if (tp->tp_flags & Py_TPFLAGS_MANAGED_DICT) {
+ OBJECT_STAT_INC(new_values);
return init_inline_values(obj, tp);
}
PyObject *dict;
@@ -5033,6 +5035,7 @@ _PyObject_MakeDictFromInstanceAttributes(PyObject *obj, PyDictValues *values)
{
assert(Py_TYPE(obj)->tp_flags & Py_TPFLAGS_MANAGED_DICT);
PyDictKeysObject *keys = CACHED_KEYS(Py_TYPE(obj));
+ OBJECT_STAT_INC(dict_materialized_on_request);
return make_dict_from_instance_attributes(keys, values);
}
@@ -5051,6 +5054,14 @@ _PyObject_StoreInstanceAttribute(PyObject *obj, PyDictValues *values,
PyErr_SetObject(PyExc_AttributeError, name);
return -1;
}
+#ifdef Py_STATS
+ if (shared_keys_usable_size(keys) > 14) {
+ OBJECT_STAT_INC(dict_materialized_too_big);
+ }
+ else {
+ OBJECT_STAT_INC(dict_materialized_new_key);
+ }
+#endif
PyObject *dict = make_dict_from_instance_attributes(keys, values);
if (dict == NULL) {
return -1;
@@ -5183,6 +5194,7 @@ PyObject_GenericGetDict(PyObject *obj, void *context)
PyObject **dictptr = _PyObject_ManagedDictPointer(obj);
if (*values_ptr) {
assert(*dictptr == NULL);
+ OBJECT_STAT_INC(dict_materialized_on_request);
*dictptr = dict = make_dict_from_instance_attributes(CACHED_KEYS(tp), *values_ptr);
if (dict != NULL) {
*values_ptr = NULL;
diff --git a/Objects/obmalloc.c b/Objects/obmalloc.c
index e3df7e8..bad4dc0 100644
--- a/Objects/obmalloc.c
+++ b/Objects/obmalloc.c
@@ -1,5 +1,6 @@
#include "Python.h"
#include "pycore_pymem.h" // _PyTraceMalloc_Config
+#include "pycore_code.h" // stats
#include <stdbool.h>
#include <stdlib.h> // malloc()
@@ -695,6 +696,7 @@ PyObject_Malloc(size_t size)
/* see PyMem_RawMalloc() */
if (size > (size_t)PY_SSIZE_T_MAX)
return NULL;
+ OBJECT_STAT_INC(allocations);
return _PyObject.malloc(_PyObject.ctx, size);
}
@@ -704,6 +706,7 @@ PyObject_Calloc(size_t nelem, size_t elsize)
/* see PyMem_RawMalloc() */
if (elsize != 0 && nelem > (size_t)PY_SSIZE_T_MAX / elsize)
return NULL;
+ OBJECT_STAT_INC(allocations);
return _PyObject.calloc(_PyObject.ctx, nelem, elsize);
}
@@ -719,6 +722,7 @@ PyObject_Realloc(void *ptr, size_t new_size)
void
PyObject_Free(void *ptr)
{
+ OBJECT_STAT_INC(frees);
_PyObject.free(_PyObject.ctx, ptr);
}