1 #pragma ident "%Z%%M% %I% %E% SMI"
4 * Copyright (c) 1990, 1993, 1994
5 * The Regents of the University of California. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. All advertising materials mentioning features or use of this software
16 * must display the following acknowledgement:
17 * This product includes software developed by the University of
18 * California, Berkeley and its contributors.
19 * 4. Neither the name of the University nor the names of its contributors
20 * may be used to endorse or promote products derived from this software
21 * without specific prior written permission.
23 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
24 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
27 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36 #if defined(LIBC_SCCS) && !defined(lint)
37 static char sccsid
[] = "@(#)rec_put.c 8.7 (Berkeley) 8/18/94";
38 #endif /* LIBC_SCCS and not lint */
40 #include <sys/types.h>
51 * __REC_PUT -- Add a recno item to the tree.
54 * dbp: pointer to access method
57 * flag: R_CURSOR, R_IAFTER, R_IBEFORE, R_NOOVERWRITE
60 * RET_ERROR, RET_SUCCESS and RET_SPECIAL if the key is
61 * already in the tree and R_NOOVERWRITE specified.
64 __rec_put(dbp
, key
, data
, flags
)
77 /* Toss any page pinned across calls. */
78 if (t
->bt_pinned
!= NULL
) {
79 mpool_put(t
->bt_mp
, t
->bt_pinned
, 0);
84 * If using fixed-length records, and the record is long, return
85 * EINVAL. If it's short, pad it out. Use the record data return
86 * memory, it's only short-term.
88 if (F_ISSET(t
, R_FIXLEN
) && data
->size
!= t
->bt_reclen
) {
89 if (data
->size
> t
->bt_reclen
)
92 if (t
->bt_rdata
.size
< t
->bt_reclen
) {
93 t
->bt_rdata
.data
= t
->bt_rdata
.data
== NULL
?
94 malloc(t
->bt_reclen
) :
95 realloc(t
->bt_rdata
.data
, t
->bt_reclen
);
96 if (t
->bt_rdata
.data
== NULL
)
98 t
->bt_rdata
.size
= t
->bt_reclen
;
100 memmove(t
->bt_rdata
.data
, data
->data
, data
->size
);
101 memset((char *)t
->bt_rdata
.data
+ data
->size
,
102 t
->bt_bval
, t
->bt_reclen
- data
->size
);
103 fdata
.data
= t
->bt_rdata
.data
;
104 fdata
.size
= t
->bt_reclen
;
106 fdata
.data
= data
->data
;
107 fdata
.size
= data
->size
;
112 if (!F_ISSET(&t
->bt_cursor
, CURS_INIT
))
114 nrec
= t
->bt_cursor
.rcursor
;
117 if ((nrec
= *(recno_t
*)key
->data
) == 0)
121 if ((nrec
= *(recno_t
*)key
->data
) == 0) {
128 if ((nrec
= *(recno_t
*)key
->data
) == 0)
132 if ((nrec
= *(recno_t
*)key
->data
) == 0)
134 if (nrec
<= t
->bt_nrecs
)
135 return (RET_SPECIAL
);
138 einval
: errno
= EINVAL
;
143 * Make sure that records up to and including the put record are
144 * already in the database. If skipping records, create empty ones.
146 if (nrec
> t
->bt_nrecs
) {
147 if (!F_ISSET(t
, R_EOF
| R_INMEM
) &&
148 t
->bt_irec(t
, nrec
) == RET_ERROR
)
150 if (nrec
> t
->bt_nrecs
+ 1) {
151 if (F_ISSET(t
, R_FIXLEN
)) {
153 (void *)malloc(t
->bt_reclen
)) == NULL
)
155 tdata
.size
= t
->bt_reclen
;
156 memset(tdata
.data
, t
->bt_bval
, tdata
.size
);
161 while (nrec
> t
->bt_nrecs
+ 1)
163 t
->bt_nrecs
, &tdata
, 0) != RET_SUCCESS
)
165 if (F_ISSET(t
, R_FIXLEN
))
170 if ((status
= __rec_iput(t
, nrec
- 1, &fdata
, flags
)) != RET_SUCCESS
)
173 if (flags
== R_SETCURSOR
)
174 t
->bt_cursor
.rcursor
= nrec
;
176 F_SET(t
, R_MODIFIED
);
177 return (__rec_ret(t
, NULL
, nrec
, key
, NULL
));
181 * __REC_IPUT -- Add a recno item to the tree.
185 * nrec: record number
189 * RET_ERROR, RET_SUCCESS
192 __rec_iput(t
, nrec
, data
, flags
)
201 indx_t idx
, nxtindex
;
205 char *dest
, db
[NOVFLSIZE
];
208 * If the data won't fit on a page, store it on indirect pages.
211 * If the insert fails later on, these pages aren't recovered.
213 if (data
->size
> t
->bt_ovflsize
) {
214 if (__ovfl_put(t
, data
, &pg
) == RET_ERROR
)
217 tdata
.size
= NOVFLSIZE
;
218 *(db_pgno_t
*)db
= pg
;
219 *(u_int32_t
*)(db
+ sizeof(db_pgno_t
)) = data
->size
;
225 /* __rec_search pins the returned page. */
226 if ((e
= __rec_search(t
, nrec
,
227 nrec
> t
->bt_nrecs
|| flags
== R_IAFTER
|| flags
== R_IBEFORE
?
228 SINSERT
: SEARCH
)) == NULL
)
235 * Add the specified key/data pair to the tree. The R_IAFTER and
236 * R_IBEFORE flags insert the key after/before the specified key.
238 * Pages are split as required.
247 if (nrec
< t
->bt_nrecs
&&
248 __rec_dleaf(t
, h
, idx
) == RET_ERROR
) {
249 mpool_put(t
->bt_mp
, h
, 0);
256 * If not enough room, split the page. The split code will insert
257 * the key and data and unpin the current page. If inserting into
258 * the offset array, shift the pointers up.
260 nbytes
= NRLEAFDBT(data
->size
);
261 if (h
->upper
- h
->lower
< nbytes
+ sizeof(indx_t
)) {
262 status
= __bt_split(t
, h
, NULL
, data
, dflags
, nbytes
, idx
);
263 if (status
== RET_SUCCESS
)
268 if (idx
< (nxtindex
= NEXTINDEX(h
)))
269 memmove(h
->linp
+ idx
+ 1, h
->linp
+ idx
,
270 (nxtindex
- idx
) * sizeof(indx_t
));
271 h
->lower
+= sizeof(indx_t
);
273 h
->linp
[idx
] = h
->upper
-= nbytes
;
274 dest
= (char *)h
+ h
->upper
;
275 WR_RLEAF(dest
, data
, dflags
);
278 F_SET(t
, B_MODIFIED
);
279 mpool_put(t
->bt_mp
, h
, MPOOL_DIRTY
);
281 return (RET_SUCCESS
);