Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
7605ae41
Commit
7605ae41
authored
Oct 31, 2016
by
Serhiy Storchaka
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Backed out changeset 6b88dfc7b25d
parent
43f46494
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
60 additions
and
63 deletions
+60
-63
Objects/dictobject.c
Objects/dictobject.c
+60
-63
No files found.
Objects/dictobject.c
View file @
7605ae41
...
@@ -1195,21 +1195,41 @@ insertdict(PyDictObject *mp, PyObject *key, Py_hash_t hash, PyObject *value)
...
@@ -1195,21 +1195,41 @@ insertdict(PyDictObject *mp, PyObject *key, Py_hash_t hash, PyObject *value)
}
}
/*
/*
Internal routine used by dictresize() to buid a hashtable of entries.
Internal routine used by dictresize() to insert an item which is
known to be absent from the dict. This routine also assumes that
the dict contains no deleted entries. Besides the performance benefit,
using insertdict() in dictresize() is dangerous (SF bug #1456209).
Note that no refcounts are changed by this routine; if needed, the caller
is responsible for incref'ing `key` and `value`.
Neither mp->ma_used nor k->dk_usable are modified by this routine; the caller
must set them correctly
*/
*/
static
void
static
void
build_indices
(
PyDictKeysObject
*
keys
,
PyDictKeyEntry
*
ep
,
Py_ssize_t
n
)
insertdict_clean
(
PyDictObject
*
mp
,
PyObject
*
key
,
Py_hash_t
hash
,
PyObject
*
value
)
{
{
size_t
mask
=
(
size_t
)
DK_SIZE
(
keys
)
-
1
;
size_t
i
;
for
(
Py_ssize_t
ix
=
0
;
ix
!=
n
;
ix
++
,
ep
++
)
{
PyDictKeysObject
*
k
=
mp
->
ma_keys
;
Py_hash_t
hash
=
ep
->
me_hash
;
size_t
mask
=
(
size_t
)
DK_SIZE
(
k
)
-
1
;
size_t
i
=
hash
&
mask
;
PyDictKeyEntry
*
ep0
=
DK_ENTRIES
(
mp
->
ma_keys
);
for
(
size_t
perturb
=
hash
;
dk_get_index
(
keys
,
i
)
!=
DKIX_EMPTY
;)
{
PyDictKeyEntry
*
ep
;
perturb
>>=
PERTURB_SHIFT
;
i
=
mask
&
((
i
<<
2
)
+
i
+
perturb
+
1
);
assert
(
k
->
dk_lookup
!=
NULL
);
}
assert
(
value
!=
NULL
);
dk_set_index
(
keys
,
i
,
ix
);
assert
(
key
!=
NULL
);
assert
(
PyUnicode_CheckExact
(
key
)
||
k
->
dk_lookup
==
lookdict
);
i
=
hash
&
mask
;
for
(
size_t
perturb
=
hash
;
dk_get_index
(
k
,
i
)
!=
DKIX_EMPTY
;)
{
perturb
>>=
PERTURB_SHIFT
;
i
=
mask
&
((
i
<<
2
)
+
i
+
perturb
+
1
);
}
}
ep
=
&
ep0
[
k
->
dk_nentries
];
assert
(
ep
->
me_value
==
NULL
);
dk_set_index
(
k
,
i
,
k
->
dk_nentries
);
k
->
dk_nentries
++
;
ep
->
me_key
=
key
;
ep
->
me_hash
=
hash
;
ep
->
me_value
=
value
;
}
}
/*
/*
...
@@ -1225,10 +1245,10 @@ but can be resplit by make_keys_shared().
...
@@ -1225,10 +1245,10 @@ but can be resplit by make_keys_shared().
static
int
static
int
dictresize
(
PyDictObject
*
mp
,
Py_ssize_t
minused
)
dictresize
(
PyDictObject
*
mp
,
Py_ssize_t
minused
)
{
{
Py_ssize_t
newsize
,
numentries
;
Py_ssize_t
i
,
newsize
;
PyDictKeysObject
*
oldkeys
;
PyDictKeysObject
*
oldkeys
;
PyObject
**
oldvalues
;
PyObject
**
oldvalues
;
PyDictKeyEntry
*
oldentries
,
*
newentries
;
PyDictKeyEntry
*
ep0
;
/* Find the smallest table size > minused. */
/* Find the smallest table size > minused. */
for
(
newsize
=
PyDict_MINSIZE
;
for
(
newsize
=
PyDict_MINSIZE
;
...
@@ -1239,14 +1259,8 @@ dictresize(PyDictObject *mp, Py_ssize_t minused)
...
@@ -1239,14 +1259,8 @@ dictresize(PyDictObject *mp, Py_ssize_t minused)
PyErr_NoMemory
();
PyErr_NoMemory
();
return
-
1
;
return
-
1
;
}
}
oldkeys
=
mp
->
ma_keys
;
oldkeys
=
mp
->
ma_keys
;
oldvalues
=
mp
->
ma_values
;
/* NOTE: Current odict checks mp->ma_keys to detect resize happen.
* So we can't reuse oldkeys even if oldkeys->dk_size == newsize.
* TODO: Try reusing oldkeys when reimplement odict.
*/
/* Allocate a new table. */
/* Allocate a new table. */
mp
->
ma_keys
=
new_keys_object
(
newsize
);
mp
->
ma_keys
=
new_keys_object
(
newsize
);
if
(
mp
->
ma_keys
==
NULL
)
{
if
(
mp
->
ma_keys
==
NULL
)
{
...
@@ -1255,59 +1269,42 @@ dictresize(PyDictObject *mp, Py_ssize_t minused)
...
@@ -1255,59 +1269,42 @@ dictresize(PyDictObject *mp, Py_ssize_t minused)
}
}
if
(
oldkeys
->
dk_lookup
==
lookdict
)
if
(
oldkeys
->
dk_lookup
==
lookdict
)
mp
->
ma_keys
->
dk_lookup
=
lookdict
;
mp
->
ma_keys
->
dk_lookup
=
lookdict
;
mp
->
ma_values
=
NULL
;
numentries
=
mp
->
ma_used
;
ep0
=
DK_ENTRIES
(
oldkeys
);
oldentries
=
DK_ENTRIES
(
oldkeys
);
/* Main loop below assumes we can transfer refcount to new keys
newentries
=
DK_ENTRIES
(
mp
->
ma_keys
);
* and that value is stored in me_value.
oldvalues
=
mp
->
ma_values
;
* Increment ref-counts and copy values here to compensate
* This (resizing a split table) should be relatively rare */
if
(
oldvalues
!=
NULL
)
{
if
(
oldvalues
!=
NULL
)
{
/* Convert split table into new combined table.
for
(
i
=
0
;
i
<
oldkeys
->
dk_nentries
;
i
++
)
{
* We must incref keys; we can transfer values.
if
(
oldvalues
[
i
]
!=
NULL
)
{
* Note that values of split table is always dense.
Py_INCREF
(
ep0
[
i
].
me_key
);
*/
ep0
[
i
].
me_value
=
oldvalues
[
i
];
for
(
Py_ssize_t
i
=
0
;
i
<
numentries
;
i
++
)
{
}
assert
(
oldvalues
[
i
]
!=
NULL
);
PyDictKeyEntry
*
ep
=
&
oldentries
[
i
];
PyObject
*
key
=
ep
->
me_key
;
Py_INCREF
(
key
);
newentries
[
i
].
me_key
=
key
;
newentries
[
i
].
me_hash
=
ep
->
me_hash
;
newentries
[
i
].
me_value
=
oldvalues
[
i
];
}
}
}
/* Main loop */
for
(
i
=
0
;
i
<
oldkeys
->
dk_nentries
;
i
++
)
{
PyDictKeyEntry
*
ep
=
&
ep0
[
i
];
if
(
ep
->
me_value
!=
NULL
)
{
insertdict_clean
(
mp
,
ep
->
me_key
,
ep
->
me_hash
,
ep
->
me_value
);
}
}
mp
->
ma_keys
->
dk_usable
-=
mp
->
ma_used
;
if
(
oldvalues
!=
NULL
)
{
/* NULL out me_value slot in oldkeys, in case it was shared */
for
(
i
=
0
;
i
<
oldkeys
->
dk_nentries
;
i
++
)
ep0
[
i
].
me_value
=
NULL
;
DK_DECREF
(
oldkeys
);
DK_DECREF
(
oldkeys
);
mp
->
ma_values
=
NULL
;
if
(
oldvalues
!=
empty_values
)
{
if
(
oldvalues
!=
empty_values
)
{
free_values
(
oldvalues
);
free_values
(
oldvalues
);
}
}
}
}
else
{
// combined table.
else
{
if
(
oldkeys
->
dk_nentries
==
numentries
)
{
memcpy
(
newentries
,
oldentries
,
numentries
*
sizeof
(
PyDictKeyEntry
));
}
else
{
PyDictKeyEntry
*
ep
=
oldentries
;
for
(
Py_ssize_t
i
=
0
;
i
<
numentries
;
i
++
)
{
while
(
ep
->
me_value
==
NULL
)
ep
++
;
newentries
[
i
]
=
*
ep
++
;
}
}
assert
(
oldkeys
->
dk_lookup
!=
lookdict_split
);
assert
(
oldkeys
->
dk_lookup
!=
lookdict_split
);
assert
(
oldkeys
->
dk_refcnt
==
1
);
assert
(
oldkeys
->
dk_refcnt
==
1
);
if
(
oldkeys
->
dk_size
==
PyDict_MINSIZE
&&
DK_DEBUG_DECREF
PyObject_FREE
(
oldkeys
);
numfreekeys
<
PyDict_MAXFREELIST
)
{
DK_DEBUG_DECREF
keys_free_list
[
numfreekeys
++
]
=
oldkeys
;
}
else
{
DK_DEBUG_DECREF
PyObject_FREE
(
oldkeys
);
}
}
}
build_indices
(
mp
->
ma_keys
,
newentries
,
numentries
);
mp
->
ma_keys
->
dk_usable
-=
numentries
;
mp
->
ma_keys
->
dk_nentries
=
numentries
;
return
0
;
return
0
;
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment