Fix DBM filters
[p5sagit/p5-mst-13.2.git] / ext / ODBM_File / ODBM_File.xs
CommitLineData
463ee0b2 1#include "EXTERN.h"
2#include "perl.h"
3#include "XSUB.h"
4
8e07c86e 5#ifdef I_DBM
bb636fa4 6/* If using the DB3 emulation, ENTER is defined both
7 * by DB3 and Perl. We drop the Perl definition now.
8 * See also INSTALL section on DB3.
9 * -- Stanislav Brabec <utx@penguin.cz> */
10# undef ENTER
8e07c86e 11# include <dbm.h>
12#else
13# ifdef I_RPCSVC_DBM
14# include <rpcsvc/dbm.h>
15# endif
16#endif
463ee0b2 17
2ef53570 18#ifndef HAS_DBMINIT_PROTO
19int dbminit(char* filename);
20int dbmclose(void);
21datum fetch(datum key);
22int store(datum key, datum dat);
23int delete(datum key);
24datum firstkey(void);
25datum nextkey(datum key);
26#endif
27
1639c7b3 28#ifdef DBM_BUG_DUPLICATE_FREE
29/*
30 * DBM on at least Ultrix and HPUX call dbmclose() from dbminit(),
31 * resulting in duplicate free() because dbmclose() does *not*
32 * check if it has already been called for this DBM.
33 * If some malloc/free calls have been done between dbmclose() and
34 * the next dbminit(), the memory might be used for something else when
35 * it is freed.
36 * Verified to work on ultrix4.3. Probably will work on HP/UX.
37 * Set DBM_BUG_DUPLICATE_FREE in the extension hint file.
38 */
39/* Close the previous dbm, and fail to open a new dbm */
40#define dbmclose() ((void) dbminit("/tmp/x/y/z/z/y"))
41#endif
42
463ee0b2 43#include <fcntl.h>
44
9fe6733a 45typedef struct {
46 void * dbp ;
47 SV * filter_fetch_key ;
48 SV * filter_store_key ;
49 SV * filter_fetch_value ;
50 SV * filter_store_value ;
51 int filtering ;
52 } ODBM_File_type;
53
54typedef ODBM_File_type * ODBM_File ;
55typedef datum datum_key ;
0bf2e707 56typedef datum datum_key_copy ;
9fe6733a 57typedef datum datum_value ;
58
a0d0e21e 59#define odbm_FETCH(db,key) fetch(key)
60#define odbm_STORE(db,key,value,flags) store(key,value)
61#define odbm_DELETE(db,key) delete(key)
62#define odbm_FIRSTKEY(db) firstkey()
63#define odbm_NEXTKEY(db,key) nextkey(key)
463ee0b2 64
df3728a2 65#define MY_CXT_KEY "ODBM_File::_guts" XS_VERSION
66
67typedef struct {
68 int x_dbmrefcnt;
69} my_cxt_t;
70
71START_MY_CXT
72
73#define dbmrefcnt (MY_CXT.x_dbmrefcnt)
463ee0b2 74
85e6fe83 75#ifndef DBM_REPLACE
463ee0b2 76#define DBM_REPLACE 0
85e6fe83 77#endif
463ee0b2 78
79MODULE = ODBM_File PACKAGE = ODBM_File PREFIX = odbm_
80
df3728a2 81BOOT:
82{
83 MY_CXT_INIT;
84}
85
463ee0b2 86ODBM_File
a0d0e21e 87odbm_TIEHASH(dbtype, filename, flags, mode)
463ee0b2 88 char * dbtype
89 char * filename
90 int flags
91 int mode
92 CODE:
93 {
46fc3d4c 94 char *tmpbuf;
9fe6733a 95 void * dbp ;
df3728a2 96 dMY_CXT;
97
463ee0b2 98 if (dbmrefcnt++)
99 croak("Old dbm can only open one database");
46fc3d4c 100 New(0, tmpbuf, strlen(filename) + 5, char);
101 SAVEFREEPV(tmpbuf);
463ee0b2 102 sprintf(tmpbuf,"%s.dir",filename);
3280af22 103 if (stat(tmpbuf, &PL_statbuf) < 0) {
463ee0b2 104 if (flags & O_CREAT) {
105 if (mode < 0 || close(creat(tmpbuf,mode)) < 0)
106 croak("ODBM_File: Can't create %s", filename);
107 sprintf(tmpbuf,"%s.pag",filename);
108 if (close(creat(tmpbuf,mode)) < 0)
109 croak("ODBM_File: Can't create %s", filename);
110 }
111 else
112 croak("ODBM_FILE: Can't open %s", filename);
113 }
9fe6733a 114 dbp = (void*)(dbminit(filename) >= 0 ? &dbmrefcnt : 0);
115 RETVAL = (ODBM_File)safemalloc(sizeof(ODBM_File_type)) ;
116 Zero(RETVAL, 1, ODBM_File_type) ;
117 RETVAL->dbp = dbp ;
6b88bc9c 118 ST(0) = sv_mortalcopy(&PL_sv_undef);
56431972 119 sv_setptrobj(ST(0), RETVAL, dbtype);
463ee0b2 120 }
121
122void
123DESTROY(db)
124 ODBM_File db
df3728a2 125 PREINIT:
126 dMY_CXT;
463ee0b2 127 CODE:
128 dbmrefcnt--;
129 dbmclose();
eb99164f 130 safefree(db);
463ee0b2 131
1b882d32 132datum_value
a0d0e21e 133odbm_FETCH(db, key)
463ee0b2 134 ODBM_File db
0bf2e707 135 datum_key_copy key
463ee0b2 136
137int
a0d0e21e 138odbm_STORE(db, key, value, flags = DBM_REPLACE)
463ee0b2 139 ODBM_File db
9fe6733a 140 datum_key key
141 datum_value value
463ee0b2 142 int flags
a0d0e21e 143 CLEANUP:
144 if (RETVAL) {
145 if (RETVAL < 0 && errno == EPERM)
146 croak("No write permission to odbm file");
748a9306 147 croak("odbm store returned %d, errno %d, key \"%s\"",
a0d0e21e 148 RETVAL,errno,key.dptr);
149 }
463ee0b2 150
151int
a0d0e21e 152odbm_DELETE(db, key)
463ee0b2 153 ODBM_File db
9fe6733a 154 datum_key key
463ee0b2 155
9fe6733a 156datum_key
a0d0e21e 157odbm_FIRSTKEY(db)
463ee0b2 158 ODBM_File db
159
9fe6733a 160datum_key
a0d0e21e 161odbm_NEXTKEY(db, key)
463ee0b2 162 ODBM_File db
9fe6733a 163 datum_key key
164
165
166#define setFilter(type) \
167 { \
168 if (db->type) \
cad2e5aa 169 RETVAL = sv_mortalcopy(db->type) ; \
170 ST(0) = RETVAL ; \
9fe6733a 171 if (db->type && (code == &PL_sv_undef)) { \
172 SvREFCNT_dec(db->type) ; \
1b882d32 173 db->type = Nullsv ; \
9fe6733a 174 } \
175 else if (code) { \
176 if (db->type) \
177 sv_setsv(db->type, code) ; \
178 else \
179 db->type = newSVsv(code) ; \
180 } \
181 }
182
183
184
185SV *
186filter_fetch_key(db, code)
187 ODBM_File db
188 SV * code
189 SV * RETVAL = &PL_sv_undef ;
190 CODE:
6a31061a 191 DBM_setFilter(db->filter_fetch_key, code) ;
9fe6733a 192
193SV *
194filter_store_key(db, code)
195 ODBM_File db
196 SV * code
197 SV * RETVAL = &PL_sv_undef ;
198 CODE:
6a31061a 199 DBM_setFilter(db->filter_store_key, code) ;
9fe6733a 200
201SV *
202filter_fetch_value(db, code)
203 ODBM_File db
204 SV * code
205 SV * RETVAL = &PL_sv_undef ;
206 CODE:
6a31061a 207 DBM_setFilter(db->filter_fetch_value, code) ;
9fe6733a 208
209SV *
210filter_store_value(db, code)
211 ODBM_File db
212 SV * code
213 SV * RETVAL = &PL_sv_undef ;
214 CODE:
6a31061a 215 DBM_setFilter(db->filter_store_value, code) ;
463ee0b2 216