Skip to content

Commit a4fd7aa

Browse files
authored
GH-115776: Allow any fixed sized object to have inline values (GH-123192)
1 parent 4b7c488 commit a4fd7aa

13 files changed

+61
-38
lines changed

Include/internal/pycore_object.h

+3-2
Original file line numberDiff line numberDiff line change
@@ -803,10 +803,11 @@ _PyObject_GetManagedDict(PyObject *obj)
803803
static inline PyDictValues *
804804
_PyObject_InlineValues(PyObject *obj)
805805
{
806+
PyTypeObject *tp = Py_TYPE(obj);
807+
assert(tp->tp_basicsize > 0 && tp->tp_basicsize % sizeof(PyObject *) == 0);
806808
assert(Py_TYPE(obj)->tp_flags & Py_TPFLAGS_INLINE_VALUES);
807809
assert(Py_TYPE(obj)->tp_flags & Py_TPFLAGS_MANAGED_DICT);
808-
assert(Py_TYPE(obj)->tp_basicsize == sizeof(PyObject));
809-
return (PyDictValues *)((char *)obj + sizeof(PyObject));
810+
return (PyDictValues *)((char *)obj + tp->tp_basicsize);
810811
}
811812

812813
extern PyObject ** _PyObject_ComputedDictPointer(PyObject *);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Enables inline values (Python's equivalent of hidden classes) on any class
2+
who's instances are of a fixed size.

Objects/object_layout.md

+4
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,10 @@ So the pre-header is these two fields:
2828
If the object has no physical dictionary, then the ``dict_pointer``
2929
is set to `NULL`.
3030

31+
In 3.13 only objects with no additional data could have inline values.
32+
That is, instances of classes with `tp_basicsize == sizeof(PyObject)`.
33+
In 3.14, any object whose class has `tp_itemsize == 0` can have inline values.
34+
In both versions, the inline values starts `tp_basicsize` bytes after the object.
3135

3236
<details>
3337
<summary> 3.12 </summary>

Objects/typeobject.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -8340,7 +8340,7 @@ type_ready_managed_dict(PyTypeObject *type)
83408340
return -1;
83418341
}
83428342
}
8343-
if (type->tp_itemsize == 0 && type->tp_basicsize == sizeof(PyObject)) {
8343+
if (type->tp_itemsize == 0) {
83448344
type->tp_flags |= Py_TPFLAGS_INLINE_VALUES;
83458345
}
83468346
return 0;

Python/bytecodes.c

+9-7
Original file line numberDiff line numberDiff line change
@@ -2012,9 +2012,10 @@ dummy_func(
20122012
DEOPT_IF(!_PyObject_InlineValues(owner_o)->valid);
20132013
}
20142014

2015-
split op(_LOAD_ATTR_INSTANCE_VALUE, (index/1, owner -- attr, null if (oparg & 1))) {
2015+
split op(_LOAD_ATTR_INSTANCE_VALUE, (offset/1, owner -- attr, null if (oparg & 1))) {
20162016
PyObject *owner_o = PyStackRef_AsPyObjectBorrow(owner);
2017-
PyObject *attr_o = _PyObject_InlineValues(owner_o)->values[index];
2017+
PyObject **value_ptr = (PyObject**)(((char *)owner_o) + offset);
2018+
PyObject *attr_o = *value_ptr;
20182019
DEOPT_IF(attr_o == NULL);
20192020
STAT_INC(LOAD_ATTR, hit);
20202021
Py_INCREF(attr_o);
@@ -2196,16 +2197,17 @@ dummy_func(
21962197
EXIT_IF(_PyObject_InlineValues(owner_o)->valid == 0);
21972198
}
21982199

2199-
op(_STORE_ATTR_INSTANCE_VALUE, (index/1, value, owner --)) {
2200+
op(_STORE_ATTR_INSTANCE_VALUE, (offset/1, value, owner --)) {
22002201
PyObject *owner_o = PyStackRef_AsPyObjectBorrow(owner);
22012202

22022203
STAT_INC(STORE_ATTR, hit);
22032204
assert(_PyObject_GetManagedDict(owner_o) == NULL);
2204-
PyDictValues *values = _PyObject_InlineValues(owner_o);
2205-
2206-
PyObject *old_value = values->values[index];
2207-
values->values[index] = PyStackRef_AsPyObjectSteal(value);
2205+
PyObject **value_ptr = (PyObject**)(((char *)owner_o) + offset);
2206+
PyObject *old_value = *value_ptr;
2207+
*value_ptr = PyStackRef_AsPyObjectSteal(value);
22082208
if (old_value == NULL) {
2209+
PyDictValues *values = _PyObject_InlineValues(owner_o);
2210+
int index = value_ptr - values->values;
22092211
_PyDictValues_AddToInsertionOrder(values, index);
22102212
}
22112213
else {

Python/executor_cases.c.h

+12-8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/gc.c

+3
Original file line numberDiff line numberDiff line change
@@ -2055,6 +2055,9 @@ _PyObject_GC_New(PyTypeObject *tp)
20552055
return NULL;
20562056
}
20572057
_PyObject_Init(op, tp);
2058+
if (tp->tp_flags & Py_TPFLAGS_INLINE_VALUES) {
2059+
_PyObject_InitInlineValues(op, tp);
2060+
}
20582061
return op;
20592062
}
20602063

Python/gc_free_threading.c

+3
Original file line numberDiff line numberDiff line change
@@ -1810,6 +1810,9 @@ _PyObject_GC_New(PyTypeObject *tp)
18101810
return NULL;
18111811
}
18121812
_PyObject_Init(op, tp);
1813+
if (tp->tp_flags & Py_TPFLAGS_INLINE_VALUES) {
1814+
_PyObject_InitInlineValues(op, tp);
1815+
}
18131816
return op;
18141817
}
18151818

Python/generated_cases.c.h

+9-6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/optimizer_bytecodes.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -452,10 +452,10 @@ dummy_func(void) {
452452
top, unused[oparg-2], bottom)) {
453453
}
454454

455-
op(_LOAD_ATTR_INSTANCE_VALUE, (index/1, owner -- attr, null if (oparg & 1))) {
455+
op(_LOAD_ATTR_INSTANCE_VALUE, (offset/1, owner -- attr, null if (oparg & 1))) {
456456
attr = sym_new_not_null(ctx);
457457
null = sym_new_null(ctx);
458-
(void)index;
458+
(void)offset;
459459
(void)owner;
460460
}
461461

Python/optimizer_cases.c.h

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/specialize.c

+10-6
Original file line numberDiff line numberDiff line change
@@ -849,15 +849,19 @@ specialize_dict_access(
849849
assert(PyUnicode_CheckExact(name));
850850
Py_ssize_t index = _PyDictKeys_StringLookup(keys, name);
851851
assert (index != DKIX_ERROR);
852-
if (index != (uint16_t)index) {
853-
SPECIALIZATION_FAIL(base_op,
854-
index == DKIX_EMPTY ?
855-
SPEC_FAIL_ATTR_NOT_IN_KEYS :
856-
SPEC_FAIL_OUT_OF_RANGE);
852+
if (index == DKIX_EMPTY) {
853+
SPECIALIZATION_FAIL(base_op, SPEC_FAIL_ATTR_NOT_IN_KEYS);
854+
return 0;
855+
}
856+
assert(index >= 0);
857+
char *value_addr = (char *)&_PyObject_InlineValues(owner)->values[index];
858+
Py_ssize_t offset = value_addr - (char *)owner;
859+
if (offset != (uint16_t)offset) {
860+
SPECIALIZATION_FAIL(base_op, SPEC_FAIL_OUT_OF_RANGE);
857861
return 0;
858862
}
859863
write_u32(cache->version, type->tp_version_tag);
860-
cache->index = (uint16_t)index;
864+
cache->index = (uint16_t)offset;
861865
instr->op.code = values_op;
862866
}
863867
else {

Tools/gdb/libpython.py

+1-4
Original file line numberDiff line numberDiff line change
@@ -69,9 +69,6 @@ def _type_unsigned_int_ptr():
6969
def _sizeof_void_p():
7070
return gdb.lookup_type('void').pointer().sizeof
7171

72-
def _sizeof_pyobject():
73-
return gdb.lookup_type('PyObject').sizeof
74-
7572
def _managed_dict_offset():
7673
# See pycore_object.h
7774
pyobj = gdb.lookup_type("PyObject")
@@ -505,7 +502,7 @@ def get_keys_values(self):
505502
dict_ptr = dict_ptr_ptr.cast(_type_char_ptr().pointer()).dereference()
506503
if int(dict_ptr):
507504
return None
508-
char_ptr = obj_ptr + _sizeof_pyobject()
505+
char_ptr = obj_ptr + typeobj.field('tp_basicsize')
509506
values_ptr = char_ptr.cast(gdb.lookup_type("PyDictValues").pointer())
510507
values = values_ptr['values']
511508
return PyKeysValuesPair(self.get_cached_keys(), values)

0 commit comments

Comments
 (0)