Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Z
ZODB
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
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Nicolas Wavrant
ZODB
Commits
94c2f2bf
Commit
94c2f2bf
authored
Aug 27, 2009
by
Jim Fulton
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Consistent indentation.
parent
283addf7
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
1215 additions
and
1215 deletions
+1215
-1215
src/BTrees/BTreeTemplate.c
src/BTrees/BTreeTemplate.c
+1215
-1215
No files found.
src/BTrees/BTreeTemplate.c
View file @
94c2f2bf
...
@@ -10,7 +10,7 @@
...
@@ -10,7 +10,7 @@
WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
FOR A PARTICULAR PURPOSE
FOR A PARTICULAR PURPOSE
****************************************************************************/
****************************************************************************/
#define BTREETEMPLATE_C "$Id$\n"
#define BTREETEMPLATE_C "$Id$\n"
...
@@ -26,115 +26,115 @@
...
@@ -26,115 +26,115 @@
static
int
static
int
BTree_check_inner
(
BTree
*
self
,
Bucket
*
nextbucket
)
BTree_check_inner
(
BTree
*
self
,
Bucket
*
nextbucket
)
{
{
int
i
;
int
i
;
Bucket
*
bucketafter
;
Bucket
*
bucketafter
;
Sized
*
child
;
Sized
*
child
;
char
*
errormsg
=
"internal error"
;
/* someone should have overriden */
char
*
errormsg
=
"internal error"
;
/* someone should have overriden */
Sized
*
activated_child
=
NULL
;
Sized
*
activated_child
=
NULL
;
int
result
=
-
1
;
/* until proved innocent */
int
result
=
-
1
;
/* until proved innocent */
#define CHECK(CONDITION, ERRORMSG) \
#define CHECK(CONDITION, ERRORMSG)
\
if (!(CONDITION)) {
\
if (!(CONDITION)) {
\
errormsg = (ERRORMSG);
\
errormsg = (ERRORMSG);
\
goto Error;
\
goto Error;
\
}
}
PER_USE_OR_RETURN
(
self
,
-
1
);
PER_USE_OR_RETURN
(
self
,
-
1
);
CHECK
(
self
->
len
>=
0
,
"BTree len < 0"
);
CHECK
(
self
->
len
>=
0
,
"BTree len < 0"
);
CHECK
(
self
->
len
<=
self
->
size
,
"BTree len > size"
);
CHECK
(
self
->
len
<=
self
->
size
,
"BTree len > size"
);
if
(
self
->
len
==
0
)
{
if
(
self
->
len
==
0
)
{
/* Empty BTree. */
/* Empty BTree. */
CHECK
(
self
->
firstbucket
==
NULL
,
CHECK
(
self
->
firstbucket
==
NULL
,
"Empty BTree has non-NULL firstbucket"
);
"Empty BTree has non-NULL firstbucket"
);
result
=
0
;
result
=
0
;
goto
Done
;
goto
Done
;
}
}
/* Non-empty BTree. */
/* Non-empty BTree. */
CHECK
(
self
->
firstbucket
!=
NULL
,
"Non-empty BTree has NULL firstbucket"
);
CHECK
(
self
->
firstbucket
!=
NULL
,
"Non-empty BTree has NULL firstbucket"
);
/* Obscure: The first bucket is pointed to at least by self->firstbucket
/* Obscure: The first bucket is pointed to at least by self->firstbucket
* and data[0].child of whichever BTree node it's a child of. However,
* and data[0].child of whichever BTree node it's a child of. However,
* if persistence is enabled then the latter BTree node may be a ghost
* if persistence is enabled then the latter BTree node may be a ghost
* at this point, and so its pointers "don't count": we can only rely
* at this point, and so its pointers "don't count": we can only rely
* on self's pointers being intact.
* on self's pointers being intact.
*/
*/
#ifdef PERSISTENT
#ifdef PERSISTENT
CHECK
(
self
->
firstbucket
->
ob_refcnt
>=
1
,
CHECK
(
self
->
firstbucket
->
ob_refcnt
>=
1
,
"Non-empty BTree firstbucket has refcount < 1"
);
"Non-empty BTree firstbucket has refcount < 1"
);
#else
#else
CHECK
(
self
->
firstbucket
->
ob_refcnt
>=
2
,
CHECK
(
self
->
firstbucket
->
ob_refcnt
>=
2
,
"Non-empty BTree firstbucket has refcount < 2"
);
"Non-empty BTree firstbucket has refcount < 2"
);
#endif
#endif
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
CHECK
(
self
->
data
[
i
].
child
!=
NULL
,
"BTree has NULL child"
);
CHECK
(
self
->
data
[
i
].
child
!=
NULL
,
"BTree has NULL child"
);
}
}
if
(
SameType_Check
(
self
,
self
->
data
[
0
].
child
))
{
if
(
SameType_Check
(
self
,
self
->
data
[
0
].
child
))
{
/* Our children are also BTrees. */
/* Our children are also BTrees. */
child
=
self
->
data
[
0
].
child
;
child
=
self
->
data
[
0
].
child
;
UNLESS
(
PER_USE
(
child
))
goto
Done
;
UNLESS
(
PER_USE
(
child
))
goto
Done
;
activated_child
=
child
;
activated_child
=
child
;
CHECK
(
self
->
firstbucket
==
BTREE
(
child
)
->
firstbucket
,
CHECK
(
self
->
firstbucket
==
BTREE
(
child
)
->
firstbucket
,
"BTree has firstbucket different than "
"BTree has firstbucket different than "
"its first child's firstbucket"
);
"its first child's firstbucket"
);
PER_ALLOW_DEACTIVATION
(
child
);
PER_ALLOW_DEACTIVATION
(
child
);
activated_child
=
NULL
;
activated_child
=
NULL
;
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
child
=
self
->
data
[
i
].
child
;
child
=
self
->
data
[
i
].
child
;
CHECK
(
SameType_Check
(
self
,
child
),
CHECK
(
SameType_Check
(
self
,
child
),
"BTree children have different types"
);
"BTree children have different types"
);
if
(
i
==
self
->
len
-
1
)
if
(
i
==
self
->
len
-
1
)
bucketafter
=
nextbucket
;
bucketafter
=
nextbucket
;
else
{
else
{
BTree
*
child2
=
BTREE
(
self
->
data
[
i
+
1
].
child
);
BTree
*
child2
=
BTREE
(
self
->
data
[
i
+
1
].
child
);
UNLESS
(
PER_USE
(
child2
))
goto
Done
;
UNLESS
(
PER_USE
(
child2
))
goto
Done
;
bucketafter
=
child2
->
firstbucket
;
bucketafter
=
child2
->
firstbucket
;
PER_ALLOW_DEACTIVATION
(
child2
);
PER_ALLOW_DEACTIVATION
(
child2
);
}
}
if
(
BTree_check_inner
(
BTREE
(
child
),
bucketafter
)
<
0
)
goto
Done
;
if
(
BTree_check_inner
(
BTREE
(
child
),
bucketafter
)
<
0
)
goto
Done
;
}
}
}
else
{
}
/* Our children are buckets. */
else
{
CHECK
(
self
->
firstbucket
==
BUCKET
(
self
->
data
[
0
].
child
),
/* Our children are buckets. */
"Bottom-level BTree node has inconsistent firstbucket belief"
);
CHECK
(
self
->
firstbucket
==
BUCKET
(
self
->
data
[
0
].
child
),
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
"Bottom-level BTree node has inconsistent firstbucket belief"
);
child
=
self
->
data
[
i
].
child
;
for
(
i
=
0
;
i
<
self
->
len
;
++
i
)
{
UNLESS
(
PER_USE
(
child
))
goto
Done
;
child
=
self
->
data
[
i
].
child
;
activated_child
=
child
;
UNLESS
(
PER_USE
(
child
))
goto
Done
;
CHECK
(
!
SameType_Check
(
self
,
child
),
activated_child
=
child
;
"BTree children have different types"
);
CHECK
(
!
SameType_Check
(
self
,
child
),
CHECK
(
child
->
len
>=
1
,
"Bucket length < 1"
);
/* no empty buckets! */
"BTree children have different types"
);
CHECK
(
child
->
len
<=
child
->
size
,
"Bucket len > size"
);
CHECK
(
child
->
len
>=
1
,
"Bucket length < 1"
);
/* no empty buckets! */
CHECK
(
child
->
len
<=
child
->
size
,
"Bucket len > size"
);
#ifdef PERSISTENT
#ifdef PERSISTENT
CHECK
(
child
->
ob_refcnt
>=
1
,
"Bucket has refcount < 1"
);
CHECK
(
child
->
ob_refcnt
>=
1
,
"Bucket has refcount < 1"
);
#else
#else
CHECK
(
child
->
ob_refcnt
>=
2
,
"Bucket has refcount < 2"
);
CHECK
(
child
->
ob_refcnt
>=
2
,
"Bucket has refcount < 2"
);
#endif
#endif
if
(
i
==
self
->
len
-
1
)
if
(
i
==
self
->
len
-
1
)
bucketafter
=
nextbucket
;
bucketafter
=
nextbucket
;
else
else
bucketafter
=
BUCKET
(
self
->
data
[
i
+
1
].
child
);
bucketafter
=
BUCKET
(
self
->
data
[
i
+
1
].
child
);
CHECK
(
BUCKET
(
child
)
->
next
==
bucketafter
,
CHECK
(
BUCKET
(
child
)
->
next
==
bucketafter
,
"Bucket next pointer is damaged"
);
"Bucket next pointer is damaged"
);
PER_ALLOW_DEACTIVATION
(
child
);
PER_ALLOW_DEACTIVATION
(
child
);
activated_child
=
NULL
;
activated_child
=
NULL
;
}
}
result
=
0
;
goto
Done
;
Error:
PyErr_SetString
(
PyExc_AssertionError
,
errormsg
);
result
=
-
1
;
Done:
/* No point updating access time -- this isn't a "real" use. */
PER_ALLOW_DEACTIVATION
(
self
);
if
(
activated_child
)
{
PER_ALLOW_DEACTIVATION
(
activated_child
);
}
}
return
result
;
}
result
=
0
;
goto
Done
;
Error:
PyErr_SetString
(
PyExc_AssertionError
,
errormsg
);
result
=
-
1
;
Done:
/* No point updating access time -- this isn't a "real" use. */
PER_ALLOW_DEACTIVATION
(
self
);
if
(
activated_child
)
{
PER_ALLOW_DEACTIVATION
(
activated_child
);
}
return
result
;
#undef CHECK
#undef CHECK
}
}
...
@@ -147,14 +147,14 @@ Done:
...
@@ -147,14 +147,14 @@ Done:
static
PyObject
*
static
PyObject
*
BTree_check
(
BTree
*
self
)
BTree_check
(
BTree
*
self
)
{
{
PyObject
*
result
=
NULL
;
PyObject
*
result
=
NULL
;
int
i
=
BTree_check_inner
(
self
,
NULL
);
int
i
=
BTree_check_inner
(
self
,
NULL
);
if
(
i
>=
0
)
{
if
(
i
>=
0
)
{
result
=
Py_None
;
result
=
Py_None
;
Py_INCREF
(
result
);
Py_INCREF
(
result
);
}
}
return
result
;
return
result
;
}
}
/*
/*
...
@@ -179,44 +179,44 @@ BTree_check(BTree *self)
...
@@ -179,44 +179,44 @@ BTree_check(BTree *self)
static
PyObject
*
static
PyObject
*
_BTree_get
(
BTree
*
self
,
PyObject
*
keyarg
,
int
has_key
)
_BTree_get
(
BTree
*
self
,
PyObject
*
keyarg
,
int
has_key
)
{
{
KEY_TYPE
key
;
KEY_TYPE
key
;
PyObject
*
result
=
NULL
;
/* guilty until proved innocent */
PyObject
*
result
=
NULL
;
/* guilty until proved innocent */
int
copied
=
1
;
int
copied
=
1
;
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
UNLESS
(
copied
)
return
NULL
;
UNLESS
(
copied
)
return
NULL
;
PER_USE_OR_RETURN
(
self
,
NULL
);
if
(
self
->
len
==
0
)
{
/* empty BTree */
if
(
has_key
)
result
=
PyInt_FromLong
(
0
);
else
PyErr_SetObject
(
PyExc_KeyError
,
keyarg
);
}
else
{
for
(;;)
{
int
i
;
Sized
*
child
;
PER_USE_OR_RETURN
(
self
,
NULL
);
BTREE_SEARCH
(
i
,
self
,
key
,
goto
Done
);
if
(
self
->
len
==
0
)
{
child
=
self
->
data
[
i
].
child
;
/* empty BTree */
has_key
+=
has_key
!=
0
;
/* bump depth counter, maybe */
if
(
has_key
)
if
(
SameType_Check
(
self
,
child
))
{
result
=
PyInt_FromLong
(
0
);
PER_UNUSE
(
self
);
else
self
=
BTREE
(
child
);
PyErr_SetObject
(
PyExc_KeyError
,
keyarg
);
PER_USE_OR_RETURN
(
self
,
NULL
);
}
}
else
{
else
{
for
(;;)
{
result
=
_bucket_get
(
BUCKET
(
child
),
keyarg
,
has_key
);
int
i
;
break
;
Sized
*
child
;
}
BTREE_SEARCH
(
i
,
self
,
key
,
goto
Done
);
child
=
self
->
data
[
i
].
child
;
has_key
+=
has_key
!=
0
;
/* bump depth counter, maybe */
if
(
SameType_Check
(
self
,
child
))
{
PER_UNUSE
(
self
);
self
=
BTREE
(
child
);
PER_USE_OR_RETURN
(
self
,
NULL
);
}
else
{
result
=
_bucket_get
(
BUCKET
(
child
),
keyarg
,
has_key
);
break
;
}
}
}
}
}
Done:
Done:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
result
;
return
result
;
}
}
static
PyObject
*
static
PyObject
*
...
@@ -232,21 +232,21 @@ BTree_get(BTree *self, PyObject *key)
...
@@ -232,21 +232,21 @@ BTree_get(BTree *self, PyObject *key)
static
Sized
*
static
Sized
*
BTree_newBucket
(
BTree
*
self
)
BTree_newBucket
(
BTree
*
self
)
{
{
PyObject
*
factory
;
PyObject
*
factory
;
Sized
*
result
;
Sized
*
result
;
/* _bucket_type_str defined in BTreeModuleTemplate.c */
/* _bucket_type_str defined in BTreeModuleTemplate.c */
factory
=
PyObject_GetAttr
((
PyObject
*
)
self
->
ob_type
,
_bucket_type_str
);
factory
=
PyObject_GetAttr
((
PyObject
*
)
self
->
ob_type
,
_bucket_type_str
);
if
(
factory
==
NULL
)
if
(
factory
==
NULL
)
return
NULL
;
return
NULL
;
/* TODO: Should we check that the factory actually returns something
/* TODO: Should we check that the factory actually returns something
of the appropriate type? How? The C code here is going to
of the appropriate type? How? The C code here is going to
depend on any custom bucket type having the same layout at the
depend on any custom bucket type having the same layout at the
C level.
C level.
*/
*/
result
=
SIZED
(
PyObject_CallObject
(
factory
,
NULL
));
result
=
SIZED
(
PyObject_CallObject
(
factory
,
NULL
));
Py_DECREF
(
factory
);
Py_DECREF
(
factory
);
return
result
;
return
result
;
}
}
/*
/*
...
@@ -264,36 +264,36 @@ BTree_newBucket(BTree *self)
...
@@ -264,36 +264,36 @@ BTree_newBucket(BTree *self)
static
int
static
int
BTree_split
(
BTree
*
self
,
int
index
,
BTree
*
next
)
BTree_split
(
BTree
*
self
,
int
index
,
BTree
*
next
)
{
{
int
next_size
;
int
next_size
;
Sized
*
child
;
Sized
*
child
;
if
(
index
<
0
||
index
>=
self
->
len
)
if
(
index
<
0
||
index
>=
self
->
len
)
index
=
self
->
len
/
2
;
index
=
self
->
len
/
2
;
next_size
=
self
->
len
-
index
;
next_size
=
self
->
len
-
index
;
ASSERT
(
index
>
0
,
"split creates empty tree"
,
-
1
);
ASSERT
(
index
>
0
,
"split creates empty tree"
,
-
1
);
ASSERT
(
next_size
>
0
,
"split creates empty tree"
,
-
1
);
ASSERT
(
next_size
>
0
,
"split creates empty tree"
,
-
1
);
next
->
data
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
next_size
);
next
->
data
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
next_size
);
if
(
!
next
->
data
)
if
(
!
next
->
data
)
return
-
1
;
return
-
1
;
memcpy
(
next
->
data
,
self
->
data
+
index
,
sizeof
(
BTreeItem
)
*
next_size
);
memcpy
(
next
->
data
,
self
->
data
+
index
,
sizeof
(
BTreeItem
)
*
next_size
);
next
->
size
=
next_size
;
/* but don't set len until we succeed */
next
->
size
=
next_size
;
/* but don't set len until we succeed */
/* Set next's firstbucket. self->firstbucket is still correct. */
/* Set next's firstbucket. self->firstbucket is still correct. */
child
=
next
->
data
[
0
].
child
;
child
=
next
->
data
[
0
].
child
;
if
(
SameType_Check
(
self
,
child
))
{
if
(
SameType_Check
(
self
,
child
))
{
PER_USE_OR_RETURN
(
child
,
-
1
);
PER_USE_OR_RETURN
(
child
,
-
1
);
next
->
firstbucket
=
BTREE
(
child
)
->
firstbucket
;
next
->
firstbucket
=
BTREE
(
child
)
->
firstbucket
;
PER_UNUSE
(
child
);
PER_UNUSE
(
child
);
}
}
else
else
next
->
firstbucket
=
BUCKET
(
child
);
next
->
firstbucket
=
BUCKET
(
child
);
Py_INCREF
(
next
->
firstbucket
);
Py_INCREF
(
next
->
firstbucket
);
next
->
len
=
next_size
;
next
->
len
=
next_size
;
self
->
len
=
index
;
self
->
len
=
index
;
return
PER_CHANGED
(
self
)
>=
0
?
0
:
-
1
;
return
PER_CHANGED
(
self
)
>=
0
?
0
:
-
1
;
}
}
...
@@ -314,32 +314,32 @@ static int BTree_grow(BTree *self, int index, int noval);
...
@@ -314,32 +314,32 @@ static int BTree_grow(BTree *self, int index, int noval);
static
int
static
int
BTree_split_root
(
BTree
*
self
,
int
noval
)
BTree_split_root
(
BTree
*
self
,
int
noval
)
{
{
BTree
*
child
;
BTree
*
child
;
BTreeItem
*
d
;
BTreeItem
*
d
;
/* Create a child BTree, and a new data vector for self. */
child
=
BTREE
(
PyObject_CallObject
(
OBJECT
(
self
->
ob_type
),
NULL
));
if
(
!
child
)
return
-
1
;
d
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
2
);
/* Create a child BTree, and a new data vector for self. */
if
(
!
d
)
{
child
=
BTREE
(
PyObject_CallObject
(
OBJECT
(
self
->
ob_type
),
NULL
));
Py_DECREF
(
child
);
if
(
!
child
)
return
-
1
;
return
-
1
;
}
/* Move our data to new BTree. */
d
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
2
);
child
->
size
=
self
->
size
;
if
(
!
d
)
{
child
->
len
=
self
->
len
;
Py_DECREF
(
child
);
child
->
data
=
self
->
data
;
return
-
1
;
child
->
firstbucket
=
self
->
firstbucket
;
}
Py_INCREF
(
child
->
firstbucket
);
/* Point self to child and split the child. */
/* Move our data to new BTree. */
self
->
data
=
d
;
child
->
size
=
self
->
size
;
self
->
len
=
1
;
child
->
len
=
self
->
len
;
self
->
size
=
2
;
child
->
data
=
self
->
data
;
self
->
data
[
0
].
child
=
SIZED
(
child
);
/* transfers reference ownership */
child
->
firstbucket
=
self
->
firstbucket
;
return
BTree_grow
(
self
,
0
,
noval
);
Py_INCREF
(
child
->
firstbucket
);
/* Point self to child and split the child. */
self
->
data
=
d
;
self
->
len
=
1
;
self
->
size
=
2
;
self
->
data
[
0
].
child
=
SIZED
(
child
);
/* transfers reference ownership */
return
BTree_grow
(
self
,
0
,
noval
);
}
}
/*
/*
...
@@ -369,83 +369,83 @@ BTree_grow(BTree *self, int index, int noval)
...
@@ -369,83 +369,83 @@ BTree_grow(BTree *self, int index, int noval)
BTreeItem
*
d
;
BTreeItem
*
d
;
if
(
self
->
len
==
self
->
size
)
{
if
(
self
->
len
==
self
->
size
)
{
if
(
self
->
size
)
{
if
(
self
->
size
)
{
d
=
BTree_Realloc
(
self
->
data
,
sizeof
(
BTreeItem
)
*
self
->
size
*
2
);
d
=
BTree_Realloc
(
self
->
data
,
sizeof
(
BTreeItem
)
*
self
->
size
*
2
);
if
(
d
==
NULL
)
if
(
d
==
NULL
)
return
-
1
;
return
-
1
;
self
->
data
=
d
;
self
->
data
=
d
;
self
->
size
*=
2
;
self
->
size
*=
2
;
}
}
else
{
else
{
d
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
2
);
d
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
2
);
if
(
d
==
NULL
)
if
(
d
==
NULL
)
return
-
1
;
return
-
1
;
self
->
data
=
d
;
self
->
data
=
d
;
self
->
size
=
2
;
self
->
size
=
2
;
}
}
}
}
if
(
self
->
len
)
{
if
(
self
->
len
)
{
d
=
self
->
data
+
index
;
d
=
self
->
data
+
index
;
v
=
d
->
child
;
v
=
d
->
child
;
/* Create a new object of the same type as the target value */
/* Create a new object of the same type as the target value */
e
=
(
Sized
*
)
PyObject_CallObject
((
PyObject
*
)
v
->
ob_type
,
NULL
);
e
=
(
Sized
*
)
PyObject_CallObject
((
PyObject
*
)
v
->
ob_type
,
NULL
);
if
(
e
==
NULL
)
if
(
e
==
NULL
)
return
-
1
;
return
-
1
;
UNLESS
(
PER_USE
(
v
))
{
UNLESS
(
PER_USE
(
v
))
{
Py_DECREF
(
e
);
Py_DECREF
(
e
);
return
-
1
;
return
-
1
;
}
}
/* Now split between the original (v) and the new (e) at the midpoint*/
/* Now split between the original (v) and the new (e) at the midpoint*/
if
(
SameType_Check
(
self
,
v
))
if
(
SameType_Check
(
self
,
v
))
i
=
BTree_split
((
BTree
*
)
v
,
-
1
,
(
BTree
*
)
e
);
i
=
BTree_split
((
BTree
*
)
v
,
-
1
,
(
BTree
*
)
e
);
else
else
i
=
bucket_split
((
Bucket
*
)
v
,
-
1
,
(
Bucket
*
)
e
);
i
=
bucket_split
((
Bucket
*
)
v
,
-
1
,
(
Bucket
*
)
e
);
PER_ALLOW_DEACTIVATION
(
v
);
PER_ALLOW_DEACTIVATION
(
v
);
if
(
i
<
0
)
{
if
(
i
<
0
)
{
Py_DECREF
(
e
);
Py_DECREF
(
e
);
assert
(
PyErr_Occurred
());
assert
(
PyErr_Occurred
());
return
-
1
;
return
-
1
;
}
}
index
++
;
index
++
;
d
++
;
d
++
;
if
(
self
->
len
>
index
)
/* Shift up the old values one array slot */
if
(
self
->
len
>
index
)
/* Shift up the old values one array slot */
memmove
(
d
+
1
,
d
,
sizeof
(
BTreeItem
)
*
(
self
->
len
-
index
));
memmove
(
d
+
1
,
d
,
sizeof
(
BTreeItem
)
*
(
self
->
len
-
index
));
if
(
SameType_Check
(
self
,
v
))
{
if
(
SameType_Check
(
self
,
v
))
{
COPY_KEY
(
d
->
key
,
BTREE
(
e
)
->
data
->
key
);
COPY_KEY
(
d
->
key
,
BTREE
(
e
)
->
data
->
key
);
/* We take the unused reference from e, so there's no
/* We take the unused reference from e, so there's no
reason to INCREF!
reason to INCREF!
*/
*/
/* INCREF_KEY(self->data[1].key); */
/* INCREF_KEY(self->data[1].key); */
}
}
else
{
else
{
COPY_KEY
(
d
->
key
,
BUCKET
(
e
)
->
keys
[
0
]);
COPY_KEY
(
d
->
key
,
BUCKET
(
e
)
->
keys
[
0
]);
INCREF_KEY
(
d
->
key
);
INCREF_KEY
(
d
->
key
);
}
}
d
->
child
=
e
;
d
->
child
=
e
;
self
->
len
++
;
self
->
len
++
;
if
(
self
->
len
>=
MAX_BTREE_SIZE
(
self
)
*
2
)
/* the root is huge */
if
(
self
->
len
>=
MAX_BTREE_SIZE
(
self
)
*
2
)
/* the root is huge */
return
BTree_split_root
(
self
,
noval
);
return
BTree_split_root
(
self
,
noval
);
}
}
else
{
else
{
/* The BTree is empty. Create an empty bucket. See CAUTION in
/* The BTree is empty. Create an empty bucket. See CAUTION in
* the comments preceding.
* the comments preceding.
*/
*/
assert
(
index
==
0
);
assert
(
index
==
0
);
d
=
self
->
data
;
d
=
self
->
data
;
d
->
child
=
BTree_newBucket
(
self
);
d
->
child
=
BTree_newBucket
(
self
);
if
(
d
->
child
==
NULL
)
if
(
d
->
child
==
NULL
)
return
-
1
;
return
-
1
;
self
->
len
=
1
;
self
->
len
=
1
;
Py_INCREF
(
d
->
child
);
Py_INCREF
(
d
->
child
);
self
->
firstbucket
=
(
Bucket
*
)
d
->
child
;
self
->
firstbucket
=
(
Bucket
*
)
d
->
child
;
}
}
return
0
;
return
0
;
...
@@ -465,50 +465,50 @@ BTree_grow(BTree *self, int index, int noval)
...
@@ -465,50 +465,50 @@ BTree_grow(BTree *self, int index, int noval)
static
Bucket
*
static
Bucket
*
BTree_lastBucket
(
BTree
*
self
)
BTree_lastBucket
(
BTree
*
self
)
{
{
Sized
*
pchild
;
Sized
*
pchild
;
Bucket
*
result
;
Bucket
*
result
;
UNLESS
(
self
->
data
&&
self
->
len
)
{
UNLESS
(
self
->
data
&&
self
->
len
)
{
IndexError
(
-
1
);
/* is this the best action to take? */
IndexError
(
-
1
);
/* is this the best action to take? */
return
NULL
;
return
NULL
;
}
}
pchild
=
self
->
data
[
self
->
len
-
1
].
child
;
pchild
=
self
->
data
[
self
->
len
-
1
].
child
;
if
(
SameType_Check
(
self
,
pchild
))
{
if
(
SameType_Check
(
self
,
pchild
))
{
self
=
BTREE
(
pchild
);
self
=
BTREE
(
pchild
);
PER_USE_OR_RETURN
(
self
,
NULL
);
PER_USE_OR_RETURN
(
self
,
NULL
);
result
=
BTree_lastBucket
(
self
);
result
=
BTree_lastBucket
(
self
);
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
}
}
else
{
else
{
Py_INCREF
(
pchild
);
Py_INCREF
(
pchild
);
result
=
BUCKET
(
pchild
);
result
=
BUCKET
(
pchild
);
}
}
return
result
;
return
result
;
}
}
static
int
static
int
BTree_deleteNextBucket
(
BTree
*
self
)
BTree_deleteNextBucket
(
BTree
*
self
)
{
{
Bucket
*
b
;
Bucket
*
b
;
UNLESS
(
PER_USE
(
self
))
return
-
1
;
UNLESS
(
PER_USE
(
self
))
return
-
1
;
b
=
BTree_lastBucket
(
self
);
b
=
BTree_lastBucket
(
self
);
if
(
b
==
NULL
)
if
(
b
==
NULL
)
goto
err
;
goto
err
;
if
(
Bucket_deleteNextBucket
(
b
)
<
0
)
if
(
Bucket_deleteNextBucket
(
b
)
<
0
)
goto
err
;
goto
err
;
Py_DECREF
(
b
);
Py_DECREF
(
b
);
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
0
;
return
0
;
err:
err:
Py_XDECREF
(
b
);
Py_XDECREF
(
b
);
PER_ALLOW_DEACTIVATION
(
self
);
PER_ALLOW_DEACTIVATION
(
self
);
return
-
1
;
return
-
1
;
}
}
/*
/*
...
@@ -536,44 +536,44 @@ BTree_deleteNextBucket(BTree *self)
...
@@ -536,44 +536,44 @@ BTree_deleteNextBucket(BTree *self)
static
int
static
int
_BTree_clear
(
BTree
*
self
)
_BTree_clear
(
BTree
*
self
)
{
{
const
int
len
=
self
->
len
;
const
int
len
=
self
->
len
;
if
(
self
->
firstbucket
)
{
if
(
self
->
firstbucket
)
{
/* Obscure: The first bucket is pointed to at least by
/* Obscure: The first bucket is pointed to at least by
* self->firstbucket and data[0].child of whichever BTree node it's
* self->firstbucket and data[0].child of whichever BTree node it's
* a child of. However, if persistence is enabled then the latter
* a child of. However, if persistence is enabled then the latter
* BTree node may be a ghost at this point, and so its pointers "don't
* BTree node may be a ghost at this point, and so its pointers "don't
* count": we can only rely on self's pointers being intact.
* count": we can only rely on self's pointers being intact.
*/
*/
#ifdef PERSISTENT
#ifdef PERSISTENT
ASSERT
(
self
->
firstbucket
->
ob_refcnt
>
0
,
ASSERT
(
self
->
firstbucket
->
ob_refcnt
>
0
,
"Invalid firstbucket pointer"
,
-
1
);
"Invalid firstbucket pointer"
,
-
1
);
#else
#else
ASSERT
(
self
->
firstbucket
->
ob_refcnt
>
1
,
ASSERT
(
self
->
firstbucket
->
ob_refcnt
>
1
,
"Invalid firstbucket pointer"
,
-
1
);
"Invalid firstbucket pointer"
,
-
1
);
#endif
#endif
Py_DECREF
(
self
->
firstbucket
);
Py_DECREF
(
self
->
firstbucket
);
self
->
firstbucket
=
NULL
;
self
->
firstbucket
=
NULL
;
}
}
if
(
self
->
data
)
{
if
(
self
->
data
)
{
int
i
;
int
i
;
if
(
len
>
0
)
{
/* 0 is special because key 0 is trash */
if
(
len
>
0
)
{
/* 0 is special because key 0 is trash */
Py_DECREF
(
self
->
data
[
0
].
child
);
Py_DECREF
(
self
->
data
[
0
].
child
);
}
}
for
(
i
=
1
;
i
<
len
;
i
++
)
{
for
(
i
=
1
;
i
<
len
;
i
++
)
{
#ifdef KEY_TYPE_IS_PYOBJECT
#ifdef KEY_TYPE_IS_PYOBJECT
DECREF_KEY
(
self
->
data
[
i
].
key
);
DECREF_KEY
(
self
->
data
[
i
].
key
);
#endif
#endif
Py_DECREF
(
self
->
data
[
i
].
child
);
Py_DECREF
(
self
->
data
[
i
].
child
);
}
free
(
self
->
data
);
self
->
data
=
NULL
;
}
}
free
(
self
->
data
);
self
->
data
=
NULL
;
}
self
->
len
=
self
->
size
=
0
;
self
->
len
=
self
->
size
=
0
;
return
0
;
return
0
;
}
}
/*
/*
...
@@ -586,224 +586,224 @@ _BTree_clear(BTree *self)
...
@@ -586,224 +586,224 @@ _BTree_clear(BTree *self)
is a set).
is a set).
Return:
Return:
-1 error
-1 error
0 successful, and number of entries didn't change
0 successful, and number of entries didn't change
>0 successful, and number of entries did change
>0 successful, and number of entries did change
Internal
Internal
There are two distinct return values > 0:
There are two distinct return values > 0:
1 Successful, number of entries changed, but firstbucket did not go away.
1 Successful, number of entries changed, but firstbucket did not go away.
2 Successful, number of entries changed, firstbucket did go away.
2 Successful, number of entries changed, firstbucket did go away.
This can only happen on a delete (value == NULL). The caller may
This can only happen on a delete (value == NULL). The caller may
need to change its own firstbucket pointer, and in any case *someone*
need to change its own firstbucket pointer, and in any case *someone*
needs to adjust the 'next' pointer of the bucket immediately preceding
needs to adjust the 'next' pointer of the bucket immediately preceding
the bucket that went away (it needs to point to the bucket immediately
the bucket that went away (it needs to point to the bucket immediately
following the bucket that went away).
following the bucket that went away).
*/
*/
static
int
static
int
_BTree_set
(
BTree
*
self
,
PyObject
*
keyarg
,
PyObject
*
value
,
_BTree_set
(
BTree
*
self
,
PyObject
*
keyarg
,
PyObject
*
value
,
int
unique
,
int
noval
)
int
unique
,
int
noval
)
{
{
int
changed
=
0
;
/* did I mutate? */
int
changed
=
0
;
/* did I mutate? */
int
min
;
/* index of child I searched */
int
min
;
/* index of child I searched */
BTreeItem
*
d
;
/* self->data[min] */
BTreeItem
*
d
;
/* self->data[min] */
int
childlength
;
/* len(self->data[min].child) */
int
childlength
;
/* len(self->data[min].child) */
int
status
;
/* our return value; and return value from callee */
int
status
;
/* our return value; and return value from callee */
int
self_was_empty
;
/* was self empty at entry? */
int
self_was_empty
;
/* was self empty at entry? */
KEY_TYPE
key
;
KEY_TYPE
key
;
int
copied
=
1
;
int
copied
=
1
;
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
if
(
!
copied
)
return
-
1
;
if
(
!
copied
)
return
-
1
;
PER_USE_OR_RETURN
(
self
,
-
1
);
PER_USE_OR_RETURN
(
self
,
-
1
);
self_was_empty
=
self
->
len
==
0
;
self_was_empty
=
self
->
len
==
0
;
if
(
self_was_empty
)
{
if
(
self_was_empty
)
{
/* We're empty. Make room. */
/* We're empty. Make room. */
if
(
value
)
{
if
(
value
)
{
if
(
BTree_grow
(
self
,
0
,
noval
)
<
0
)
if
(
BTree_grow
(
self
,
0
,
noval
)
<
0
)
goto
Error
;
goto
Error
;
}
}
else
{
else
{
/* Can't delete a key from an empty BTree. */
/* Can't delete a key from an empty BTree. */
PyErr_SetObject
(
PyExc_KeyError
,
keyarg
);
PyErr_SetObject
(
PyExc_KeyError
,
keyarg
);
goto
Error
;
goto
Error
;
}
}
}
}
/* Find the right child to search, and hand the work off to it. */
/* Find the right child to search, and hand the work off to it. */
BTREE_SEARCH
(
min
,
self
,
key
,
goto
Error
);
BTREE_SEARCH
(
min
,
self
,
key
,
goto
Error
);
d
=
self
->
data
+
min
;
d
=
self
->
data
+
min
;
if
(
SameType_Check
(
self
,
d
->
child
))
if
(
SameType_Check
(
self
,
d
->
child
))
status
=
_BTree_set
(
BTREE
(
d
->
child
),
keyarg
,
value
,
unique
,
noval
);
status
=
_BTree_set
(
BTREE
(
d
->
child
),
keyarg
,
value
,
unique
,
noval
);
else
{
else
{
int
bucket_changed
=
0
;
int
bucket_changed
=
0
;
status
=
_bucket_set
(
BUCKET
(
d
->
child
),
keyarg
,
status
=
_bucket_set
(
BUCKET
(
d
->
child
),
keyarg
,
value
,
unique
,
noval
,
&
bucket_changed
);
value
,
unique
,
noval
,
&
bucket_changed
);
#ifdef PERSISTENT
#ifdef PERSISTENT
/* If a BTree contains only a single bucket, BTree.__getstate__()
/* If a BTree contains only a single bucket, BTree.__getstate__()
* includes the bucket's entire state, and the bucket doesn't get
* includes the bucket's entire state, and the bucket doesn't get
* an oid of its own. So if we have a single oid-less bucket that
* an oid of its own. So if we have a single oid-less bucket that
* changed, it's *our* oid that should be marked as changed -- the
* changed, it's *our* oid that should be marked as changed -- the
* bucket doesn't have one.
* bucket doesn't have one.
*/
if
(
bucket_changed
&&
self
->
len
==
1
&&
self
->
data
[
0
].
child
->
oid
==
NULL
)
{
changed
=
1
;
}
#endif
}
if
(
status
==
0
)
goto
Done
;
if
(
status
<
0
)
goto
Error
;
assert
(
status
==
1
||
status
==
2
);
/* The child changed size. Get its new size. Note that since the tree
* rooted at the child changed size, so did the tree rooted at self:
* our status must be >= 1 too.
*/
*/
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
if
(
bucket_changed
childlength
=
d
->
child
->
len
;
&&
self
->
len
==
1
PER_UNUSE
(
d
->
child
);
&&
self
->
data
[
0
].
child
->
oid
==
NULL
)
{
changed
=
1
;
}
#endif
}
if
(
status
==
0
)
goto
Done
;
if
(
status
<
0
)
goto
Error
;
assert
(
status
==
1
||
status
==
2
);
/* The child changed size. Get its new size. Note that since the tree
* rooted at the child changed size, so did the tree rooted at self:
* our status must be >= 1 too.
*/
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
childlength
=
d
->
child
->
len
;
PER_UNUSE
(
d
->
child
);
if
(
value
)
{
/* A bucket got bigger -- if it's "too big", split it. */
int
toobig
;
assert
(
status
==
1
);
/* can be 2 only on deletes */
if
(
SameType_Check
(
self
,
d
->
child
))
toobig
=
childlength
>
MAX_BTREE_SIZE
(
d
->
child
);
else
toobig
=
childlength
>
MAX_BUCKET_SIZE
(
d
->
child
);
if
(
value
)
{
if
(
toobig
)
{
/* A bucket got bigger -- if it's "too big", split it. */
if
(
BTree_grow
(
self
,
min
,
noval
)
<
0
)
goto
Error
;
int
toobig
;
changed
=
1
;
/* BTree_grow mutated self */
assert
(
status
==
1
);
/* can be 2 only on deletes */
if
(
SameType_Check
(
self
,
d
->
child
))
toobig
=
childlength
>
MAX_BTREE_SIZE
(
d
->
child
);
else
toobig
=
childlength
>
MAX_BUCKET_SIZE
(
d
->
child
);
if
(
toobig
)
{
if
(
BTree_grow
(
self
,
min
,
noval
)
<
0
)
goto
Error
;
changed
=
1
;
/* BTree_grow mutated self */
}
goto
Done
;
/* and status still == 1 */
}
}
goto
Done
;
/* and status still == 1 */
}
/* A bucket got smaller. This is much harder, and despite that we
/* A bucket got smaller. This is much harder, and despite that we
* don't try to rebalance the tree.
* don't try to rebalance the tree.
*/
if
(
status
==
2
)
{
/* this is the last reference to child status */
/* Two problems to solve: May have to adjust our own firstbucket,
* and the bucket that went away needs to get unlinked.
*/
*/
if
(
status
==
2
)
{
/* this is the last reference to child status */
if
(
min
)
{
/* Two problems to solve: May have to adjust our own firstbucket,
/* This wasn't our firstbucket, so no need to adjust ours (note
* and the bucket that went away needs to get unlinked.
* that it can't be the firstbucket of any node above us either).
*/
* Tell "the tree to the left" to do the unlinking.
if
(
min
)
{
*/
/* This wasn't our firstbucket, so no need to adjust ours (note
if
(
BTree_deleteNextBucket
(
BTREE
(
d
[
-
1
].
child
))
<
0
)
goto
Error
;
* that it can't be the firstbucket of any node above us either).
status
=
1
;
/* we solved the child's firstbucket problem */
* Tell "the tree to the left" to do the unlinking.
*/
if
(
BTree_deleteNextBucket
(
BTREE
(
d
[
-
1
].
child
))
<
0
)
goto
Error
;
status
=
1
;
/* we solved the child's firstbucket problem */
}
else
{
/* This was our firstbucket. Update to new firstbucket value. */
Bucket
*
nextbucket
;
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
nextbucket
=
BTREE
(
d
->
child
)
->
firstbucket
;
PER_UNUSE
(
d
->
child
);
Py_XINCREF
(
nextbucket
);
Py_DECREF
(
self
->
firstbucket
);
self
->
firstbucket
=
nextbucket
;
changed
=
1
;
/* The caller has to do the unlinking -- we can't. Also, since
* it was our firstbucket, it may also be theirs.
*/
assert
(
status
==
2
);
}
}
}
else
{
/* If the child isn't empty, we're done! We did all that was possible for
/* This was our firstbucket. Update to new firstbucket value. */
* us to do with the firstbucket problems the child gave us, and since the
Bucket
*
nextbucket
;
* child isn't empty don't create any new firstbucket problems of our own.
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
*/
nextbucket
=
BTREE
(
d
->
child
)
->
firstbucket
;
if
(
childlength
)
goto
Done
;
PER_UNUSE
(
d
->
child
);
/* The child became empty: we need to remove it from self->data.
Py_XINCREF
(
nextbucket
);
* But first, if we're a bottom-level node, we've got more bucket-fiddling
Py_DECREF
(
self
->
firstbucket
);
* to set up.
self
->
firstbucket
=
nextbucket
;
*/
changed
=
1
;
if
(
!
SameType_Check
(
self
,
d
->
child
))
{
/* We're about to delete a bucket. */
/* The caller has to do the unlinking -- we can't. Also, since
if
(
min
)
{
* it was our firstbucket, it may also be theirs.
/* It's not our first bucket, so we can tell the previous
*/
* bucket to adjust its reference to it. It can't be anyone
assert
(
status
==
2
);
* else's first bucket either, so the caller needn't do anything.
*/
if
(
Bucket_deleteNextBucket
(
BUCKET
(
d
[
-
1
].
child
))
<
0
)
goto
Error
;
/* status should be 1, and already is: if it were 2, the
* block above would have set it to 1 in its min != 0 branch.
*/
assert
(
status
==
1
);
}
else
{
Bucket
*
nextbucket
;
/* It's our first bucket. We can't unlink it directly. */
/* 'changed' will be set true by the deletion code following. */
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
nextbucket
=
BUCKET
(
d
->
child
)
->
next
;
PER_UNUSE
(
d
->
child
);
Py_XINCREF
(
nextbucket
);
Py_DECREF
(
self
->
firstbucket
);
self
->
firstbucket
=
nextbucket
;
status
=
2
;
/* we're giving our caller a new firstbucket problem */
}
}
}
}
/* Remove the child from self->data. */
/* If the child isn't empty, we're done! We did all that was possible for
Py_DECREF
(
d
->
child
);
* us to do with the firstbucket problems the child gave us, and since the
#ifdef KEY_TYPE_IS_PYOBJECT
* child isn't empty don't create any new firstbucket problems of our own.
*/
if
(
childlength
)
goto
Done
;
/* The child became empty: we need to remove it from self->data.
* But first, if we're a bottom-level node, we've got more bucket-fiddling
* to set up.
*/
if
(
!
SameType_Check
(
self
,
d
->
child
))
{
/* We're about to delete a bucket. */
if
(
min
)
{
if
(
min
)
{
DECREF_KEY
(
d
->
key
);
/* It's not our first bucket, so we can tell the previous
* bucket to adjust its reference to it. It can't be anyone
* else's first bucket either, so the caller needn't do anything.
*/
if
(
Bucket_deleteNextBucket
(
BUCKET
(
d
[
-
1
].
child
))
<
0
)
goto
Error
;
/* status should be 1, and already is: if it were 2, the
* block above would have set it to 1 in its min != 0 branch.
*/
assert
(
status
==
1
);
}
}
else
if
(
self
->
len
>
1
)
{
else
{
/* We're deleting the first child of a BTree with more than one
Bucket
*
nextbucket
;
* child. The key at d+1 is about to be shifted into slot 0,
/* It's our first bucket. We can't unlink it directly. */
* and hence never to be referenced again (the key in slot 0 is
/* 'changed' will be set true by the deletion code following. */
* trash).
UNLESS
(
PER_USE
(
d
->
child
))
goto
Error
;
*/
nextbucket
=
BUCKET
(
d
->
child
)
->
next
;
DECREF_KEY
((
d
+
1
)
->
key
);
PER_UNUSE
(
d
->
child
);
Py_XINCREF
(
nextbucket
);
Py_DECREF
(
self
->
firstbucket
);
self
->
firstbucket
=
nextbucket
;
status
=
2
;
/* we're giving our caller a new firstbucket problem */
}
}
/* Else min==0 and len==1: we're emptying the BTree entirely, and
}
* there is no key in need of decrefing.
/* Remove the child from self->data. */
Py_DECREF
(
d
->
child
);
#ifdef KEY_TYPE_IS_PYOBJECT
if
(
min
)
{
DECREF_KEY
(
d
->
key
);
}
else
if
(
self
->
len
>
1
)
{
/* We're deleting the first child of a BTree with more than one
* child. The key at d+1 is about to be shifted into slot 0,
* and hence never to be referenced again (the key in slot 0 is
* trash).
*/
*/
DECREF_KEY
((
d
+
1
)
->
key
);
}
/* Else min==0 and len==1: we're emptying the BTree entirely, and
* there is no key in need of decrefing.
*/
#endif
#endif
--
self
->
len
;
--
self
->
len
;
if
(
min
<
self
->
len
)
if
(
min
<
self
->
len
)
memmove
(
d
,
d
+
1
,
(
self
->
len
-
min
)
*
sizeof
(
BTreeItem
));
memmove
(
d
,
d
+
1
,
(
self
->
len
-
min
)
*
sizeof
(
BTreeItem
));
changed
=
1
;
changed
=
1
;
Done:
Done:
#ifdef PERSISTENT
#ifdef PERSISTENT
if
(
changed
)
{
if
(
changed
)
{
if
(
PER_CHANGED
(
self
)
<
0
)
goto
Error
;
if
(
PER_CHANGED
(
self
)
<
0
)
goto
Error
;
}
}
#endif
#endif
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
status
;
return
status
;
Error:
Error:
assert
(
PyErr_Occurred
());
assert
(
PyErr_Occurred
());
if
(
self_was_empty
)
{
if
(
self_was_empty
)
{
/* BTree_grow may have left the BTree in an invalid state. Make
/* BTree_grow may have left the BTree in an invalid state. Make
* sure the tree is a legitimate empty tree.
* sure the tree is a legitimate empty tree.
*/
*/
_BTree_clear
(
self
);
_BTree_clear
(
self
);
}
}
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
-
1
;
return
-
1
;
}
}
/*
/*
...
@@ -821,52 +821,52 @@ Error:
...
@@ -821,52 +821,52 @@ Error:
static
int
static
int
BTree_setitem
(
BTree
*
self
,
PyObject
*
key
,
PyObject
*
v
)
BTree_setitem
(
BTree
*
self
,
PyObject
*
key
,
PyObject
*
v
)
{
{
if
(
_BTree_set
(
self
,
key
,
v
,
0
,
0
)
<
0
)
if
(
_BTree_set
(
self
,
key
,
v
,
0
,
0
)
<
0
)
return
-
1
;
return
-
1
;
return
0
;
return
0
;
}
}
#ifdef PERSISTENT
#ifdef PERSISTENT
static
PyObject
*
static
PyObject
*
BTree__p_deactivate
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
keywords
)
BTree__p_deactivate
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
keywords
)
{
{
int
ghostify
=
1
;
int
ghostify
=
1
;
PyObject
*
force
=
NULL
;
PyObject
*
force
=
NULL
;
if
(
args
&&
PyTuple_GET_SIZE
(
args
)
>
0
)
{
if
(
args
&&
PyTuple_GET_SIZE
(
args
)
>
0
)
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_deactivate takes not positional arguments"
);
"_p_deactivate takes not positional arguments"
);
return
NULL
;
return
NULL
;
}
}
if
(
keywords
)
{
if
(
keywords
)
{
int
size
=
PyDict_Size
(
keywords
);
int
size
=
PyDict_Size
(
keywords
);
force
=
PyDict_GetItemString
(
keywords
,
"force"
);
force
=
PyDict_GetItemString
(
keywords
,
"force"
);
if
(
force
)
if
(
force
)
size
--
;
size
--
;
if
(
size
)
{
if
(
size
)
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_deactivate only accepts keyword arg force"
);
"_p_deactivate only accepts keyword arg force"
);
return
NULL
;
return
NULL
;
}
}
}
}
if
(
self
->
jar
&&
self
->
oid
)
{
if
(
self
->
jar
&&
self
->
oid
)
{
ghostify
=
self
->
state
==
cPersistent_UPTODATE_STATE
;
ghostify
=
self
->
state
==
cPersistent_UPTODATE_STATE
;
if
(
!
ghostify
&&
force
)
{
if
(
!
ghostify
&&
force
)
{
if
(
PyObject_IsTrue
(
force
))
if
(
PyObject_IsTrue
(
force
))
ghostify
=
1
;
ghostify
=
1
;
if
(
PyErr_Occurred
())
if
(
PyErr_Occurred
())
return
NULL
;
return
NULL
;
}
}
if
(
ghostify
)
{
if
(
ghostify
)
{
if
(
_BTree_clear
(
self
)
<
0
)
if
(
_BTree_clear
(
self
)
<
0
)
return
NULL
;
return
NULL
;
PER_GHOSTIFY
(
self
);
PER_GHOSTIFY
(
self
);
}
}
}
}
Py_INCREF
(
Py_None
);
Py_INCREF
(
Py_None
);
return
Py_None
;
return
Py_None
;
}
}
#endif
#endif
...
@@ -878,9 +878,9 @@ BTree_clear(BTree *self)
...
@@ -878,9 +878,9 @@ BTree_clear(BTree *self)
if
(
self
->
len
)
if
(
self
->
len
)
{
{
if
(
_BTree_clear
(
self
)
<
0
)
if
(
_BTree_clear
(
self
)
<
0
)
goto
err
;
goto
err
;
if
(
PER_CHANGED
(
self
)
<
0
)
if
(
PER_CHANGED
(
self
)
<
0
)
goto
err
;
goto
err
;
}
}
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
...
@@ -888,7 +888,7 @@ BTree_clear(BTree *self)
...
@@ -888,7 +888,7 @@ BTree_clear(BTree *self)
Py_INCREF
(
Py_None
);
Py_INCREF
(
Py_None
);
return
Py_None
;
return
Py_None
;
err:
err:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
NULL
;
return
NULL
;
}
}
...
@@ -923,170 +923,170 @@ err:
...
@@ -923,170 +923,170 @@ err:
static
PyObject
*
static
PyObject
*
BTree_getstate
(
BTree
*
self
)
BTree_getstate
(
BTree
*
self
)
{
{
PyObject
*
r
=
NULL
;
PyObject
*
r
=
NULL
;
PyObject
*
o
;
PyObject
*
o
;
int
i
,
l
;
int
i
,
l
;
UNLESS
(
PER_USE
(
self
))
return
NULL
;
UNLESS
(
PER_USE
(
self
))
return
NULL
;
if
(
self
->
len
)
{
if
(
self
->
len
)
{
r
=
PyTuple_New
(
self
->
len
*
2
-
1
);
r
=
PyTuple_New
(
self
->
len
*
2
-
1
);
if
(
r
==
NULL
)
if
(
r
==
NULL
)
goto
err
;
goto
err
;
if
(
self
->
len
==
1
if
(
self
->
len
==
1
&&
self
->
data
->
child
->
ob_type
!=
self
->
ob_type
&&
self
->
data
->
child
->
ob_type
!=
self
->
ob_type
#ifdef PERSISTENT
#ifdef PERSISTENT
&&
BUCKET
(
self
->
data
->
child
)
->
oid
==
NULL
&&
BUCKET
(
self
->
data
->
child
)
->
oid
==
NULL
#endif
#endif
)
{
)
{
/* We have just one bucket. Save its data directly. */
/* We have just one bucket. Save its data directly. */
o
=
bucket_getstate
((
Bucket
*
)
self
->
data
->
child
);
o
=
bucket_getstate
((
Bucket
*
)
self
->
data
->
child
);
if
(
o
==
NULL
)
if
(
o
==
NULL
)
goto
err
;
goto
err
;
PyTuple_SET_ITEM
(
r
,
0
,
o
);
PyTuple_SET_ITEM
(
r
,
0
,
o
);
ASSIGN
(
r
,
Py_BuildValue
(
"(O)"
,
r
));
ASSIGN
(
r
,
Py_BuildValue
(
"(O)"
,
r
));
}
else
{
for
(
i
=
0
,
l
=
0
;
i
<
self
->
len
;
i
++
)
{
if
(
i
)
{
COPY_KEY_TO_OBJECT
(
o
,
self
->
data
[
i
].
key
);
PyTuple_SET_ITEM
(
r
,
l
,
o
);
l
++
;
}
o
=
(
PyObject
*
)
self
->
data
[
i
].
child
;
Py_INCREF
(
o
);
PyTuple_SET_ITEM
(
r
,
l
,
o
);
l
++
;
}
ASSIGN
(
r
,
Py_BuildValue
(
"OO"
,
r
,
self
->
firstbucket
));
}
}
}
else
{
else
{
r
=
Py_None
;
for
(
i
=
0
,
l
=
0
;
i
<
self
->
len
;
i
++
)
{
Py_INCREF
(
r
);
if
(
i
)
{
COPY_KEY_TO_OBJECT
(
o
,
self
->
data
[
i
].
key
);
PyTuple_SET_ITEM
(
r
,
l
,
o
);
l
++
;
}
o
=
(
PyObject
*
)
self
->
data
[
i
].
child
;
Py_INCREF
(
o
);
PyTuple_SET_ITEM
(
r
,
l
,
o
);
l
++
;
}
ASSIGN
(
r
,
Py_BuildValue
(
"OO"
,
r
,
self
->
firstbucket
));
}
}
PER_UNUSE
(
self
);
}
else
{
r
=
Py_None
;
Py_INCREF
(
r
);
}
PER_UNUSE
(
self
);
return
r
;
return
r
;
err:
err:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
Py_XDECREF
(
r
);
Py_XDECREF
(
r
);
return
NULL
;
return
NULL
;
}
}
static
int
static
int
_BTree_setstate
(
BTree
*
self
,
PyObject
*
state
,
int
noval
)
_BTree_setstate
(
BTree
*
self
,
PyObject
*
state
,
int
noval
)
{
{
PyObject
*
items
,
*
firstbucket
=
NULL
;
PyObject
*
items
,
*
firstbucket
=
NULL
;
BTreeItem
*
d
;
BTreeItem
*
d
;
int
len
,
l
,
i
,
copied
=
1
;
int
len
,
l
,
i
,
copied
=
1
;
if
(
_BTree_clear
(
self
)
<
0
)
if
(
_BTree_clear
(
self
)
<
0
)
return
-
1
;
return
-
1
;
/* The state of a BTree can be one of the following:
/* The state of a BTree can be one of the following:
None -- an empty BTree
None -- an empty BTree
A one-tuple -- a single bucket btree
A one-tuple -- a single bucket btree
A two-tuple -- a BTree with more than one bucket
A two-tuple -- a BTree with more than one bucket
See comments for BTree_getstate() for the details.
See comments for BTree_getstate() for the details.
*/
*/
if
(
state
==
Py_None
)
if
(
state
==
Py_None
)
return
0
;
return
0
;
if
(
!
PyArg_ParseTuple
(
state
,
"O|O:__setstate__"
,
&
items
,
&
firstbucket
))
if
(
!
PyArg_ParseTuple
(
state
,
"O|O:__setstate__"
,
&
items
,
&
firstbucket
))
return
-
1
;
return
-
1
;
if
(
!
PyTuple_Check
(
items
))
{
PyErr_SetString
(
PyExc_TypeError
,
"tuple required for first state element"
);
return
-
1
;
}
len
=
PyTuple_Size
(
items
);
if
(
!
PyTuple_Check
(
items
))
{
if
(
len
<
0
)
PyErr_SetString
(
PyExc_TypeError
,
return
-
1
;
"tuple required for first state element"
);
len
=
(
len
+
1
)
/
2
;
return
-
1
;
}
len
=
PyTuple_Size
(
items
);
if
(
len
<
0
)
return
-
1
;
len
=
(
len
+
1
)
/
2
;
assert
(
len
>
0
);
/* If the BTree is empty, it's state is None. */
assert
(
len
>
0
);
/* If the BTree is empty, it's state is None. */
assert
(
self
->
size
==
0
);
/* We called _BTree_clear(). */
assert
(
self
->
size
==
0
);
/* We called _BTree_clear(). */
self
->
data
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
len
);
self
->
data
=
BTree_Malloc
(
sizeof
(
BTreeItem
)
*
len
);
if
(
self
->
data
==
NULL
)
if
(
self
->
data
==
NULL
)
return
-
1
;
return
-
1
;
self
->
size
=
len
;
self
->
size
=
len
;
for
(
i
=
0
,
d
=
self
->
data
,
l
=
0
;
i
<
len
;
i
++
,
d
++
)
{
for
(
i
=
0
,
d
=
self
->
data
,
l
=
0
;
i
<
len
;
i
++
,
d
++
)
{
PyObject
*
v
;
PyObject
*
v
;
if
(
i
)
{
/* skip the first key slot */
if
(
i
)
{
/* skip the first key slot */
COPY_KEY_FROM_ARG
(
d
->
key
,
PyTuple_GET_ITEM
(
items
,
l
),
copied
);
COPY_KEY_FROM_ARG
(
d
->
key
,
PyTuple_GET_ITEM
(
items
,
l
),
copied
);
l
++
;
l
++
;
if
(
!
copied
)
if
(
!
copied
)
return
-
1
;
return
-
1
;
INCREF_KEY
(
d
->
key
);
INCREF_KEY
(
d
->
key
);
}
}
v
=
PyTuple_GET_ITEM
(
items
,
l
);
v
=
PyTuple_GET_ITEM
(
items
,
l
);
if
(
PyTuple_Check
(
v
))
{
if
(
PyTuple_Check
(
v
))
{
/* Handle the special case in __getstate__() for a BTree
/* Handle the special case in __getstate__() for a BTree
with a single bucket. */
with a single bucket. */
d
->
child
=
BTree_newBucket
(
self
);
d
->
child
=
BTree_newBucket
(
self
);
if
(
!
d
->
child
)
if
(
!
d
->
child
)
return
-
1
;
return
-
1
;
if
(
noval
)
{
if
(
noval
)
{
if
(
_set_setstate
(
BUCKET
(
d
->
child
),
v
)
<
0
)
if
(
_set_setstate
(
BUCKET
(
d
->
child
),
v
)
<
0
)
return
-
1
;
return
-
1
;
}
}
else
{
else
{
if
(
_bucket_setstate
(
BUCKET
(
d
->
child
),
v
)
<
0
)
if
(
_bucket_setstate
(
BUCKET
(
d
->
child
),
v
)
<
0
)
return
-
1
;
return
-
1
;
}
}
}
}
else
{
else
{
d
->
child
=
(
Sized
*
)
v
;
d
->
child
=
(
Sized
*
)
v
;
Py_INCREF
(
v
);
Py_INCREF
(
v
);
}
l
++
;
}
}
l
++
;
}
if
(
!
firstbucket
)
if
(
!
firstbucket
)
firstbucket
=
(
PyObject
*
)
self
->
data
->
child
;
firstbucket
=
(
PyObject
*
)
self
->
data
->
child
;
if
(
!
PyObject_IsInstance
(
firstbucket
,
(
PyObject
*
)
if
(
!
PyObject_IsInstance
(
firstbucket
,
(
PyObject
*
)
(
noval
?
&
SetType
:
&
BucketType
)))
{
(
noval
?
&
SetType
:
&
BucketType
)))
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"No firstbucket in non-empty BTree"
);
"No firstbucket in non-empty BTree"
);
return
-
1
;
return
-
1
;
}
}
self
->
firstbucket
=
BUCKET
(
firstbucket
);
self
->
firstbucket
=
BUCKET
(
firstbucket
);
Py_INCREF
(
firstbucket
);
Py_INCREF
(
firstbucket
);
#ifndef PERSISTENT
#ifndef PERSISTENT
/* firstbucket is also the child of some BTree node, but that node may
/* firstbucket is also the child of some BTree node, but that node may
* be a ghost if persistence is enabled.
* be a ghost if persistence is enabled.
*/
*/
assert
(
self
->
firstbucket
->
ob_refcnt
>
1
);
assert
(
self
->
firstbucket
->
ob_refcnt
>
1
);
#endif
#endif
self
->
len
=
len
;
self
->
len
=
len
;
return
0
;
return
0
;
}
}
static
PyObject
*
static
PyObject
*
BTree_setstate
(
BTree
*
self
,
PyObject
*
arg
)
BTree_setstate
(
BTree
*
self
,
PyObject
*
arg
)
{
{
int
r
;
int
r
;
PER_PREVENT_DEACTIVATION
(
self
);
PER_PREVENT_DEACTIVATION
(
self
);
r
=
_BTree_setstate
(
self
,
arg
,
0
);
r
=
_BTree_setstate
(
self
,
arg
,
0
);
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
if
(
r
<
0
)
if
(
r
<
0
)
return
NULL
;
return
NULL
;
Py_INCREF
(
Py_None
);
Py_INCREF
(
Py_None
);
return
Py_None
;
return
Py_None
;
}
}
#ifdef PERSISTENT
#ifdef PERSISTENT
...
@@ -1114,7 +1114,7 @@ get_bucket_state(PyObject *t)
...
@@ -1114,7 +1114,7 @@ get_bucket_state(PyObject *t)
return
Py_None
;
/* an empty BTree */
return
Py_None
;
/* an empty BTree */
if
(
!
PyTuple_Check
(
t
))
{
if
(
!
PyTuple_Check
(
t
))
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_resolveConflict: expected tuple or None for state"
);
"_p_resolveConflict: expected tuple or None for state"
);
return
NULL
;
return
NULL
;
}
}
...
@@ -1127,22 +1127,22 @@ get_bucket_state(PyObject *t)
...
@@ -1127,22 +1127,22 @@ get_bucket_state(PyObject *t)
if
(
PyTuple_GET_SIZE
(
t
)
!=
1
)
{
if
(
PyTuple_GET_SIZE
(
t
)
!=
1
)
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_resolveConflict: expected 1- or 2-tuple for state"
);
"_p_resolveConflict: expected 1- or 2-tuple for state"
);
return
NULL
;
return
NULL
;
}
}
t
=
PyTuple_GET_ITEM
(
t
,
0
);
t
=
PyTuple_GET_ITEM
(
t
,
0
);
if
(
!
PyTuple_Check
(
t
)
||
PyTuple_GET_SIZE
(
t
)
!=
1
)
{
if
(
!
PyTuple_Check
(
t
)
||
PyTuple_GET_SIZE
(
t
)
!=
1
)
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_resolveConflict: expected 1-tuple containing "
"_p_resolveConflict: expected 1-tuple containing "
"bucket state"
);
"bucket state"
);
return
NULL
;
return
NULL
;
}
}
t
=
PyTuple_GET_ITEM
(
t
,
0
);
t
=
PyTuple_GET_ITEM
(
t
,
0
);
if
(
!
PyTuple_Check
(
t
))
{
if
(
!
PyTuple_Check
(
t
))
{
PyErr_SetString
(
PyExc_TypeError
,
PyErr_SetString
(
PyExc_TypeError
,
"_p_resolveConflict: expected tuple for bucket state"
);
"_p_resolveConflict: expected tuple for bucket state"
);
return
NULL
;
return
NULL
;
}
}
...
@@ -1157,188 +1157,188 @@ get_bucket_state(PyObject *t)
...
@@ -1157,188 +1157,188 @@ get_bucket_state(PyObject *t)
static
PyObject
*
static
PyObject
*
BTree__p_resolveConflict
(
BTree
*
self
,
PyObject
*
args
)
BTree__p_resolveConflict
(
BTree
*
self
,
PyObject
*
args
)
{
{
PyObject
*
s
[
3
];
PyObject
*
s
[
3
];
PyObject
*
x
,
*
y
,
*
z
;
PyObject
*
x
,
*
y
,
*
z
;
if
(
!
PyArg_ParseTuple
(
args
,
"OOO"
,
&
x
,
&
y
,
&
z
))
if
(
!
PyArg_ParseTuple
(
args
,
"OOO"
,
&
x
,
&
y
,
&
z
))
return
NULL
;
return
NULL
;
s
[
0
]
=
get_bucket_state
(
x
);
if
(
s
[
0
]
==
NULL
)
return
NULL
;
s
[
1
]
=
get_bucket_state
(
y
);
if
(
s
[
1
]
==
NULL
)
return
NULL
;
s
[
2
]
=
get_bucket_state
(
z
);
if
(
s
[
2
]
==
NULL
)
return
NULL
;
if
(
PyObject_IsInstance
((
PyObject
*
)
self
,
(
PyObject
*
)
&
BTreeType
))
x
=
_bucket__p_resolveConflict
(
OBJECT
(
&
BucketType
),
s
);
else
x
=
_bucket__p_resolveConflict
(
OBJECT
(
&
SetType
),
s
);
if
(
x
==
NULL
)
s
[
0
]
=
get_bucket_state
(
x
);
return
NULL
;
if
(
s
[
0
]
==
NULL
)
return
NULL
;
s
[
1
]
=
get_bucket_state
(
y
);
if
(
s
[
1
]
==
NULL
)
return
NULL
;
s
[
2
]
=
get_bucket_state
(
z
);
if
(
s
[
2
]
==
NULL
)
return
NULL
;
if
(
PyObject_IsInstance
((
PyObject
*
)
self
,
(
PyObject
*
)
&
BTreeType
))
x
=
_bucket__p_resolveConflict
(
OBJECT
(
&
BucketType
),
s
);
else
x
=
_bucket__p_resolveConflict
(
OBJECT
(
&
SetType
),
s
);
if
(
x
==
NULL
)
return
NULL
;
return
Py_BuildValue
(
"((N))"
,
x
);
return
Py_BuildValue
(
"((N))"
,
x
);
}
}
#endif
#endif
/*
/*
BTree_findRangeEnd -- Find one end, expressed as a bucket and
BTree_findRangeEnd -- Find one end, expressed as a bucket and
position, for a range search.
position, for a range search.
If low, return bucket and index of the smallest item >= key,
If low, return bucket and index of the smallest item >= key,
otherwise return bucket and index of the largest item <= key.
otherwise return bucket and index of the largest item <= key.
If exclude_equal, exact matches aren't acceptable; if one is found,
If exclude_equal, exact matches aren't acceptable; if one is found,
move right if low, or left if !low (this is for range searches exclusive
move right if low, or left if !low (this is for range searches exclusive
of an endpoint).
of an endpoint).
Return:
Return:
-1 Error; offset and bucket unchanged
-1 Error; offset and bucket unchanged
0 Not found; offset and bucket unchanged
0 Not found; offset and bucket unchanged
1 Correct bucket and offset stored; the caller owns a new reference
1 Correct bucket and offset stored; the caller owns a new reference
to the bucket.
to the bucket.
Internal:
Internal:
We do binary searches in BTree nodes downward, at each step following
We do binary searches in BTree nodes downward, at each step following
C(i) where K(i) <= key < K(i+1). As always, K(i) <= C(i) < K(i+1) too.
C(i) where K(i) <= key < K(i+1). As always, K(i) <= C(i) < K(i+1) too.
(See Maintainer.txt for the meaning of that notation.) That eventually
(See Maintainer.txt for the meaning of that notation.) That eventually
leads to a bucket where we do Bucket_findRangeEnd. That usually works,
leads to a bucket where we do Bucket_findRangeEnd. That usually works,
but there are two cases where it can fail to find the correct answer:
but there are two cases where it can fail to find the correct answer:
1. On a low search, we find a bucket with keys >= K(i), but that doesn't
1. On a low search, we find a bucket with keys >= K(i), but that doesn't
imply there are keys in the bucket >= key. For example, suppose
imply there are keys in the bucket >= key. For example, suppose
a bucket has keys in 1..100, its successor's keys are in 200..300, and
a bucket has keys in 1..100, its successor's keys are in 200..300, and
we're doing a low search on 150. We'll end up in the first bucket,
we're doing a low search on 150. We'll end up in the first bucket,
but there are no keys >= 150 in it. K(i+1) > key, though, and all
but there are no keys >= 150 in it. K(i+1) > key, though, and all
the keys in C(i+1) >= K(i+1) > key, so the first key in the next
the keys in C(i+1) >= K(i+1) > key, so the first key in the next
bucket (if any) is the correct result. This is easy to find by
bucket (if any) is the correct result. This is easy to find by
following the bucket 'next' pointer.
following the bucket 'next' pointer.
2. On a high search, again that the keys in the bucket are >= K(i)
2. On a high search, again that the keys in the bucket are >= K(i)
doesn't imply that any key in the bucket is <= key, but it's harder
doesn't imply that any key in the bucket is <= key, but it's harder
for this to fail (and an earlier version of this routine didn't
for this to fail (and an earlier version of this routine didn't
catch it): if K(i) itself is in the bucket, it works (then
catch it): if K(i) itself is in the bucket, it works (then
K(i) <= key is *a* key in the bucket that's in the desired range).
K(i) <= key is *a* key in the bucket that's in the desired range).
But when keys get deleted from buckets, they aren't also deleted from
But when keys get deleted from buckets, they aren't also deleted from
BTree nodes, so there's no guarantee that K(i) is in the bucket.
BTree nodes, so there's no guarantee that K(i) is in the bucket.
For example, delete the smallest key S from some bucket, and S
For example, delete the smallest key S from some bucket, and S
remains in the interior BTree nodes. Do a high search for S, and
remains in the interior BTree nodes. Do a high search for S, and
the BTree nodes direct the search to the bucket S used to be in,
the BTree nodes direct the search to the bucket S used to be in,
but all keys remaining in that bucket are > S. The largest key in
but all keys remaining in that bucket are > S. The largest key in
the *preceding* bucket (if any) is < K(i), though, and K(i) <= key,
the *preceding* bucket (if any) is < K(i), though, and K(i) <= key,
so the largest key in the preceding bucket is < key and so is the
so the largest key in the preceding bucket is < key and so is the
proper result.
proper result.
This is harder to get at efficiently, as buckets are linked only in
This is harder to get at efficiently, as buckets are linked only in
the increasing direction. While we're searching downward,
the increasing direction. While we're searching downward,
deepest_smaller is set to the node deepest in the tree where
deepest_smaller is set to the node deepest in the tree where
we *could* have gone to the left of C(i). The rightmost bucket in
we *could* have gone to the left of C(i). The rightmost bucket in
deepest_smaller's subtree is the bucket preceding the bucket we find
deepest_smaller's subtree is the bucket preceding the bucket we find
at first. This is clumsy to get at, but efficient.
at first. This is clumsy to get at, but efficient.
*/
*/
static
int
static
int
BTree_findRangeEnd
(
BTree
*
self
,
PyObject
*
keyarg
,
int
low
,
int
exclude_equal
,
BTree_findRangeEnd
(
BTree
*
self
,
PyObject
*
keyarg
,
int
low
,
int
exclude_equal
,
Bucket
**
bucket
,
int
*
offset
)
{
Bucket
**
bucket
,
int
*
offset
)
{
Sized
*
deepest_smaller
=
NULL
;
/* last possibility to move left */
Sized
*
deepest_smaller
=
NULL
;
/* last possibility to move left */
int
deepest_smaller_is_btree
=
0
;
/* Boolean; if false, it's a bucket */
int
deepest_smaller_is_btree
=
0
;
/* Boolean; if false, it's a bucket */
Bucket
*
pbucket
;
Bucket
*
pbucket
;
int
self_got_rebound
=
0
;
/* Boolean; when true, deactivate self */
int
self_got_rebound
=
0
;
/* Boolean; when true, deactivate self */
int
result
=
-
1
;
/* Until proven innocent */
int
result
=
-
1
;
/* Until proven innocent */
int
i
;
int
i
;
KEY_TYPE
key
;
KEY_TYPE
key
;
int
copied
=
1
;
int
copied
=
1
;
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
UNLESS
(
copied
)
return
-
1
;
/* We don't need to: PER_USE_OR_RETURN(self, -1);
because the caller does. */
UNLESS
(
self
->
data
&&
self
->
len
)
return
0
;
/* Search downward until hitting a bucket, stored in pbucket. */
COPY_KEY_FROM_ARG
(
key
,
keyarg
,
copied
);
for
(;;)
{
UNLESS
(
copied
)
return
-
1
;
Sized
*
pchild
;
int
pchild_is_btree
;
BTREE_SEARCH
(
i
,
self
,
key
,
goto
Done
);
/* We don't need to: PER_USE_OR_RETURN(self, -1);
pchild
=
self
->
data
[
i
].
child
;
because the caller does. */
pchild_is_btree
=
SameType_Check
(
self
,
pchild
);
UNLESS
(
self
->
data
&&
self
->
len
)
return
0
;
if
(
i
)
{
deepest_smaller
=
self
->
data
[
i
-
1
].
child
;
deepest_smaller_is_btree
=
pchild_is_btree
;
}
if
(
pchild_is_btree
)
{
/* Search downward until hitting a bucket, stored in pbucket. */
if
(
self_got_rebound
)
{
for
(;;)
{
PER_UNUSE
(
self
);
Sized
*
pchild
;
}
int
pchild_is_btree
;
self
=
BTREE
(
pchild
);
self_got_rebound
=
1
;
BTREE_SEARCH
(
i
,
self
,
key
,
goto
Done
);
PER_USE_OR_RETURN
(
self
,
-
1
);
pchild
=
self
->
data
[
i
].
child
;
}
pchild_is_btree
=
SameType_Check
(
self
,
pchild
);
else
{
if
(
i
)
{
pbucket
=
BUCKET
(
pchild
);
deepest_smaller
=
self
->
data
[
i
-
1
].
child
;
break
;
deepest_smaller_is_btree
=
pchild_is_btree
;
}
}
}
/* Search the bucket for a suitable key. */
if
(
pchild_is_btree
)
{
i
=
Bucket_findRangeEnd
(
pbucket
,
keyarg
,
low
,
exclude_equal
,
offset
);
if
(
self_got_rebound
)
{
if
(
i
<
0
)
PER_UNUSE
(
self
);
goto
Done
;
}
if
(
i
>
0
)
{
self
=
BTREE
(
pchild
);
Py_INCREF
(
pbucket
);
self_got_rebound
=
1
;
*
bucket
=
pbucket
;
PER_USE_OR_RETURN
(
self
,
-
1
);
result
=
1
;
goto
Done
;
}
}
/* This may be one of the two difficult cases detailed in the comments. */
else
{
if
(
low
)
{
pbucket
=
BUCKET
(
pchild
);
Bucket
*
next
;
break
;
UNLESS
(
PER_USE
(
pbucket
))
goto
Done
;
next
=
pbucket
->
next
;
if
(
next
)
{
result
=
1
;
Py_INCREF
(
next
);
*
bucket
=
next
;
*
offset
=
0
;
}
else
result
=
0
;
PER_UNUSE
(
pbucket
);
}
}
/* High-end search: if it's possible to go left, do so. */
}
else
if
(
deepest_smaller
)
{
if
(
deepest_smaller_is_btree
)
{
/* Search the bucket for a suitable key. */
UNLESS
(
PER_USE
(
deepest_smaller
))
goto
Done
;
i
=
Bucket_findRangeEnd
(
pbucket
,
keyarg
,
low
,
exclude_equal
,
offset
);
/* We own the reference this returns. */
if
(
i
<
0
)
pbucket
=
BTree_lastBucket
(
BTREE
(
deepest_smaller
));
goto
Done
;
PER_UNUSE
(
deepest_smaller
);
if
(
i
>
0
)
{
if
(
pbucket
==
NULL
)
goto
Done
;
/* error */
Py_INCREF
(
pbucket
);
}
*
bucket
=
pbucket
;
else
{
result
=
1
;
pbucket
=
BUCKET
(
deepest_smaller
);
goto
Done
;
Py_INCREF
(
pbucket
);
}
}
/* This may be one of the two difficult cases detailed in the comments. */
UNLESS
(
PER_USE
(
pbucket
))
goto
Done
;
if
(
low
)
{
result
=
1
;
Bucket
*
next
;
*
bucket
=
pbucket
;
/* transfer ownership to caller */
*
offset
=
pbucket
->
len
-
1
;
UNLESS
(
PER_USE
(
pbucket
))
goto
Done
;
PER_UNUSE
(
pbucket
);
next
=
pbucket
->
next
;
if
(
next
)
{
result
=
1
;
Py_INCREF
(
next
);
*
bucket
=
next
;
*
offset
=
0
;
}
}
else
else
result
=
0
;
/* simply not found */
result
=
0
;
PER_UNUSE
(
pbucket
);
Done:
}
if
(
self_got_rebound
)
{
/* High-end search: if it's possible to go left, do so. */
PER_UNUSE
(
self
);
else
if
(
deepest_smaller
)
{
if
(
deepest_smaller_is_btree
)
{
UNLESS
(
PER_USE
(
deepest_smaller
))
goto
Done
;
/* We own the reference this returns. */
pbucket
=
BTree_lastBucket
(
BTREE
(
deepest_smaller
));
PER_UNUSE
(
deepest_smaller
);
if
(
pbucket
==
NULL
)
goto
Done
;
/* error */
}
}
return
result
;
else
{
pbucket
=
BUCKET
(
deepest_smaller
);
Py_INCREF
(
pbucket
);
}
UNLESS
(
PER_USE
(
pbucket
))
goto
Done
;
result
=
1
;
*
bucket
=
pbucket
;
/* transfer ownership to caller */
*
offset
=
pbucket
->
len
-
1
;
PER_UNUSE
(
pbucket
);
}
else
result
=
0
;
/* simply not found */
Done:
if
(
self_got_rebound
)
{
PER_UNUSE
(
self
);
}
return
result
;
}
}
static
PyObject
*
static
PyObject
*
...
@@ -1391,7 +1391,7 @@ BTree_maxminKey(BTree *self, PyObject *args, int min)
...
@@ -1391,7 +1391,7 @@ BTree_maxminKey(BTree *self, PyObject *args, int min)
}
}
assert
(
bucket
->
len
);
assert
(
bucket
->
len
);
offset
=
bucket
->
len
-
1
;
offset
=
bucket
->
len
-
1
;
}
}
COPY_KEY_TO_OBJECT
(
key
,
bucket
->
keys
[
offset
]);
COPY_KEY_TO_OBJECT
(
key
,
bucket
->
keys
[
offset
]);
PER_UNUSE
(
bucket
);
PER_UNUSE
(
bucket
);
...
@@ -1401,8 +1401,8 @@ BTree_maxminKey(BTree *self, PyObject *args, int min)
...
@@ -1401,8 +1401,8 @@ BTree_maxminKey(BTree *self, PyObject *args, int min)
empty:
empty:
PyErr_SetString
(
PyExc_ValueError
,
PyErr_SetString
(
PyExc_ValueError
,
empty_tree
?
"empty tree"
:
empty_tree
?
"empty tree"
:
"no key satisfies the conditions"
);
"no key satisfies the conditions"
);
err:
err:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
if
(
bucket
)
if
(
bucket
)
...
@@ -1435,82 +1435,82 @@ BTree_maxKey(BTree *self, PyObject *args)
...
@@ -1435,82 +1435,82 @@ BTree_maxKey(BTree *self, PyObject *args)
static
PyObject
*
static
PyObject
*
BTree_rangeSearch
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
,
char
type
)
BTree_rangeSearch
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
,
char
type
)
{
{
PyObject
*
min
=
Py_None
;
PyObject
*
min
=
Py_None
;
PyObject
*
max
=
Py_None
;
PyObject
*
max
=
Py_None
;
int
excludemin
=
0
;
int
excludemin
=
0
;
int
excludemax
=
0
;
int
excludemax
=
0
;
int
rc
;
int
rc
;
Bucket
*
lowbucket
=
NULL
;
Bucket
*
lowbucket
=
NULL
;
Bucket
*
highbucket
=
NULL
;
Bucket
*
highbucket
=
NULL
;
int
lowoffset
;
int
lowoffset
;
int
highoffset
;
int
highoffset
;
PyObject
*
result
;
PyObject
*
result
;
if
(
args
)
{
if
(
args
)
{
if
(
!
PyArg_ParseTupleAndKeywords
(
args
,
kw
,
"|OOii"
,
search_keywords
,
if
(
!
PyArg_ParseTupleAndKeywords
(
args
,
kw
,
"|OOii"
,
search_keywords
,
&
min
,
&
min
,
&
max
,
&
max
,
&
excludemin
,
&
excludemin
,
&
excludemax
))
&
excludemax
))
return
NULL
;
return
NULL
;
}
}
UNLESS
(
PER_USE
(
self
))
return
NULL
;
UNLESS
(
PER_USE
(
self
))
return
NULL
;
UNLESS
(
self
->
data
&&
self
->
len
)
goto
empty
;
UNLESS
(
self
->
data
&&
self
->
len
)
goto
empty
;
/* Find the low range */
/* Find the low range */
if
(
min
!=
Py_None
)
{
if
(
min
!=
Py_None
)
{
if
((
rc
=
BTree_findRangeEnd
(
self
,
min
,
1
,
excludemin
,
if
((
rc
=
BTree_findRangeEnd
(
self
,
min
,
1
,
excludemin
,
&
lowbucket
,
&
lowoffset
))
<=
0
)
{
&
lowbucket
,
&
lowoffset
))
<=
0
)
{
if
(
rc
<
0
)
goto
err
;
if
(
rc
<
0
)
goto
err
;
goto
empty
;
goto
empty
;
}
}
}
else
{
}
lowbucket
=
self
->
firstbucket
;
else
{
lowoffset
=
0
;
lowbucket
=
self
->
firstbucket
;
if
(
excludemin
)
{
lowoffset
=
0
;
int
bucketlen
;
if
(
excludemin
)
{
UNLESS
(
PER_USE
(
lowbucket
))
goto
err
;
int
bucketlen
;
bucketlen
=
lowbucket
->
len
;
UNLESS
(
PER_USE
(
lowbucket
))
goto
err
;
PER_UNUSE
(
lowbucket
);
bucketlen
=
lowbucket
->
len
;
if
(
bucketlen
>
1
)
PER_UNUSE
(
lowbucket
);
lowoffset
=
1
;
if
(
bucketlen
>
1
)
lowoffset
=
1
;
else
if
(
self
->
len
<
2
)
else
if
(
self
->
len
<
2
)
goto
empty
;
goto
empty
;
else
{
/* move to first item in next bucket */
else
{
/* move to first item in next bucket */
Bucket
*
next
;
Bucket
*
next
;
UNLESS
(
PER_USE
(
lowbucket
))
goto
err
;
UNLESS
(
PER_USE
(
lowbucket
))
goto
err
;
next
=
lowbucket
->
next
;
next
=
lowbucket
->
next
;
PER_UNUSE
(
lowbucket
);
PER_UNUSE
(
lowbucket
);
assert
(
next
!=
NULL
);
assert
(
next
!=
NULL
);
lowbucket
=
next
;
lowbucket
=
next
;
/* and lowoffset is still 0 */
/* and lowoffset is still 0 */
assert
(
lowoffset
==
0
);
assert
(
lowoffset
==
0
);
}
}
}
Py_INCREF
(
lowbucket
);
}
}
Py_INCREF
(
lowbucket
);
}
/* Find the high range */
/* Find the high range */
if
(
max
!=
Py_None
)
{
if
(
max
!=
Py_None
)
{
if
((
rc
=
BTree_findRangeEnd
(
self
,
max
,
0
,
excludemax
,
if
((
rc
=
BTree_findRangeEnd
(
self
,
max
,
0
,
excludemax
,
&
highbucket
,
&
highoffset
))
<=
0
)
{
&
highbucket
,
&
highoffset
))
<=
0
)
{
Py_DECREF
(
lowbucket
);
Py_DECREF
(
lowbucket
);
if
(
rc
<
0
)
goto
err
;
if
(
rc
<
0
)
goto
err
;
goto
empty
;
goto
empty
;
}
}
}
else
{
}
int
bucketlen
;
else
{
highbucket
=
BTree_lastBucket
(
self
);
int
bucketlen
;
assert
(
highbucket
!=
NULL
);
/* we know self isn't empty */
highbucket
=
BTree_lastBucket
(
self
);
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
assert
(
highbucket
!=
NULL
);
/* we know self isn't empty */
bucketlen
=
highbucket
->
len
;
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
PER_UNUSE
(
highbucket
);
bucketlen
=
highbucket
->
len
;
highoffset
=
bucketlen
-
1
;
PER_UNUSE
(
highbucket
);
if
(
excludemax
)
{
highoffset
=
bucketlen
-
1
;
if
(
excludemax
)
{
if
(
highoffset
>
0
)
if
(
highoffset
>
0
)
--
highoffset
;
--
highoffset
;
else
if
(
self
->
len
<
2
)
else
if
(
self
->
len
<
2
)
...
@@ -1521,73 +1521,73 @@ BTree_rangeSearch(BTree *self, PyObject *args, PyObject *kw, char type)
...
@@ -1521,73 +1521,73 @@ BTree_rangeSearch(BTree *self, PyObject *args, PyObject *kw, char type)
Py_DECREF
(
highbucket
);
Py_DECREF
(
highbucket
);
status
=
PreviousBucket
(
&
highbucket
,
self
->
firstbucket
);
status
=
PreviousBucket
(
&
highbucket
,
self
->
firstbucket
);
if
(
status
<
0
)
{
if
(
status
<
0
)
{
Py_DECREF
(
lowbucket
);
Py_DECREF
(
lowbucket
);
goto
err
;
goto
err
;
}
}
assert
(
status
>
0
);
assert
(
status
>
0
);
Py_INCREF
(
highbucket
);
Py_INCREF
(
highbucket
);
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
highoffset
=
highbucket
->
len
-
1
;
PER_UNUSE
(
highbucket
);
}
}
assert
(
highoffset
>=
0
);
}
/* It's still possible that the range is empty, even if min < max. For
* example, if min=3 and max=4, and 3 and 4 aren't in the BTree, but 2 and
* 5 are, then the low position points to the 5 now and the high position
* points to the 2 now. They're not necessarily even in the same bucket,
* so there's no trick we can play with pointer compares to get out
* cheap in general.
*/
if
(
lowbucket
==
highbucket
&&
lowoffset
>
highoffset
)
goto
empty_and_decref_buckets
;
/* definitely empty */
/* The buckets differ, or they're the same and the offsets show a non-
* empty range.
*/
if
(
min
!=
Py_None
&&
max
!=
Py_None
&&
/* both args user-supplied */
lowbucket
!=
highbucket
)
/* and different buckets */
{
KEY_TYPE
first
;
KEY_TYPE
last
;
int
cmp
;
/* Have to check the hard way: see how the endpoints compare. */
UNLESS
(
PER_USE
(
lowbucket
))
goto
err_and_decref_buckets
;
COPY_KEY
(
first
,
lowbucket
->
keys
[
lowoffset
]);
PER_UNUSE
(
lowbucket
);
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
COPY_KEY
(
last
,
highbucket
->
keys
[
highoffset
])
;
highoffset
=
highbucket
->
len
-
1
;
PER_UNUSE
(
highbucket
);
PER_UNUSE
(
highbucket
);
}
TEST_KEY_SET_OR
(
cmp
,
first
,
last
)
goto
err_and_decref_buckets
;
if
(
cmp
>
0
)
goto
empty_and_decref_buckets
;
}
}
assert
(
highoffset
>=
0
);
}
PER_UNUSE
(
self
);
/* It's still possible that the range is empty, even if min < max. For
* example, if min=3 and max=4, and 3 and 4 aren't in the BTree, but 2 and
* 5 are, then the low position points to the 5 now and the high position
* points to the 2 now. They're not necessarily even in the same bucket,
* so there's no trick we can play with pointer compares to get out
* cheap in general.
*/
if
(
lowbucket
==
highbucket
&&
lowoffset
>
highoffset
)
goto
empty_and_decref_buckets
;
/* definitely empty */
/* The buckets differ, or they're the same and the offsets show a non-
* empty range.
*/
if
(
min
!=
Py_None
&&
max
!=
Py_None
&&
/* both args user-supplied */
lowbucket
!=
highbucket
)
/* and different buckets */
{
KEY_TYPE
first
;
KEY_TYPE
last
;
int
cmp
;
/* Have to check the hard way: see how the endpoints compare. */
UNLESS
(
PER_USE
(
lowbucket
))
goto
err_and_decref_buckets
;
COPY_KEY
(
first
,
lowbucket
->
keys
[
lowoffset
]);
PER_UNUSE
(
lowbucket
);
UNLESS
(
PER_USE
(
highbucket
))
goto
err_and_decref_buckets
;
COPY_KEY
(
last
,
highbucket
->
keys
[
highoffset
]);
PER_UNUSE
(
highbucket
);
TEST_KEY_SET_OR
(
cmp
,
first
,
last
)
goto
err_and_decref_buckets
;
if
(
cmp
>
0
)
goto
empty_and_decref_buckets
;
}
PER_UNUSE
(
self
);
result
=
newBTreeItems
(
type
,
lowbucket
,
lowoffset
,
highbucket
,
highoffset
);
result
=
newBTreeItems
(
type
,
lowbucket
,
lowoffset
,
highbucket
,
highoffset
);
Py_DECREF
(
lowbucket
);
Py_DECREF
(
lowbucket
);
Py_DECREF
(
highbucket
);
Py_DECREF
(
highbucket
);
return
result
;
return
result
;
err_and_decref_buckets:
err_and_decref_buckets:
Py_DECREF
(
lowbucket
);
Py_DECREF
(
lowbucket
);
Py_DECREF
(
highbucket
);
Py_DECREF
(
highbucket
);
err:
err:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
NULL
;
return
NULL
;
empty_and_decref_buckets:
empty_and_decref_buckets:
Py_DECREF
(
lowbucket
);
Py_DECREF
(
lowbucket
);
Py_DECREF
(
highbucket
);
Py_DECREF
(
highbucket
);
empty:
empty:
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
return
newBTreeItems
(
type
,
0
,
0
,
0
,
0
);
return
newBTreeItems
(
type
,
0
,
0
,
0
,
0
);
}
}
/*
/*
...
@@ -1596,7 +1596,7 @@ BTree_rangeSearch(BTree *self, PyObject *args, PyObject *kw, char type)
...
@@ -1596,7 +1596,7 @@ BTree_rangeSearch(BTree *self, PyObject *args, PyObject *kw, char type)
static
PyObject
*
static
PyObject
*
BTree_keys
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_keys
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'k'
);
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'k'
);
}
}
/*
/*
...
@@ -1605,7 +1605,7 @@ BTree_keys(BTree *self, PyObject *args, PyObject *kw)
...
@@ -1605,7 +1605,7 @@ BTree_keys(BTree *self, PyObject *args, PyObject *kw)
static
PyObject
*
static
PyObject
*
BTree_values
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_values
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'v'
);
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'v'
);
}
}
/*
/*
...
@@ -1614,7 +1614,7 @@ BTree_values(BTree *self, PyObject *args, PyObject *kw)
...
@@ -1614,7 +1614,7 @@ BTree_values(BTree *self, PyObject *args, PyObject *kw)
static
PyObject
*
static
PyObject
*
BTree_items
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_items
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'i'
);
return
BTree_rangeSearch
(
self
,
args
,
kw
,
'i'
);
}
}
static
PyObject
*
static
PyObject
*
...
@@ -1703,36 +1703,36 @@ BTree_getm(BTree *self, PyObject *args)
...
@@ -1703,36 +1703,36 @@ BTree_getm(BTree *self, PyObject *args)
static
PyObject
*
static
PyObject
*
BTree_has_key
(
BTree
*
self
,
PyObject
*
key
)
BTree_has_key
(
BTree
*
self
,
PyObject
*
key
)
{
{
return
_BTree_get
(
self
,
key
,
1
);
return
_BTree_get
(
self
,
key
,
1
);
}
}
static
PyObject
*
static
PyObject
*
BTree_setdefault
(
BTree
*
self
,
PyObject
*
args
)
BTree_setdefault
(
BTree
*
self
,
PyObject
*
args
)
{
{
PyObject
*
key
;
PyObject
*
key
;
PyObject
*
failobj
;
/* default */
PyObject
*
failobj
;
/* default */
PyObject
*
value
;
/* return value */
PyObject
*
value
;
/* return value */
if
(
!
PyArg_UnpackTuple
(
args
,
"setdefault"
,
2
,
2
,
&
key
,
&
failobj
))
if
(
!
PyArg_UnpackTuple
(
args
,
"setdefault"
,
2
,
2
,
&
key
,
&
failobj
))
return
NULL
;
return
NULL
;
value
=
_BTree_get
(
self
,
key
,
0
);
value
=
_BTree_get
(
self
,
key
,
0
);
if
(
value
!=
NULL
)
if
(
value
!=
NULL
)
return
value
;
return
value
;
/* The key isn't in the tree. If that's not due to a KeyError exception,
/* The key isn't in the tree. If that's not due to a KeyError exception,
* pass back the unexpected exception.
* pass back the unexpected exception.
*/
*/
if
(
!
PyErr_ExceptionMatches
(
PyExc_KeyError
))
if
(
!
PyErr_ExceptionMatches
(
PyExc_KeyError
))
return
NULL
;
return
NULL
;
PyErr_Clear
();
PyErr_Clear
();
/* Associate `key` with `failobj` in the tree, and return `failobj`. */
/* Associate `key` with `failobj` in the tree, and return `failobj`. */
value
=
failobj
;
value
=
failobj
;
if
(
_BTree_set
(
self
,
key
,
failobj
,
0
,
0
)
<
0
)
if
(
_BTree_set
(
self
,
key
,
failobj
,
0
,
0
)
<
0
)
value
=
NULL
;
value
=
NULL
;
Py_XINCREF
(
value
);
Py_XINCREF
(
value
);
return
value
;
return
value
;
}
}
/* forward declaration */
/* forward declaration */
...
@@ -1742,42 +1742,42 @@ BTree_length_or_nonzero(BTree *self, int nonzero);
...
@@ -1742,42 +1742,42 @@ BTree_length_or_nonzero(BTree *self, int nonzero);
static
PyObject
*
static
PyObject
*
BTree_pop
(
BTree
*
self
,
PyObject
*
args
)
BTree_pop
(
BTree
*
self
,
PyObject
*
args
)
{
{
PyObject
*
key
;
PyObject
*
key
;
PyObject
*
failobj
=
NULL
;
/* default */
PyObject
*
failobj
=
NULL
;
/* default */
PyObject
*
value
;
/* return value */
PyObject
*
value
;
/* return value */
if
(
!
PyArg_UnpackTuple
(
args
,
"pop"
,
1
,
2
,
&
key
,
&
failobj
))
return
NULL
;
value
=
_BTree_get
(
self
,
key
,
0
);
if
(
value
!=
NULL
)
{
/* Delete key and associated value. */
if
(
_BTree_set
(
self
,
key
,
NULL
,
0
,
0
)
<
0
)
{
Py_DECREF
(
value
);
return
NULL
;;
}
return
value
;
}
/* The key isn't in the tree. If that's not due to a KeyError exception,
if
(
!
PyArg_UnpackTuple
(
args
,
"pop"
,
1
,
2
,
&
key
,
&
failobj
))
* pass back the unexpected exception.
return
NULL
;
*/
if
(
!
PyErr_ExceptionMatches
(
PyExc_KeyError
))
return
NULL
;
if
(
failobj
!=
NULL
)
{
value
=
_BTree_get
(
self
,
key
,
0
);
/* Clear the KeyError and return the explicit default. */
if
(
value
!=
NULL
)
{
PyErr_Clear
();
/* Delete key and associated value. */
Py_INCREF
(
failobj
);
if
(
_BTree_set
(
self
,
key
,
NULL
,
0
,
0
)
<
0
)
{
return
failobj
;
Py_DECREF
(
value
);
return
NULL
;;
}
}
return
value
;
}
/* No default given. The only difference in this case is the error
/* The key isn't in the tree. If that's not due to a KeyError exception,
* message, which depends on whether the tree is empty.
* pass back the unexpected exception.
*/
*/
if
(
BTree_length_or_nonzero
(
self
,
1
)
==
0
)
/* tree is empty */
if
(
!
PyErr_ExceptionMatches
(
PyExc_KeyError
))
PyErr_SetString
(
PyExc_KeyError
,
"pop(): BTree is empty"
);
return
NULL
;
return
NULL
;
if
(
failobj
!=
NULL
)
{
/* Clear the KeyError and return the explicit default. */
PyErr_Clear
();
Py_INCREF
(
failobj
);
return
failobj
;
}
/* No default given. The only difference in this case is the error
* message, which depends on whether the tree is empty.
*/
if
(
BTree_length_or_nonzero
(
self
,
1
)
==
0
)
/* tree is empty */
PyErr_SetString
(
PyExc_KeyError
,
"pop(): BTree is empty"
);
return
NULL
;
}
}
/* Search BTree self for key. This is the sq_contains slot of the
/* Search BTree self for key. This is the sq_contains slot of the
...
@@ -1791,14 +1791,14 @@ BTree_pop(BTree *self, PyObject *args)
...
@@ -1791,14 +1791,14 @@ BTree_pop(BTree *self, PyObject *args)
static
int
static
int
BTree_contains
(
BTree
*
self
,
PyObject
*
key
)
BTree_contains
(
BTree
*
self
,
PyObject
*
key
)
{
{
PyObject
*
asobj
=
_BTree_get
(
self
,
key
,
1
);
PyObject
*
asobj
=
_BTree_get
(
self
,
key
,
1
);
int
result
=
-
1
;
int
result
=
-
1
;
if
(
asobj
!=
NULL
)
{
if
(
asobj
!=
NULL
)
{
result
=
PyInt_AsLong
(
asobj
)
?
1
:
0
;
result
=
PyInt_AsLong
(
asobj
)
?
1
:
0
;
Py_DECREF
(
asobj
);
Py_DECREF
(
asobj
);
}
}
return
result
;
return
result
;
}
}
static
PyObject
*
static
PyObject
*
...
@@ -1825,42 +1825,42 @@ BTree_addUnique(BTree *self, PyObject *args)
...
@@ -1825,42 +1825,42 @@ BTree_addUnique(BTree *self, PyObject *args)
static
PyObject
*
static
PyObject
*
buildBTreeIter
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
,
char
kind
)
buildBTreeIter
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
,
char
kind
)
{
{
BTreeIter
*
result
=
NULL
;
BTreeIter
*
result
=
NULL
;
BTreeItems
*
items
=
(
BTreeItems
*
)
BTree_rangeSearch
(
self
,
args
,
kw
,
kind
);
BTreeItems
*
items
=
(
BTreeItems
*
)
BTree_rangeSearch
(
self
,
args
,
kw
,
kind
);
if
(
items
)
{
if
(
items
)
{
result
=
BTreeIter_new
(
items
);
result
=
BTreeIter_new
(
items
);
Py_DECREF
(
items
);
Py_DECREF
(
items
);
}
}
return
(
PyObject
*
)
result
;
return
(
PyObject
*
)
result
;
}
}
/* The implementation of iter(BTree_or_TreeSet); the BTree tp_iter slot. */
/* The implementation of iter(BTree_or_TreeSet); the BTree tp_iter slot. */
static
PyObject
*
static
PyObject
*
BTree_getiter
(
BTree
*
self
)
BTree_getiter
(
BTree
*
self
)
{
{
return
buildBTreeIter
(
self
,
NULL
,
NULL
,
'k'
);
return
buildBTreeIter
(
self
,
NULL
,
NULL
,
'k'
);
}
}
/* The implementation of BTree.iterkeys(). */
/* The implementation of BTree.iterkeys(). */
static
PyObject
*
static
PyObject
*
BTree_iterkeys
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_iterkeys
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
buildBTreeIter
(
self
,
args
,
kw
,
'k'
);
return
buildBTreeIter
(
self
,
args
,
kw
,
'k'
);
}
}
/* The implementation of BTree.itervalues(). */
/* The implementation of BTree.itervalues(). */
static
PyObject
*
static
PyObject
*
BTree_itervalues
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_itervalues
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
buildBTreeIter
(
self
,
args
,
kw
,
'v'
);
return
buildBTreeIter
(
self
,
args
,
kw
,
'v'
);
}
}
/* The implementation of BTree.iteritems(). */
/* The implementation of BTree.iteritems(). */
static
PyObject
*
static
PyObject
*
BTree_iteritems
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
BTree_iteritems
(
BTree
*
self
,
PyObject
*
args
,
PyObject
*
kw
)
{
{
return
buildBTreeIter
(
self
,
args
,
kw
,
'i'
);
return
buildBTreeIter
(
self
,
args
,
kw
,
'i'
);
}
}
/* End of iterator support. */
/* End of iterator support. */
...
@@ -1875,172 +1875,172 @@ BTree_iteritems(BTree *self, PyObject *args, PyObject *kw)
...
@@ -1875,172 +1875,172 @@ BTree_iteritems(BTree *self, PyObject *args, PyObject *kw)
*/
*/
static
struct
PyMemberDef
BTree_members
[]
=
{
static
struct
PyMemberDef
BTree_members
[]
=
{
{
"_firstbucket"
,
T_OBJECT
,
offsetof
(
BTree
,
firstbucket
),
RO
},
{
"_firstbucket"
,
T_OBJECT
,
offsetof
(
BTree
,
firstbucket
),
RO
},
{
NULL
}
{
NULL
}
};
};
static
struct
PyMethodDef
BTree_methods
[]
=
{
static
struct
PyMethodDef
BTree_methods
[]
=
{
{
"__getstate__"
,
(
PyCFunction
)
BTree_getstate
,
METH_NOARGS
,
{
"__getstate__"
,
(
PyCFunction
)
BTree_getstate
,
METH_NOARGS
,
"__getstate__() -> state
\n\n
"
"__getstate__() -> state
\n\n
"
"Return the picklable state of the BTree."
},
"Return the picklable state of the BTree."
},
{
"__setstate__"
,
(
PyCFunction
)
BTree_setstate
,
METH_O
,
{
"__setstate__"
,
(
PyCFunction
)
BTree_setstate
,
METH_O
,
"__setstate__(state)
\n\n
"
"__setstate__(state)
\n\n
"
"Set the state of the BTree."
},
"Set the state of the BTree."
},
{
"has_key"
,
(
PyCFunction
)
BTree_has_key
,
METH_O
,
{
"has_key"
,
(
PyCFunction
)
BTree_has_key
,
METH_O
,
"has_key(key)
\n\n
"
"has_key(key)
\n\n
"
"Return true if the BTree contains the given key."
},
"Return true if the BTree contains the given key."
},
{
"keys"
,
(
PyCFunction
)
BTree_keys
,
METH_KEYWORDS
,
{
"keys"
,
(
PyCFunction
)
BTree_keys
,
METH_KEYWORDS
,
"keys([min, max]) -> list of keys
\n\n
"
"keys([min, max]) -> list of keys
\n\n
"
"Returns the keys of the BTree. If min and max are supplied, only
\n
"
"Returns the keys of the BTree. If min and max are supplied, only
\n
"
"keys greater than min and less than max are returned."
},
"keys greater than min and less than max are returned."
},
{
"values"
,
(
PyCFunction
)
BTree_values
,
METH_KEYWORDS
,
{
"values"
,
(
PyCFunction
)
BTree_values
,
METH_KEYWORDS
,
"values([min, max]) -> list of values
\n\n
"
"values([min, max]) -> list of values
\n\n
"
"Returns the values of the BTree. If min and max are supplied, only
\n
"
"Returns the values of the BTree. If min and max are supplied, only
\n
"
"values corresponding to keys greater than min and less than max are
\n
"
"values corresponding to keys greater than min and less than max are
\n
"
"returned."
},
"returned."
},
{
"items"
,
(
PyCFunction
)
BTree_items
,
METH_KEYWORDS
,
{
"items"
,
(
PyCFunction
)
BTree_items
,
METH_KEYWORDS
,
"items([min, max]) -> -- list of key, value pairs
\n\n
"
"items([min, max]) -> -- list of key, value pairs
\n\n
"
"Returns the items of the BTree. If min and max are supplied, only
\n
"
"Returns the items of the BTree. If min and max are supplied, only
\n
"
"items with keys greater than min and less than max are returned."
},
"items with keys greater than min and less than max are returned."
},
{
"byValue"
,
(
PyCFunction
)
BTree_byValue
,
METH_O
,
{
"byValue"
,
(
PyCFunction
)
BTree_byValue
,
METH_O
,
"byValue(min) -> list of value, key pairs
\n\n
"
"byValue(min) -> list of value, key pairs
\n\n
"
"Returns list of value, key pairs where the value is >= min. The
\n
"
"Returns list of value, key pairs where the value is >= min. The
\n
"
"list is sorted by value. Note that items() returns keys in the
\n
"
"list is sorted by value. Note that items() returns keys in the
\n
"
"opposite order."
},
"opposite order."
},
{
"get"
,
(
PyCFunction
)
BTree_getm
,
METH_VARARGS
,
{
"get"
,
(
PyCFunction
)
BTree_getm
,
METH_VARARGS
,
"get(key[, default=None]) -> Value for key or default
\n\n
"
"get(key[, default=None]) -> Value for key or default
\n\n
"
"Return the value or the default if the key is not found."
},
"Return the value or the default if the key is not found."
},
{
"setdefault"
,
(
PyCFunction
)
BTree_setdefault
,
METH_VARARGS
,
{
"setdefault"
,
(
PyCFunction
)
BTree_setdefault
,
METH_VARARGS
,
"D.setdefault(k, d) -> D.get(k, d), also set D[k]=d if k not in D.
\n\n
"
"D.setdefault(k, d) -> D.get(k, d), also set D[k]=d if k not in D.
\n\n
"
"Return the value like get() except that if key is missing, d is both
\n
"
"Return the value like get() except that if key is missing, d is both
\n
"
"returned and inserted into the BTree as the value of k."
},
"returned and inserted into the BTree as the value of k."
},
{
"pop"
,
(
PyCFunction
)
BTree_pop
,
METH_VARARGS
,
{
"pop"
,
(
PyCFunction
)
BTree_pop
,
METH_VARARGS
,
"D.pop(k[, d]) -> v, remove key and return the corresponding value.
\n\n
"
"D.pop(k[, d]) -> v, remove key and return the corresponding value.
\n\n
"
"If key is not found, d is returned if given, otherwise KeyError
\n
"
"If key is not found, d is returned if given, otherwise KeyError
\n
"
"is raised."
},
"is raised."
},
{
"maxKey"
,
(
PyCFunction
)
BTree_maxKey
,
METH_VARARGS
,
{
"maxKey"
,
(
PyCFunction
)
BTree_maxKey
,
METH_VARARGS
,
"maxKey([max]) -> key
\n\n
"
"maxKey([max]) -> key
\n\n
"
"Return the largest key in the BTree. If max is specified, return
\n
"
"Return the largest key in the BTree. If max is specified, return
\n
"
"the largest key <= max."
},
"the largest key <= max."
},
{
"minKey"
,
(
PyCFunction
)
BTree_minKey
,
METH_VARARGS
,
{
"minKey"
,
(
PyCFunction
)
BTree_minKey
,
METH_VARARGS
,
"minKey([mi]) -> key
\n\n
"
"minKey([mi]) -> key
\n\n
"
"Return the smallest key in the BTree. If min is specified, return
\n
"
"Return the smallest key in the BTree. If min is specified, return
\n
"
"the smallest key >= min."
},
"the smallest key >= min."
},
{
"clear"
,
(
PyCFunction
)
BTree_clear
,
METH_NOARGS
,
{
"clear"
,
(
PyCFunction
)
BTree_clear
,
METH_NOARGS
,
"clear()
\n\n
Remove all of the items from the BTree."
},
"clear()
\n\n
Remove all of the items from the BTree."
},
{
"insert"
,
(
PyCFunction
)
BTree_addUnique
,
METH_VARARGS
,
{
"insert"
,
(
PyCFunction
)
BTree_addUnique
,
METH_VARARGS
,
"insert(key, value) -> 0 or 1
\n\n
"
"insert(key, value) -> 0 or 1
\n\n
"
"Add an item if the key is not already used. Return 1 if the item was
\n
"
"Add an item if the key is not already used. Return 1 if the item was
\n
"
"added, or 0 otherwise."
},
"added, or 0 otherwise."
},
{
"update"
,
(
PyCFunction
)
Mapping_update
,
METH_O
,
{
"update"
,
(
PyCFunction
)
Mapping_update
,
METH_O
,
"update(collection)
\n\n
Add the items from the given collection."
},
"update(collection)
\n\n
Add the items from the given collection."
},
{
"iterkeys"
,
(
PyCFunction
)
BTree_iterkeys
,
METH_KEYWORDS
,
{
"iterkeys"
,
(
PyCFunction
)
BTree_iterkeys
,
METH_KEYWORDS
,
"B.iterkeys([min[,max]]) -> an iterator over the keys of B"
},
"B.iterkeys([min[,max]]) -> an iterator over the keys of B"
},
{
"itervalues"
,
(
PyCFunction
)
BTree_itervalues
,
METH_KEYWORDS
,
{
"itervalues"
,
(
PyCFunction
)
BTree_itervalues
,
METH_KEYWORDS
,
"B.itervalues([min[,max]]) -> an iterator over the values of B"
},
"B.itervalues([min[,max]]) -> an iterator over the values of B"
},
{
"iteritems"
,
(
PyCFunction
)
BTree_iteritems
,
METH_KEYWORDS
,
{
"iteritems"
,
(
PyCFunction
)
BTree_iteritems
,
METH_KEYWORDS
,
"B.iteritems([min[,max]]) -> an iterator over the (key, value) items of B"
},
"B.iteritems([min[,max]]) -> an iterator over the (key, value) items of B"
},
{
"_check"
,
(
PyCFunction
)
BTree_check
,
METH_NOARGS
,
{
"_check"
,
(
PyCFunction
)
BTree_check
,
METH_NOARGS
,
"Perform sanity check on BTree, and raise exception if flawed."
},
"Perform sanity check on BTree, and raise exception if flawed."
},
#ifdef PERSISTENT
#ifdef PERSISTENT
{
"_p_resolveConflict"
,
(
PyCFunction
)
BTree__p_resolveConflict
,
{
"_p_resolveConflict"
,
(
PyCFunction
)
BTree__p_resolveConflict
,
METH_VARARGS
,
METH_VARARGS
,
"_p_resolveConflict() -- Reinitialize from a newly created copy"
},
"_p_resolveConflict() -- Reinitialize from a newly created copy"
},
{
"_p_deactivate"
,
(
PyCFunction
)
BTree__p_deactivate
,
METH_KEYWORDS
,
{
"_p_deactivate"
,
(
PyCFunction
)
BTree__p_deactivate
,
METH_KEYWORDS
,
"_p_deactivate()
\n\n
Reinitialize from a newly created copy."
},
"_p_deactivate()
\n\n
Reinitialize from a newly created copy."
},
#endif
#endif
{
NULL
,
NULL
}
{
NULL
,
NULL
}
};
};
static
int
static
int
BTree_init
(
PyObject
*
self
,
PyObject
*
args
,
PyObject
*
kwds
)
BTree_init
(
PyObject
*
self
,
PyObject
*
args
,
PyObject
*
kwds
)
{
{
PyObject
*
v
=
NULL
;
PyObject
*
v
=
NULL
;
if
(
!
PyArg_ParseTuple
(
args
,
"|O:"
MOD_NAME_PREFIX
"BTree"
,
&
v
))
if
(
!
PyArg_ParseTuple
(
args
,
"|O:"
MOD_NAME_PREFIX
"BTree"
,
&
v
))
return
-
1
;
return
-
1
;
if
(
v
)
if
(
v
)
return
update_from_seq
(
self
,
v
);
return
update_from_seq
(
self
,
v
);
else
else
return
0
;
return
0
;
}
}
static
void
static
void
BTree_dealloc
(
BTree
*
self
)
BTree_dealloc
(
BTree
*
self
)
{
{
if
(
self
->
state
!=
cPersistent_GHOST_STATE
)
if
(
self
->
state
!=
cPersistent_GHOST_STATE
)
_BTree_clear
(
self
);
_BTree_clear
(
self
);
cPersistenceCAPI
->
pertype
->
tp_dealloc
((
PyObject
*
)
self
);
cPersistenceCAPI
->
pertype
->
tp_dealloc
((
PyObject
*
)
self
);
}
}
static
int
static
int
BTree_traverse
(
BTree
*
self
,
visitproc
visit
,
void
*
arg
)
BTree_traverse
(
BTree
*
self
,
visitproc
visit
,
void
*
arg
)
{
{
int
err
=
0
;
int
err
=
0
;
int
i
,
len
;
int
i
,
len
;
#define VISIT(SLOT) \
#define VISIT(SLOT) \
if (SLOT) {
\
if (SLOT) {
\
err = visit((PyObject *)(SLOT), arg);
\
err = visit((PyObject *)(SLOT), arg);
\
if (err)
\
if (err)
\
goto Done;
\
goto Done;
\
}
}
if
(
self
->
ob_type
==
&
BTreeType
)
if
(
self
->
ob_type
==
&
BTreeType
)
assert
(
self
->
ob_type
->
tp_dictoffset
==
0
);
assert
(
self
->
ob_type
->
tp_dictoffset
==
0
);
/* Call our base type's traverse function. Because BTrees are
/* Call our base type's traverse function. Because BTrees are
* subclasses of Peristent, there must be one.
* subclasses of Peristent, there must be one.
*/
*/
err
=
cPersistenceCAPI
->
pertype
->
tp_traverse
((
PyObject
*
)
self
,
visit
,
arg
);
err
=
cPersistenceCAPI
->
pertype
->
tp_traverse
((
PyObject
*
)
self
,
visit
,
arg
);
if
(
err
)
if
(
err
)
goto
Done
;
goto
Done
;
/* If this is registered with the persistence system, cleaning up cycles
/* If this is registered with the persistence system, cleaning up cycles
* is the database's problem. It would be horrid to unghostify BTree
* is the database's problem. It would be horrid to unghostify BTree
* nodes here just to chase pointers every time gc runs.
* nodes here just to chase pointers every time gc runs.
*/
*/
if
(
self
->
state
==
cPersistent_GHOST_STATE
)
if
(
self
->
state
==
cPersistent_GHOST_STATE
)
goto
Done
;
goto
Done
;
len
=
self
->
len
;
len
=
self
->
len
;
#ifdef KEY_TYPE_IS_PYOBJECT
#ifdef KEY_TYPE_IS_PYOBJECT
/* Keys are Python objects so need to be traversed. Note that the
/* Keys are Python objects so need to be traversed. Note that the
* key 0 slot is unused and should not be traversed.
* key 0 slot is unused and should not be traversed.
*/
*/
for
(
i
=
1
;
i
<
len
;
i
++
)
for
(
i
=
1
;
i
<
len
;
i
++
)
VISIT
(
self
->
data
[
i
].
key
);
VISIT
(
self
->
data
[
i
].
key
);
#endif
#endif
/* Children are always pointers, and child 0 is legit. */
/* Children are always pointers, and child 0 is legit. */
for
(
i
=
0
;
i
<
len
;
i
++
)
for
(
i
=
0
;
i
<
len
;
i
++
)
VISIT
(
self
->
data
[
i
].
child
);
VISIT
(
self
->
data
[
i
].
child
);
VISIT
(
self
->
firstbucket
);
VISIT
(
self
->
firstbucket
);
Done:
Done:
return
err
;
return
err
;
#undef VISIT
#undef VISIT
}
}
...
@@ -2048,9 +2048,9 @@ Done:
...
@@ -2048,9 +2048,9 @@ Done:
static
int
static
int
BTree_tp_clear
(
BTree
*
self
)
BTree_tp_clear
(
BTree
*
self
)
{
{
if
(
self
->
state
!=
cPersistent_GHOST_STATE
)
if
(
self
->
state
!=
cPersistent_GHOST_STATE
)
_BTree_clear
(
self
);
_BTree_clear
(
self
);
return
0
;
return
0
;
}
}
/*
/*
...
@@ -2071,25 +2071,25 @@ BTree_tp_clear(BTree *self)
...
@@ -2071,25 +2071,25 @@ BTree_tp_clear(BTree *self)
static
Py_ssize_t
static
Py_ssize_t
BTree_length_or_nonzero
(
BTree
*
self
,
int
nonzero
)
BTree_length_or_nonzero
(
BTree
*
self
,
int
nonzero
)
{
{
int
result
;
int
result
;
Bucket
*
b
;
Bucket
*
b
;
Bucket
*
next
;
Bucket
*
next
;
PER_USE_OR_RETURN
(
self
,
-
1
);
PER_USE_OR_RETURN
(
self
,
-
1
);
b
=
self
->
firstbucket
;
b
=
self
->
firstbucket
;
PER_UNUSE
(
self
);
PER_UNUSE
(
self
);
if
(
nonzero
)
if
(
nonzero
)
return
b
!=
NULL
;
return
b
!=
NULL
;
result
=
0
;
result
=
0
;
while
(
b
)
{
while
(
b
)
{
PER_USE_OR_RETURN
(
b
,
-
1
);
PER_USE_OR_RETURN
(
b
,
-
1
);
result
+=
b
->
len
;
result
+=
b
->
len
;
next
=
b
->
next
;
next
=
b
->
next
;
PER_UNUSE
(
b
);
PER_UNUSE
(
b
);
b
=
next
;
b
=
next
;
}
}
return
result
;
return
result
;
}
}
static
Py_ssize_t
static
Py_ssize_t
...
@@ -2105,16 +2105,16 @@ static PyMappingMethods BTree_as_mapping = {
...
@@ -2105,16 +2105,16 @@ static PyMappingMethods BTree_as_mapping = {
};
};
static
PySequenceMethods
BTree_as_sequence
=
{
static
PySequenceMethods
BTree_as_sequence
=
{
(
lenfunc
)
0
,
/* sq_length */
(
lenfunc
)
0
,
/* sq_length */
(
binaryfunc
)
0
,
/* sq_concat */
(
binaryfunc
)
0
,
/* sq_concat */
(
ssizeargfunc
)
0
,
/* sq_repeat */
(
ssizeargfunc
)
0
,
/* sq_repeat */
(
ssizeargfunc
)
0
,
/* sq_item */
(
ssizeargfunc
)
0
,
/* sq_item */
(
ssizessizeargfunc
)
0
,
/* sq_slice */
(
ssizessizeargfunc
)
0
,
/* sq_slice */
(
ssizeobjargproc
)
0
,
/* sq_ass_item */
(
ssizeobjargproc
)
0
,
/* sq_ass_item */
(
ssizessizeobjargproc
)
0
,
/* sq_ass_slice */
(
ssizessizeobjargproc
)
0
,
/* sq_ass_slice */
(
objobjproc
)
BTree_contains
,
/* sq_contains */
(
objobjproc
)
BTree_contains
,
/* sq_contains */
0
,
/* sq_inplace_concat */
0
,
/* sq_inplace_concat */
0
,
/* sq_inplace_repeat */
0
,
/* sq_inplace_repeat */
};
};
static
Py_ssize_t
static
Py_ssize_t
...
@@ -2128,44 +2128,44 @@ static PyNumberMethods BTree_as_number_for_nonzero = {
...
@@ -2128,44 +2128,44 @@ static PyNumberMethods BTree_as_number_for_nonzero = {
(
inquiry
)
BTree_nonzero
};
(
inquiry
)
BTree_nonzero
};
static
PyTypeObject
BTreeType
=
{
static
PyTypeObject
BTreeType
=
{
PyObject_HEAD_INIT
(
NULL
)
/* PyPersist_Type */
PyObject_HEAD_INIT
(
NULL
)
/* PyPersist_Type */
0
,
/* ob_size */
0
,
/* ob_size */
MODULE_NAME
MOD_NAME_PREFIX
"BTree"
,
/* tp_name */
MODULE_NAME
MOD_NAME_PREFIX
"BTree"
,
/* tp_name */
sizeof
(
BTree
),
/* tp_basicsize */
sizeof
(
BTree
),
/* tp_basicsize */
0
,
/* tp_itemsize */
0
,
/* tp_itemsize */
(
destructor
)
BTree_dealloc
,
/* tp_dealloc */
(
destructor
)
BTree_dealloc
,
/* tp_dealloc */
0
,
/* tp_print */
0
,
/* tp_print */
0
,
/* tp_getattr */
0
,
/* tp_getattr */
0
,
/* tp_setattr */
0
,
/* tp_setattr */
0
,
/* tp_compare */
0
,
/* tp_compare */
0
,
/* tp_repr */
0
,
/* tp_repr */
&
BTree_as_number_for_nonzero
,
/* tp_as_number */
&
BTree_as_number_for_nonzero
,
/* tp_as_number */
&
BTree_as_sequence
,
/* tp_as_sequence */
&
BTree_as_sequence
,
/* tp_as_sequence */
&
BTree_as_mapping
,
/* tp_as_mapping */
&
BTree_as_mapping
,
/* tp_as_mapping */
0
,
/* tp_hash */
0
,
/* tp_hash */
0
,
/* tp_call */
0
,
/* tp_call */
0
,
/* tp_str */
0
,
/* tp_str */
0
,
/* tp_getattro */
0
,
/* tp_getattro */
0
,
/* tp_setattro */
0
,
/* tp_setattro */
0
,
/* tp_as_buffer */
0
,
/* tp_as_buffer */
Py_TPFLAGS_DEFAULT
|
Py_TPFLAGS_HAVE_GC
|
Py_TPFLAGS_DEFAULT
|
Py_TPFLAGS_HAVE_GC
|
Py_TPFLAGS_BASETYPE
,
/* tp_flags */
Py_TPFLAGS_BASETYPE
,
/* tp_flags */
0
,
/* tp_doc */
0
,
/* tp_doc */
(
traverseproc
)
BTree_traverse
,
/* tp_traverse */
(
traverseproc
)
BTree_traverse
,
/* tp_traverse */
(
inquiry
)
BTree_tp_clear
,
/* tp_clear */
(
inquiry
)
BTree_tp_clear
,
/* tp_clear */
0
,
/* tp_richcompare */
0
,
/* tp_richcompare */
0
,
/* tp_weaklistoffset */
0
,
/* tp_weaklistoffset */
(
getiterfunc
)
BTree_getiter
,
/* tp_iter */
(
getiterfunc
)
BTree_getiter
,
/* tp_iter */
0
,
/* tp_iternext */
0
,
/* tp_iternext */
BTree_methods
,
/* tp_methods */
BTree_methods
,
/* tp_methods */
BTree_members
,
/* tp_members */
BTree_members
,
/* tp_members */
0
,
/* tp_getset */
0
,
/* tp_getset */
0
,
/* tp_base */
0
,
/* tp_base */
0
,
/* tp_dict */
0
,
/* tp_dict */
0
,
/* tp_descr_get */
0
,
/* tp_descr_get */
0
,
/* tp_descr_set */
0
,
/* tp_descr_set */
0
,
/* tp_dictoffset */
0
,
/* tp_dictoffset */
BTree_init
,
/* tp_init */
BTree_init
,
/* tp_init */
0
,
/* tp_alloc */
0
,
/* tp_alloc */
0
,
/*PyType_GenericNew,*/
/* tp_new */
0
,
/*PyType_GenericNew,*/
/* tp_new */
};
};
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