ctdb: Make TDB_SEQNUM work synchronously with ctdb
[samba.git] / ctdb / common / ctdb_ltdb.c
1 /* 
2    ctdb ltdb code
3
4    Copyright (C) Andrew Tridgell  2006
5    Copyright (C) Ronnie sahlberg  2011
6
7    This program is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 3 of the License, or
10    (at your option) any later version.
11    
12    This program is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16    
17    You should have received a copy of the GNU General Public License
18    along with this program; if not, see <http://www.gnu.org/licenses/>.
19 */
20
21 #include "replace.h"
22 #include "system/network.h"
23 #include "system/filesys.h"
24
25 #include <tdb.h>
26
27 #include "lib/tdb_wrap/tdb_wrap.h"
28 #include "lib/util/dlinklist.h"
29 #include "lib/util/debug.h"
30
31 #include "ctdb_private.h"
32
33 #include "common/common.h"
34 #include "common/logging.h"
35
36
37 /*
38  * Calculate tdb flags based on databse type
39  */
40 int ctdb_db_tdb_flags(uint8_t db_flags, bool with_valgrind, bool with_mutex)
41 {
42         int tdb_flags = 0;
43
44         if (db_flags & CTDB_DB_FLAGS_PERSISTENT) {
45                 tdb_flags = TDB_DEFAULT;
46
47         } else if (db_flags & CTDB_DB_FLAGS_REPLICATED) {
48                 tdb_flags = TDB_NOSYNC |
49                             TDB_CLEAR_IF_FIRST |
50                             TDB_INCOMPATIBLE_HASH;
51
52         } else {
53                 tdb_flags = TDB_NOSYNC |
54                             TDB_CLEAR_IF_FIRST |
55                             TDB_INCOMPATIBLE_HASH;
56
57 #ifdef TDB_MUTEX_LOCKING
58                 if (with_mutex && tdb_runtime_check_for_robust_mutexes()) {
59                         tdb_flags |= TDB_MUTEX_LOCKING;
60                 }
61 #endif
62
63         }
64
65         tdb_flags |= TDB_DISALLOW_NESTING;
66         if (with_valgrind) {
67                 tdb_flags |= TDB_NOMMAP;
68         }
69
70         return tdb_flags;
71 }
72
73 /*
74   find an attached ctdb_db handle given a name
75  */
76 struct ctdb_db_context *ctdb_db_handle(struct ctdb_context *ctdb, const char *name)
77 {
78         struct ctdb_db_context *tmp_db;
79         for (tmp_db=ctdb->db_list;tmp_db;tmp_db=tmp_db->next) {
80                 if (strcmp(name, tmp_db->db_name) == 0) {
81                         return tmp_db;
82                 }
83         }
84         return NULL;
85 }
86
87 bool ctdb_db_persistent(struct ctdb_db_context *ctdb_db)
88 {
89         if (ctdb_db->db_flags & CTDB_DB_FLAGS_PERSISTENT) {
90                 return true;
91         }
92         return false;
93 }
94
95 bool ctdb_db_replicated(struct ctdb_db_context *ctdb_db)
96 {
97         if (ctdb_db->db_flags & CTDB_DB_FLAGS_REPLICATED) {
98                 return true;
99         }
100         return false;
101 }
102
103 bool ctdb_db_volatile(struct ctdb_db_context *ctdb_db)
104 {
105         if ((ctdb_db->db_flags & CTDB_DB_FLAGS_PERSISTENT) ||
106             (ctdb_db->db_flags & CTDB_DB_FLAGS_REPLICATED)) {
107                 return false;
108         }
109         return true;
110 }
111
112 bool ctdb_db_readonly(struct ctdb_db_context *ctdb_db)
113 {
114         if (ctdb_db->db_flags & CTDB_DB_FLAGS_READONLY) {
115                 return true;
116         }
117         return false;
118 }
119
120 void ctdb_db_set_readonly(struct ctdb_db_context *ctdb_db)
121 {
122         ctdb_db->db_flags |= CTDB_DB_FLAGS_READONLY;
123 }
124
125 void ctdb_db_reset_readonly(struct ctdb_db_context *ctdb_db)
126 {
127         ctdb_db->db_flags &= ~CTDB_DB_FLAGS_READONLY;
128 }
129
130 bool ctdb_db_sticky(struct ctdb_db_context *ctdb_db)
131 {
132         if (ctdb_db->db_flags & CTDB_DB_FLAGS_STICKY) {
133                 return true;
134         }
135         return false;
136 }
137
138 void ctdb_db_set_sticky(struct ctdb_db_context *ctdb_db)
139 {
140         ctdb_db->db_flags |= CTDB_DB_FLAGS_STICKY;
141 }
142
143 /*
144   return the lmaster given a key
145 */
146 uint32_t ctdb_lmaster(struct ctdb_context *ctdb, const TDB_DATA *key)
147 {
148         uint32_t idx, lmaster;
149
150         idx = ctdb_hash(key) % ctdb->vnn_map->size;
151         lmaster = ctdb->vnn_map->map[idx];
152
153         return lmaster;
154 }
155
156
157 /*
158   construct an initial header for a record with no ltdb header yet
159 */
160 static void ltdb_initial_header(struct ctdb_db_context *ctdb_db, 
161                                 TDB_DATA key,
162                                 struct ctdb_ltdb_header *header)
163 {
164         ZERO_STRUCTP(header);
165         /* initial dmaster is the lmaster */
166         header->dmaster = ctdb_lmaster(ctdb_db->ctdb, &key);
167         header->flags = CTDB_REC_FLAG_AUTOMATIC;
168 }
169
170
171 /*
172   fetch a record from the ltdb, separating out the header information
173   and returning the body of the record. A valid (initial) header is
174   returned if the record is not present
175 */
176 int ctdb_ltdb_fetch(struct ctdb_db_context *ctdb_db, 
177                     TDB_DATA key, struct ctdb_ltdb_header *header, 
178                     TALLOC_CTX *mem_ctx, TDB_DATA *data)
179 {
180         TDB_DATA rec;
181         struct ctdb_context *ctdb = ctdb_db->ctdb;
182
183         rec = tdb_fetch(ctdb_db->ltdb->tdb, key);
184         if (rec.dsize < sizeof(*header)) {
185                 /* return an initial header */
186                 if (rec.dptr) free(rec.dptr);
187                 if (ctdb->vnn_map == NULL) {
188                         /* called from the client */
189                         ZERO_STRUCTP(data);
190                         header->dmaster = (uint32_t)-1;
191                         return -1;
192                 }
193                 ltdb_initial_header(ctdb_db, key, header);
194                 if (data) {
195                         *data = tdb_null;
196                 }
197                 if (ctdb_db_persistent(ctdb_db) ||
198                     header->dmaster == ctdb_db->ctdb->pnn) {
199                         if (ctdb_ltdb_store(ctdb_db, key, header, tdb_null) != 0) {
200                                 DEBUG(DEBUG_NOTICE,
201                                       (__location__ "failed to store initial header\n"));
202                         }
203                 }
204                 return 0;
205         }
206
207         *header = *(struct ctdb_ltdb_header *)rec.dptr;
208
209         if (data) {
210                 data->dsize = rec.dsize - sizeof(struct ctdb_ltdb_header);
211                 data->dptr = talloc_memdup(mem_ctx, 
212                                            sizeof(struct ctdb_ltdb_header)+rec.dptr,
213                                            data->dsize);
214         }
215
216         free(rec.dptr);
217         if (data) {
218                 CTDB_NO_MEMORY(ctdb, data->dptr);
219         }
220
221         return 0;
222 }
223
224 /*
225   write a record to a normal database
226 */
227 int ctdb_ltdb_store(struct ctdb_db_context *ctdb_db, TDB_DATA key, 
228                     struct ctdb_ltdb_header *header, TDB_DATA data)
229 {
230         struct ctdb_context *ctdb = ctdb_db->ctdb;
231         TDB_DATA rec[2];
232         uint32_t hsize = sizeof(struct ctdb_ltdb_header);
233         int ret;
234
235         if (ctdb_db->ctdb_ltdb_store_fn) {
236                 return ctdb_db->ctdb_ltdb_store_fn(ctdb_db, key, header, data);
237         }
238
239         if (ctdb->flags & CTDB_FLAG_TORTURE) {
240                 TDB_DATA old;
241                 struct ctdb_ltdb_header *h2;
242
243                 old = tdb_fetch(ctdb_db->ltdb->tdb, key);
244                 h2 = (struct ctdb_ltdb_header *)old.dptr;
245                 if (old.dptr != NULL && old.dsize >= hsize &&
246                     h2->rsn > header->rsn) {
247                         DEBUG(DEBUG_ERR,
248                               ("RSN regression! %"PRIu64" %"PRIu64"\n",
249                                h2->rsn, header->rsn));
250                 }
251                 if (old.dptr != NULL) {
252                         free(old.dptr);
253                 }
254         }
255
256         rec[0].dsize = hsize;
257         rec[0].dptr = (uint8_t *)header;
258
259         rec[1].dsize = data.dsize;
260         rec[1].dptr = data.dptr;
261
262         ret = tdb_storev(ctdb_db->ltdb->tdb, key, rec, 2, TDB_REPLACE);
263         if (ret != 0) {
264                 DEBUG(DEBUG_ERR, (__location__ " Failed to store dynamic data\n"));
265         }
266
267         return ret;
268 }
269
270 /*
271   lock a record in the ltdb, given a key
272  */
273 int ctdb_ltdb_lock(struct ctdb_db_context *ctdb_db, TDB_DATA key)
274 {
275         return tdb_chainlock(ctdb_db->ltdb->tdb, key);
276 }
277
278 /*
279   unlock a record in the ltdb, given a key
280  */
281 int ctdb_ltdb_unlock(struct ctdb_db_context *ctdb_db, TDB_DATA key)
282 {
283         int ret = tdb_chainunlock(ctdb_db->ltdb->tdb, key);
284         if (ret != 0) {
285                 DEBUG(DEBUG_ERR,("tdb_chainunlock failed on db %s [%s]\n", ctdb_db->db_name, tdb_errorstr(ctdb_db->ltdb->tdb)));
286         }
287         return ret;
288 }
289
290
291 /*
292   delete a record from a normal database
293 */
294 int ctdb_ltdb_delete(struct ctdb_db_context *ctdb_db, TDB_DATA key)
295 {
296         if (! ctdb_db_volatile(ctdb_db)) {
297                 DEBUG(DEBUG_WARNING,
298                       ("Ignored deletion of empty record from "
299                        "non-volatile database\n"));
300                 return 0;
301         }
302         if (tdb_delete(ctdb_db->ltdb->tdb, key) != 0) {
303                 DEBUG(DEBUG_ERR,("Failed to delete empty record."));
304                 return -1;
305         }
306         return 0;
307 }
308
309 int ctdb_trackingdb_add_pnn(struct ctdb_context *ctdb, TDB_DATA *data, uint32_t pnn)
310 {
311         int byte_pos = pnn / 8;
312         int bit_mask   = 1 << (pnn % 8);
313
314         if (byte_pos + 1 > data->dsize) {
315                 char *buf;
316
317                 buf = malloc(byte_pos + 1);
318                 memset(buf, 0, byte_pos + 1);
319                 if (buf == NULL) {
320                         DEBUG(DEBUG_ERR, ("Out of memory when allocating buffer of %d bytes for trackingdb\n", byte_pos + 1));
321                         return -1;
322                 }
323                 if (data->dptr != NULL) {
324                         memcpy(buf, data->dptr, data->dsize);
325                         free(data->dptr);
326                 }
327                 data->dptr  = (uint8_t *)buf;
328                 data->dsize = byte_pos + 1;
329         }
330
331         data->dptr[byte_pos] |= bit_mask;
332         return 0;
333 }
334
335 void ctdb_trackingdb_traverse(struct ctdb_context *ctdb, TDB_DATA data, ctdb_trackingdb_cb cb, void *private_data)
336 {
337         int i;
338
339         for(i = 0; i < data.dsize; i++) {
340                 int j;
341
342                 for (j=0; j<8; j++) {
343                         int mask = 1<<j;
344
345                         if (data.dptr[i] & mask) {
346                                 cb(ctdb, i * 8 + j, private_data);
347                         }
348                 }
349         }
350 }
351
352 /*
353   this is the dummy null procedure that all databases support
354 */
355 int ctdb_null_func(struct ctdb_call_info *call)
356 {
357         return 0;
358 }
359
360 /*
361   this is a plain fetch procedure that all databases support
362 */
363 int ctdb_fetch_func(struct ctdb_call_info *call)
364 {
365         call->reply_data = &call->record_data;
366         return 0;
367 }
368
369 /*
370   this is a plain fetch procedure that all databases support
371   this returns the full record including the ltdb header
372 */
373 int ctdb_fetch_with_header_func(struct ctdb_call_info *call)
374 {
375         call->reply_data = talloc(call, TDB_DATA);
376         if (call->reply_data == NULL) {
377                 return -1;
378         }
379         call->reply_data->dsize = sizeof(struct ctdb_ltdb_header) + call->record_data.dsize;
380         call->reply_data->dptr  = talloc_size(call->reply_data, call->reply_data->dsize);
381         if (call->reply_data->dptr == NULL) {
382                 return -1;
383         }
384         memcpy(call->reply_data->dptr, call->header, sizeof(struct ctdb_ltdb_header));
385         memcpy(&call->reply_data->dptr[sizeof(struct ctdb_ltdb_header)], call->record_data.dptr, call->record_data.dsize);
386
387         return 0;
388 }
389