2 * Copyright (c) 1990 The Regents of the University of California.
5 * This code is derived from software contributed to Berkeley by
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 * must display the following acknowledgement:
18 * This product includes software developed by the University of
19 * California, Berkeley and its contributors.
20 * 4. Neither the name of the University nor the names of its contributors
21 * may be used to endorse or promote products derived from this software
22 * without specific prior written permission.
24 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 #if defined(LIBC_SCCS) && !defined(lint)
38 static char sccsid
[] = "@(#)delete.c 5.2 (Berkeley) 2/22/91";
39 #endif /* LIBC_SCCS and not lint */
41 #include <sys/types.h>
48 * _BT_CRSRDEL -- Delete the item pointed to by the cursor.
50 * This routine deletes the item most recently returned by a scan
51 * through the tree. Since it only makes sense to delete the current
52 * record once, we make sure that we don't try to delete twice without
56 * t -- tree in which to do deletion
59 * RET_SUCCESS, RET_ERROR.
62 * The call to _bt_delone marks the cursor, so we can tell that
63 * the current record has been deleted.
74 /* a cursor must exist, and can't have deleted the current key yet */
75 if (!(t
->bt_flags
& BTF_SEQINIT
) || (c
->c_flags
& CRSR_BEFORE
)) {
80 if (_bt_getpage(t
, c
->c_pgno
) == RET_ERROR
)
83 if (c
->c_index
>= NEXTINDEX(t
->bt_curpage
)) {
88 return (_bt_delone(t
, c
->c_index
));
92 * _BT_DELONE -- Delete a single entry from a btree.
94 * This routine physically removes a btree entry from a leaf page.
95 * IDATUM items are *never* removed from internal nodes, regardless
96 * of whether the entries that originally caused them to be added
97 * are removed from the tree or not. In addition, pages made empty
98 * by element deletion are not actually reclaimed. They are,
99 * however, made available for reuse.
101 * To delete an item from a page, we pack the remaining items at
102 * the end of the page, overwriting the deleted item's entry. We
103 * move the line pointers backward on the page, overwriting the
104 * original item's line pointer. This guarantees that the space in
105 * the middle of the page is free -- a property that our insertion
106 * strategy relies on.
108 * This routine doesn't reclaim pages all of whose entries have
109 * been deleted. These pages are available for reuse, however.
110 * If an item is deleted that was too big to fit on a page, then
111 * the blocks that it occupies are put on a free list for reuse.
114 * t -- btree from which to delete item
115 * index -- index of entry on current page to delete
118 * RET_SUCCESS, RET_ERROR.
121 * Physically changes page layout, adjusts internal page
122 * state to reflect the deletion of the item, and updates
123 * the list of free pages for this tree.
141 /* deletion may confuse an active scan. fix it. */
143 if (t
->bt_flags
& BTF_SEQINIT
&& t
->bt_curpage
->h_pgno
== c
->c_pgno
)
144 if (_bt_fixscan(t
, index
, (DATUM
*) NULL
, DELETE
) == RET_ERROR
)
148 off
= h
->h_linp
[index
];
149 d
= (DATUM
*) GETDATUM(h
, index
);
151 /* if this is a big item, reclaim the space it occupies */
152 if (d
->d_flags
& D_BIGKEY
) {
153 bcopy(&(d
->d_bytes
[0]),
156 if (_bt_delindir(t
, chain
) == RET_ERROR
)
159 d
= (DATUM
*) GETDATUM(h
, index
);
161 if (d
->d_flags
& D_BIGDATA
) {
162 bcopy(&(d
->d_bytes
[d
->d_ksize
]),
165 if (_bt_delindir(t
, chain
) == RET_ERROR
)
168 d
= (DATUM
*) GETDATUM(h
, index
);
171 /* move the data down on the page */
172 nbytes
= d
->d_ksize
+ d
->d_dsize
173 + (sizeof(DATUM
) - sizeof(char));
174 nbytes
= LONGALIGN(nbytes
);
175 src
= ((char *) h
) + h
->h_upper
;
177 nmoved
= (int) (((char *) d
) - src
);
178 (void) bcopy(src
, dest
, nmoved
);
180 /* next move the line pointers up */
181 src
= (char *) &(h
->h_linp
[index
+ 1]);
182 dest
= (char *) &(h
->h_linp
[index
]);
183 nmoved
= (int) (((char *) &(h
->h_linp
[NEXTINDEX(h
)])) - src
);
184 (void) bcopy(src
, dest
, nmoved
);
186 /* remember that we freed up some space */
187 h
->h_upper
+= nbytes
;
188 h
->h_lower
-= sizeof(index_t
);
190 /* adjust offsets in line pointers affected by moving the data */
192 for (i
= 0; i
< top
; i
++) {
193 if (h
->h_linp
[i
] < off
)
194 h
->h_linp
[i
] += nbytes
;
198 h
->h_flags
|= F_DIRTY
;
200 return (RET_SUCCESS
);