1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2009 Lennart Poettering
5
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as
8 published by the Free Software Foundation; either version 2.1 of the
9 License, or (at your option) any later version.
10
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
15
16 You should have received a copy of the GNU Lesser General Public
17 License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
18 ***/
19
20 #ifdef HAVE_CONFIG_H
21 #include <config.h>
22 #endif
23
24 #include <fcntl.h>
25 #include <unistd.h>
26 #include <errno.h>
27
28 /* Some versions of tdb lack inclusion of signal.h in the header files but use sigatomic_t */
29 #include <signal.h>
30 #include <tdb.h>
31
32 #include <pulse/xmalloc.h>
33 #include <pulsecore/core-util.h>
34 #include <pulsecore/log.h>
35
36 #include "database.h"
37
38 #define MAKE_TDB_CONTEXT(x) ((struct tdb_context*) (x))
39
datum_to_tdb(TDB_DATA * to,const pa_datum * from)40 static inline TDB_DATA* datum_to_tdb(TDB_DATA *to, const pa_datum *from) {
41 pa_assert(from);
42 pa_assert(to);
43
44 to->dptr = from->data;
45 to->dsize = from->size;
46
47 return to;
48 }
49
datum_from_tdb(pa_datum * to,const TDB_DATA * from)50 static inline pa_datum* datum_from_tdb(pa_datum *to, const TDB_DATA *from) {
51 pa_assert(from);
52 pa_assert(to);
53
54 to->data = from->dptr;
55 to->size = from->dsize;
56
57 return to;
58 }
59
pa_datum_free(pa_datum * d)60 void pa_datum_free(pa_datum *d) {
61 pa_assert(d);
62
63 free(d->data); /* tdb uses raw malloc/free hence we should do that here, too */
64 pa_zero(d);
65 }
66
tdb_open_cloexec(const char * name,int hash_size,int tdb_flags,int open_flags,mode_t mode)67 static struct tdb_context *tdb_open_cloexec(
68 const char *name,
69 int hash_size,
70 int tdb_flags,
71 int open_flags,
72 mode_t mode) {
73
74 /* Mimics pa_open_cloexec() */
75
76 struct tdb_context *c;
77
78 #ifdef O_NOCTTY
79 open_flags |= O_NOCTTY;
80 #endif
81
82 #ifdef O_CLOEXEC
83 errno = 0;
84 if ((c = tdb_open(name, hash_size, tdb_flags, open_flags | O_CLOEXEC, mode)))
85 goto finish;
86
87 if (errno != EINVAL)
88 return NULL;
89 #endif
90
91 errno = 0;
92 if (!(c = tdb_open(name, hash_size, tdb_flags, open_flags, mode)))
93 return NULL;
94
95 finish:
96 pa_make_fd_cloexec(tdb_fd(c));
97 return c;
98 }
99
pa_database_get_filename_suffix(void)100 const char* pa_database_get_filename_suffix(void) {
101 return ".tdb";
102 }
103
pa_database_open_internal(const char * path,bool for_write)104 pa_database* pa_database_open_internal(const char *path, bool for_write) {
105 struct tdb_context *c;
106
107 pa_assert(path);
108
109 if ((c = tdb_open_cloexec(path, 0, TDB_NOSYNC|TDB_NOLOCK, (for_write ? O_RDWR|O_CREAT : O_RDONLY), 0644)))
110 pa_log_debug("Opened TDB database '%s'", path);
111
112 if (!c) {
113 if (errno == 0)
114 errno = EIO;
115 return NULL;
116 }
117
118 return (pa_database*) c;
119 }
120
pa_database_close(pa_database * db)121 void pa_database_close(pa_database *db) {
122 pa_assert(db);
123
124 tdb_close(MAKE_TDB_CONTEXT(db));
125 }
126
pa_database_get(pa_database * db,const pa_datum * key,pa_datum * data)127 pa_datum* pa_database_get(pa_database *db, const pa_datum *key, pa_datum* data) {
128 TDB_DATA tdb_key, tdb_data;
129
130 pa_assert(db);
131 pa_assert(key);
132 pa_assert(data);
133
134 tdb_data = tdb_fetch(MAKE_TDB_CONTEXT(db), *datum_to_tdb(&tdb_key, key));
135
136 return tdb_data.dptr ?
137 datum_from_tdb(data, &tdb_data) :
138 NULL;
139 }
140
pa_database_set(pa_database * db,const pa_datum * key,const pa_datum * data,bool overwrite)141 int pa_database_set(pa_database *db, const pa_datum *key, const pa_datum* data, bool overwrite) {
142 TDB_DATA tdb_key, tdb_data;
143
144 pa_assert(db);
145 pa_assert(key);
146 pa_assert(data);
147
148 return tdb_store(MAKE_TDB_CONTEXT(db),
149 *datum_to_tdb(&tdb_key, key),
150 *datum_to_tdb(&tdb_data, data),
151 overwrite ? TDB_REPLACE : TDB_INSERT) != 0 ? -1 : 0;
152 }
153
pa_database_unset(pa_database * db,const pa_datum * key)154 int pa_database_unset(pa_database *db, const pa_datum *key) {
155 TDB_DATA tdb_key;
156
157 pa_assert(db);
158 pa_assert(key);
159
160 return tdb_delete(MAKE_TDB_CONTEXT(db), *datum_to_tdb(&tdb_key, key)) != 0 ? -1 : 0;
161 }
162
pa_database_clear(pa_database * db)163 int pa_database_clear(pa_database *db) {
164 pa_assert(db);
165
166 return tdb_wipe_all(MAKE_TDB_CONTEXT(db)) != 0 ? -1 : 0;
167 }
168
pa_database_size(pa_database * db)169 signed pa_database_size(pa_database *db) {
170 TDB_DATA tdb_key;
171 unsigned n = 0;
172
173 pa_assert(db);
174
175 /* This sucks */
176
177 tdb_key = tdb_firstkey(MAKE_TDB_CONTEXT(db));
178
179 while (tdb_key.dptr) {
180 TDB_DATA next;
181
182 n++;
183
184 next = tdb_nextkey(MAKE_TDB_CONTEXT(db), tdb_key);
185 free(tdb_key.dptr);
186 tdb_key = next;
187 }
188
189 return (signed) n;
190 }
191
pa_database_first(pa_database * db,pa_datum * key,pa_datum * data)192 pa_datum* pa_database_first(pa_database *db, pa_datum *key, pa_datum *data) {
193 TDB_DATA tdb_key, tdb_data;
194
195 pa_assert(db);
196 pa_assert(key);
197
198 tdb_key = tdb_firstkey(MAKE_TDB_CONTEXT(db));
199
200 if (!tdb_key.dptr)
201 return NULL;
202
203 if (data) {
204 tdb_data = tdb_fetch(MAKE_TDB_CONTEXT(db), tdb_key);
205
206 if (!tdb_data.dptr) {
207 free(tdb_key.dptr);
208 return NULL;
209 }
210
211 datum_from_tdb(data, &tdb_data);
212 }
213
214 datum_from_tdb(key, &tdb_key);
215
216 return key;
217 }
218
pa_database_next(pa_database * db,const pa_datum * key,pa_datum * next,pa_datum * data)219 pa_datum* pa_database_next(pa_database *db, const pa_datum *key, pa_datum *next, pa_datum *data) {
220 TDB_DATA tdb_key, tdb_data;
221
222 pa_assert(db);
223 pa_assert(key);
224
225 tdb_key = tdb_nextkey(MAKE_TDB_CONTEXT(db), *datum_to_tdb(&tdb_key, key));
226
227 if (!tdb_key.dptr)
228 return NULL;
229
230 if (data) {
231 tdb_data = tdb_fetch(MAKE_TDB_CONTEXT(db), tdb_key);
232
233 if (!tdb_data.dptr) {
234 free(tdb_key.dptr);
235 return NULL;
236 }
237
238 datum_from_tdb(data, &tdb_data);
239 }
240
241 datum_from_tdb(next, &tdb_key);
242
243 return next;
244 }
245
pa_database_sync(pa_database * db)246 int pa_database_sync(pa_database *db) {
247 pa_assert(db);
248
249 return 0;
250 }
251