LCOV - code coverage report
Current view: top level - source3/lib - g_lock.c (source / functions) Hit Total Coverage
Test: coverage report for support-claim-type-attributes 6b5c566e Lines: 699 880 79.4 %
Date: 2023-11-21 12:31:41 Functions: 44 47 93.6 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/CIFS implementation.
       3             :    global locks based on dbwrap and messaging
       4             :    Copyright (C) 2009 by Volker Lendecke
       5             : 
       6             :    This program is free software; you can redistribute it and/or modify
       7             :    it under the terms of the GNU General Public License as published by
       8             :    the Free Software Foundation; either version 3 of the License, or
       9             :    (at your option) any later version.
      10             : 
      11             :    This program is distributed in the hope that it will be useful,
      12             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      13             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      14             :    GNU General Public License for more details.
      15             : 
      16             :    You should have received a copy of the GNU General Public License
      17             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      18             : */
      19             : 
      20             : #include "replace.h"
      21             : #include "system/filesys.h"
      22             : #include "lib/util/server_id.h"
      23             : #include "lib/util/debug.h"
      24             : #include "lib/util/talloc_stack.h"
      25             : #include "lib/util/samba_util.h"
      26             : #include "lib/util_path.h"
      27             : #include "dbwrap/dbwrap.h"
      28             : #include "dbwrap/dbwrap_open.h"
      29             : #include "dbwrap/dbwrap_watch.h"
      30             : #include "g_lock.h"
      31             : #include "util_tdb.h"
      32             : #include "../lib/util/tevent_ntstatus.h"
      33             : #include "messages.h"
      34             : #include "serverid.h"
      35             : 
      36             : struct g_lock_ctx {
      37             :         struct db_context *db;
      38             :         struct messaging_context *msg;
      39             :         enum dbwrap_lock_order lock_order;
      40             :         bool busy;
      41             : };
      42             : 
      43             : struct g_lock {
      44             :         struct server_id exclusive;
      45             :         size_t num_shared;
      46             :         uint8_t *shared;
      47             :         uint64_t unique_lock_epoch;
      48             :         uint64_t unique_data_epoch;
      49             :         size_t datalen;
      50             :         uint8_t *data;
      51             : };
      52             : 
      53     1873746 : static bool g_lock_parse(uint8_t *buf, size_t buflen, struct g_lock *lck)
      54             : {
      55        4910 :         struct server_id exclusive;
      56        4910 :         size_t num_shared, shared_len;
      57        4910 :         uint64_t unique_lock_epoch;
      58        4910 :         uint64_t unique_data_epoch;
      59             : 
      60     1873746 :         if (buflen < (SERVER_ID_BUF_LENGTH + /* exclusive */
      61             :                       sizeof(uint64_t) +     /* seqnum */
      62             :                       sizeof(uint32_t))) {   /* num_shared */
      63     1242870 :                 struct g_lock ret = {
      64             :                         .exclusive.pid = 0,
      65      414290 :                         .unique_lock_epoch = generate_unique_u64(0),
      66      414290 :                         .unique_data_epoch = generate_unique_u64(0),
      67             :                 };
      68      414290 :                 *lck = ret;
      69      414290 :                 return true;
      70             :         }
      71             : 
      72     1459456 :         server_id_get(&exclusive, buf);
      73     1459456 :         buf += SERVER_ID_BUF_LENGTH;
      74     1459456 :         buflen -= SERVER_ID_BUF_LENGTH;
      75             : 
      76     1459456 :         unique_lock_epoch = BVAL(buf, 0);
      77     1459456 :         buf += sizeof(uint64_t);
      78     1459456 :         buflen -= sizeof(uint64_t);
      79             : 
      80     1459456 :         unique_data_epoch = BVAL(buf, 0);
      81     1459456 :         buf += sizeof(uint64_t);
      82     1459456 :         buflen -= sizeof(uint64_t);
      83             : 
      84     1459456 :         num_shared = IVAL(buf, 0);
      85     1459456 :         buf += sizeof(uint32_t);
      86     1459456 :         buflen -= sizeof(uint32_t);
      87             : 
      88     1459456 :         if (num_shared > buflen/SERVER_ID_BUF_LENGTH) {
      89           0 :                 DBG_DEBUG("num_shared=%zu, buflen=%zu\n",
      90             :                           num_shared,
      91             :                           buflen);
      92           0 :                 return false;
      93             :         }
      94             : 
      95     1459456 :         shared_len = num_shared * SERVER_ID_BUF_LENGTH;
      96             : 
      97     1459456 :         *lck = (struct g_lock) {
      98             :                 .exclusive = exclusive,
      99             :                 .num_shared = num_shared,
     100             :                 .shared = buf,
     101             :                 .unique_lock_epoch = unique_lock_epoch,
     102             :                 .unique_data_epoch = unique_data_epoch,
     103     1459456 :                 .datalen = buflen-shared_len,
     104     1459456 :                 .data = buf+shared_len,
     105             :         };
     106             : 
     107     1459456 :         return true;
     108             : }
     109             : 
     110          81 : static void g_lock_get_shared(const struct g_lock *lck,
     111             :                               size_t i,
     112             :                               struct server_id *shared)
     113             : {
     114          72 :         if (i >= lck->num_shared) {
     115           0 :                 abort();
     116             :         }
     117          77 :         server_id_get(shared, lck->shared + i*SERVER_ID_BUF_LENGTH);
     118          72 : }
     119             : 
     120          20 : static void g_lock_del_shared(struct g_lock *lck, size_t i)
     121             : {
     122          20 :         if (i >= lck->num_shared) {
     123           0 :                 abort();
     124             :         }
     125          20 :         lck->num_shared -= 1;
     126          20 :         if (i < lck->num_shared) {
     127          23 :                 memcpy(lck->shared + i*SERVER_ID_BUF_LENGTH,
     128           5 :                        lck->shared + lck->num_shared*SERVER_ID_BUF_LENGTH,
     129             :                        SERVER_ID_BUF_LENGTH);
     130             :         }
     131          20 : }
     132             : 
     133     1183329 : static NTSTATUS g_lock_store(
     134             :         struct db_record *rec,
     135             :         struct g_lock *lck,
     136             :         struct server_id *new_shared,
     137             :         const TDB_DATA *new_dbufs,
     138             :         size_t num_new_dbufs)
     139     1183329 : {
     140        2509 :         uint8_t exclusive[SERVER_ID_BUF_LENGTH];
     141        2509 :         uint8_t seqnum_buf[sizeof(uint64_t)*2];
     142        2509 :         uint8_t sizebuf[sizeof(uint32_t)];
     143        2509 :         uint8_t new_shared_buf[SERVER_ID_BUF_LENGTH];
     144             : 
     145     1183329 :         struct TDB_DATA dbufs[6 + num_new_dbufs];
     146             : 
     147     1183329 :         dbufs[0] = (TDB_DATA) {
     148             :                 .dptr = exclusive, .dsize = sizeof(exclusive),
     149             :         };
     150     1183329 :         dbufs[1] = (TDB_DATA) {
     151             :                 .dptr = seqnum_buf, .dsize = sizeof(seqnum_buf),
     152             :         };
     153     1183329 :         dbufs[2] = (TDB_DATA) {
     154             :                 .dptr = sizebuf, .dsize = sizeof(sizebuf),
     155             :         };
     156     1183329 :         dbufs[3] = (TDB_DATA) {
     157     1183329 :                 .dptr = lck->shared,
     158     1183329 :                 .dsize = lck->num_shared * SERVER_ID_BUF_LENGTH,
     159             :         };
     160     1183329 :         dbufs[4] = (TDB_DATA) { 0 };
     161     1183329 :         dbufs[5] = (TDB_DATA) {
     162     1183329 :                 .dptr = lck->data, .dsize = lck->datalen,
     163             :         };
     164             : 
     165     1183329 :         if (num_new_dbufs != 0) {
     166        1937 :                 memcpy(&dbufs[6],
     167             :                        new_dbufs,
     168             :                        num_new_dbufs * sizeof(TDB_DATA));
     169             :         }
     170             : 
     171     1183329 :         server_id_put(exclusive, lck->exclusive);
     172     1183329 :         SBVAL(seqnum_buf, 0, lck->unique_lock_epoch);
     173     1183329 :         SBVAL(seqnum_buf, 8, lck->unique_data_epoch);
     174             : 
     175     1183329 :         if (new_shared != NULL) {
     176          20 :                 if (lck->num_shared >= UINT32_MAX) {
     177           0 :                         return NT_STATUS_BUFFER_OVERFLOW;
     178             :                 }
     179             : 
     180          20 :                 server_id_put(new_shared_buf, *new_shared);
     181             : 
     182          20 :                 dbufs[4] = (TDB_DATA) {
     183             :                         .dptr = new_shared_buf,
     184             :                         .dsize = sizeof(new_shared_buf),
     185             :                 };
     186             : 
     187          20 :                 lck->num_shared += 1;
     188             :         }
     189             : 
     190     1183329 :         SIVAL(sizebuf, 0, lck->num_shared);
     191             : 
     192     1183329 :         return dbwrap_record_storev(rec, dbufs, ARRAY_SIZE(dbufs), 0);
     193             : }
     194             : 
     195         349 : struct g_lock_ctx *g_lock_ctx_init_backend(
     196             :         TALLOC_CTX *mem_ctx,
     197             :         struct messaging_context *msg,
     198             :         struct db_context **backend)
     199             : {
     200          21 :         struct g_lock_ctx *result;
     201             : 
     202         349 :         result = talloc_zero(mem_ctx, struct g_lock_ctx);
     203         349 :         if (result == NULL) {
     204           0 :                 return NULL;
     205             :         }
     206         349 :         result->msg = msg;
     207         349 :         result->lock_order = DBWRAP_LOCK_ORDER_NONE;
     208             : 
     209         349 :         result->db = db_open_watched(result, backend, msg);
     210         349 :         if (result->db == NULL) {
     211           0 :                 DBG_WARNING("db_open_watched failed\n");
     212           0 :                 TALLOC_FREE(result);
     213           0 :                 return NULL;
     214             :         }
     215         328 :         return result;
     216             : }
     217             : 
     218         188 : void g_lock_set_lock_order(struct g_lock_ctx *ctx,
     219             :                            enum dbwrap_lock_order lock_order)
     220             : {
     221         188 :         ctx->lock_order = lock_order;
     222         188 : }
     223             : 
     224         161 : struct g_lock_ctx *g_lock_ctx_init(TALLOC_CTX *mem_ctx,
     225             :                                    struct messaging_context *msg)
     226             : {
     227         161 :         char *db_path = NULL;
     228         161 :         struct db_context *backend = NULL;
     229         161 :         struct g_lock_ctx *ctx = NULL;
     230             : 
     231         161 :         db_path = lock_path(mem_ctx, "g_lock.tdb");
     232         161 :         if (db_path == NULL) {
     233           0 :                 return NULL;
     234             :         }
     235             : 
     236         161 :         backend = db_open(
     237             :                 mem_ctx,
     238             :                 db_path,
     239             :                 0,
     240             :                 TDB_CLEAR_IF_FIRST|TDB_INCOMPATIBLE_HASH|TDB_VOLATILE,
     241             :                 O_RDWR|O_CREAT,
     242             :                 0600,
     243             :                 DBWRAP_LOCK_ORDER_3,
     244             :                 DBWRAP_FLAG_NONE);
     245         161 :         TALLOC_FREE(db_path);
     246         161 :         if (backend == NULL) {
     247           0 :                 DBG_WARNING("Could not open g_lock.tdb\n");
     248           0 :                 return NULL;
     249             :         }
     250             : 
     251         161 :         ctx = g_lock_ctx_init_backend(mem_ctx, msg, &backend);
     252         161 :         return ctx;
     253             : }
     254             : 
     255         321 : static void g_lock_cleanup_dead(
     256             :         struct g_lock *lck,
     257             :         struct server_id *dead_blocker)
     258             : {
     259          15 :         bool exclusive_died;
     260          15 :         struct server_id_buf tmp;
     261             : 
     262         321 :         if (dead_blocker == NULL) {
     263         317 :                 return;
     264             :         }
     265             : 
     266           4 :         exclusive_died = server_id_equal(dead_blocker, &lck->exclusive);
     267             : 
     268           4 :         if (exclusive_died) {
     269           1 :                 DBG_DEBUG("Exclusive holder %s died\n",
     270             :                           server_id_str_buf(lck->exclusive, &tmp));
     271           1 :                 lck->exclusive.pid = 0;
     272             :         }
     273             : 
     274           4 :         if (lck->num_shared != 0) {
     275           3 :                 bool shared_died;
     276           3 :                 struct server_id shared;
     277             : 
     278           3 :                 g_lock_get_shared(lck, 0, &shared);
     279           3 :                 shared_died = server_id_equal(dead_blocker, &shared);
     280             : 
     281           3 :                 if (shared_died) {
     282           3 :                         DBG_DEBUG("Shared holder %s died\n",
     283             :                                   server_id_str_buf(shared, &tmp));
     284           3 :                         g_lock_del_shared(lck, 0);
     285             :                 }
     286             :         }
     287             : }
     288             : 
     289         302 : static ssize_t g_lock_find_shared(
     290             :         struct g_lock *lck,
     291             :         const struct server_id *self)
     292             : {
     293          11 :         size_t i;
     294             : 
     295         311 :         for (i=0; i<lck->num_shared; i++) {
     296          13 :                 struct server_id shared;
     297          13 :                 bool same;
     298             : 
     299          13 :                 g_lock_get_shared(lck, i, &shared);
     300             : 
     301          13 :                 same = server_id_equal(self, &shared);
     302          13 :                 if (same) {
     303           4 :                         return i;
     304             :                 }
     305             :         }
     306             : 
     307         291 :         return -1;
     308             : }
     309             : 
     310         321 : static void g_lock_cleanup_shared(struct g_lock *lck)
     311             : {
     312          28 :         size_t i;
     313          28 :         struct server_id check;
     314          28 :         bool exists;
     315             : 
     316         321 :         if (lck->num_shared == 0) {
     317         302 :                 return;
     318             :         }
     319             : 
     320             :         /*
     321             :          * Read locks can stay around forever if the process dies. Do
     322             :          * a heuristic check for process existence: Check one random
     323             :          * process for existence. Hopefully this will keep runaway
     324             :          * read locks under control.
     325             :          */
     326          19 :         i = generate_random() % lck->num_shared;
     327          19 :         g_lock_get_shared(lck, i, &check);
     328             : 
     329          19 :         exists = serverid_exists(&check);
     330          19 :         if (!exists) {
     331           6 :                 struct server_id_buf tmp;
     332           6 :                 DBG_DEBUG("Shared locker %s died -- removing\n",
     333             :                           server_id_str_buf(check, &tmp));
     334           6 :                 g_lock_del_shared(lck, i);
     335             :         }
     336             : }
     337             : 
     338             : struct g_lock_lock_cb_state {
     339             :         struct g_lock_ctx *ctx;
     340             :         struct db_record *rec;
     341             :         struct g_lock *lck;
     342             :         struct server_id *new_shared;
     343             :         g_lock_lock_cb_fn_t cb_fn;
     344             :         void *cb_private;
     345             :         TALLOC_CTX *update_mem_ctx;
     346             :         TDB_DATA updated_data;
     347             :         bool existed;
     348             :         bool modified;
     349             :         bool unlock;
     350             : };
     351             : 
     352     3241131 : NTSTATUS g_lock_lock_cb_dump(struct g_lock_lock_cb_state *cb_state,
     353             :                              void (*fn)(struct server_id exclusive,
     354             :                                         size_t num_shared,
     355             :                                         const struct server_id *shared,
     356             :                                         const uint8_t *data,
     357             :                                         size_t datalen,
     358             :                                         void *private_data),
     359             :                              void *private_data)
     360             : {
     361     3241131 :         struct g_lock *lck = cb_state->lck;
     362             : 
     363             :         /* We allow a cn_fn only for G_LOCK_WRITE for now... */
     364     3241131 :         SMB_ASSERT(lck->num_shared == 0);
     365             : 
     366     3241131 :         fn(lck->exclusive,
     367             :            0, /* num_shared */
     368             :            NULL, /* shared */
     369     3241131 :            lck->data,
     370             :            lck->datalen,
     371             :            private_data);
     372             : 
     373     3241131 :         return NT_STATUS_OK;
     374             : }
     375             : 
     376      750970 : NTSTATUS g_lock_lock_cb_writev(struct g_lock_lock_cb_state *cb_state,
     377             :                                const TDB_DATA *dbufs,
     378             :                                size_t num_dbufs)
     379             : {
     380        1915 :         NTSTATUS status;
     381             : 
     382      750970 :         status = dbwrap_merge_dbufs(&cb_state->updated_data,
     383             :                                     cb_state->update_mem_ctx,
     384             :                                     dbufs, num_dbufs);
     385      750970 :         if (!NT_STATUS_IS_OK(status)) {
     386           0 :                 return status;
     387             :         }
     388             : 
     389      750970 :         cb_state->modified = true;
     390      750970 :         cb_state->lck->data = cb_state->updated_data.dptr;
     391      750970 :         cb_state->lck->datalen = cb_state->updated_data.dsize;
     392             : 
     393      750970 :         return NT_STATUS_OK;
     394             : }
     395             : 
     396      452420 : void g_lock_lock_cb_unlock(struct g_lock_lock_cb_state *cb_state)
     397             : {
     398      452420 :         cb_state->unlock = true;
     399      452420 : }
     400             : 
     401             : struct g_lock_lock_cb_watch_data_state {
     402             :         struct tevent_context *ev;
     403             :         struct g_lock_ctx *ctx;
     404             :         TDB_DATA key;
     405             :         struct server_id blocker;
     406             :         bool blockerdead;
     407             :         uint64_t unique_lock_epoch;
     408             :         uint64_t unique_data_epoch;
     409             :         uint64_t watch_instance;
     410             :         NTSTATUS status;
     411             : };
     412             : 
     413             : static void g_lock_lock_cb_watch_data_done(struct tevent_req *subreq);
     414             : 
     415         499 : struct tevent_req *g_lock_lock_cb_watch_data_send(
     416             :         TALLOC_CTX *mem_ctx,
     417             :         struct tevent_context *ev,
     418             :         struct g_lock_lock_cb_state *cb_state,
     419             :         struct server_id blocker)
     420             : {
     421         499 :         struct tevent_req *req = NULL;
     422         499 :         struct g_lock_lock_cb_watch_data_state *state = NULL;
     423         499 :         struct tevent_req *subreq = NULL;
     424         499 :         TDB_DATA key = dbwrap_record_get_key(cb_state->rec);
     425             : 
     426         499 :         req = tevent_req_create(
     427             :                 mem_ctx, &state, struct g_lock_lock_cb_watch_data_state);
     428         499 :         if (req == NULL) {
     429           0 :                 return NULL;
     430             :         }
     431         499 :         state->ev = ev;
     432         499 :         state->ctx = cb_state->ctx;
     433         499 :         state->blocker = blocker;
     434             : 
     435         499 :         state->key = tdb_data_talloc_copy(state, key);
     436         499 :         if (tevent_req_nomem(state->key.dptr, req)) {
     437           0 :                 return tevent_req_post(req, ev);
     438             :         }
     439             : 
     440         499 :         state->unique_lock_epoch = cb_state->lck->unique_lock_epoch;
     441         499 :         state->unique_data_epoch = cb_state->lck->unique_data_epoch;
     442             : 
     443         499 :         DBG_DEBUG("state->unique_data_epoch=%"PRIu64"\n", state->unique_data_epoch);
     444             : 
     445         499 :         subreq = dbwrap_watched_watch_send(
     446         499 :                 state, state->ev, cb_state->rec, 0, state->blocker);
     447         499 :         if (tevent_req_nomem(subreq, req)) {
     448           0 :                 return tevent_req_post(req, ev);
     449             :         }
     450         499 :         tevent_req_set_callback(subreq, g_lock_lock_cb_watch_data_done, req);
     451             : 
     452         499 :         return req;
     453             : }
     454             : 
     455         712 : static void g_lock_lock_cb_watch_data_done_fn(
     456             :         struct db_record *rec,
     457             :         TDB_DATA value,
     458             :         void *private_data)
     459             : {
     460         712 :         struct tevent_req *req = talloc_get_type_abort(
     461             :                 private_data, struct tevent_req);
     462         712 :         struct g_lock_lock_cb_watch_data_state *state = tevent_req_data(
     463             :                 req, struct g_lock_lock_cb_watch_data_state);
     464         712 :         struct tevent_req *subreq = NULL;
     465           0 :         struct g_lock lck;
     466           0 :         bool ok;
     467             : 
     468         712 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
     469         712 :         if (!ok) {
     470           0 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     471           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
     472           0 :                 return;
     473             :         }
     474             : 
     475         712 :         if (lck.unique_data_epoch != state->unique_data_epoch) {
     476         447 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     477         447 :                 DBG_DEBUG("lck.unique_data_epoch=%"PRIu64", "
     478             :                           "state->unique_data_epoch=%"PRIu64"\n",
     479             :                           lck.unique_data_epoch,
     480             :                           state->unique_data_epoch);
     481         447 :                 state->status = NT_STATUS_OK;
     482         447 :                 return;
     483             :         }
     484             : 
     485             :         /*
     486             :          * The lock epoch changed, so we better
     487             :          * remove ourself from the waiter list
     488             :          * (most likely the first position)
     489             :          * and re-add us at the end of the list.
     490             :          *
     491             :          * This gives other lock waiters a change
     492             :          * to make progress.
     493             :          *
     494             :          * Otherwise we'll keep our waiter instance alive,
     495             :          * keep waiting (most likely at first position).
     496             :          */
     497         265 :         if (lck.unique_lock_epoch != state->unique_lock_epoch) {
     498         241 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     499         241 :                 state->watch_instance = dbwrap_watched_watch_add_instance(rec);
     500         241 :                 state->unique_lock_epoch = lck.unique_lock_epoch;
     501             :         }
     502             : 
     503         265 :         subreq = dbwrap_watched_watch_send(
     504             :                 state, state->ev, rec, state->watch_instance, state->blocker);
     505         265 :         if (subreq == NULL) {
     506           0 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     507           0 :                 state->status = NT_STATUS_NO_MEMORY;
     508           0 :                 return;
     509             :         }
     510         265 :         tevent_req_set_callback(subreq, g_lock_lock_cb_watch_data_done, req);
     511             : 
     512         265 :         state->status = NT_STATUS_EVENT_PENDING;
     513             : }
     514             : 
     515         712 : static void g_lock_lock_cb_watch_data_done(struct tevent_req *subreq)
     516             : {
     517         712 :         struct tevent_req *req = tevent_req_callback_data(
     518             :                 subreq, struct tevent_req);
     519         712 :         struct g_lock_lock_cb_watch_data_state *state = tevent_req_data(
     520             :                 req, struct g_lock_lock_cb_watch_data_state);
     521           0 :         NTSTATUS status;
     522         712 :         uint64_t instance = 0;
     523             : 
     524         712 :         status = dbwrap_watched_watch_recv(
     525             :                 subreq, &instance, &state->blockerdead, &state->blocker);
     526         712 :         TALLOC_FREE(subreq);
     527         712 :         if (tevent_req_nterror(req, status)) {
     528           0 :                 DBG_DEBUG("dbwrap_watched_watch_recv returned %s\n",
     529             :                           nt_errstr(status));
     530         265 :                 return;
     531             :         }
     532             : 
     533         712 :         state->watch_instance = instance;
     534             : 
     535         712 :         status = dbwrap_do_locked(
     536         712 :                 state->ctx->db, state->key, g_lock_lock_cb_watch_data_done_fn, req);
     537         712 :         if (tevent_req_nterror(req, status)) {
     538           0 :                 DBG_DEBUG("dbwrap_do_locked returned %s\n", nt_errstr(status));
     539           0 :                 return;
     540             :         }
     541         712 :         if (NT_STATUS_EQUAL(state->status, NT_STATUS_EVENT_PENDING)) {
     542         265 :                 return;
     543             :         }
     544         447 :         if (tevent_req_nterror(req, state->status)) {
     545           0 :                 return;
     546             :         }
     547         447 :         tevent_req_done(req);
     548             : }
     549             : 
     550         447 : NTSTATUS g_lock_lock_cb_watch_data_recv(
     551             :         struct tevent_req *req,
     552             :         bool *blockerdead,
     553             :         struct server_id *blocker)
     554             : {
     555         447 :         struct g_lock_lock_cb_watch_data_state *state = tevent_req_data(
     556             :                 req, struct g_lock_lock_cb_watch_data_state);
     557           0 :         NTSTATUS status;
     558             : 
     559         447 :         if (tevent_req_is_nterror(req, &status)) {
     560           0 :                 return status;
     561             :         }
     562         447 :         if (blockerdead != NULL) {
     563         447 :                 *blockerdead = state->blockerdead;
     564             :         }
     565         447 :         if (blocker != NULL) {
     566         447 :                 *blocker = state->blocker;
     567             :         }
     568             : 
     569         447 :         return NT_STATUS_OK;
     570             : }
     571             : 
     572        2542 : void g_lock_lock_cb_wake_watchers(struct g_lock_lock_cb_state *cb_state)
     573             : {
     574        2542 :         struct g_lock *lck = cb_state->lck;
     575             : 
     576        2542 :         lck->unique_data_epoch = generate_unique_u64(lck->unique_data_epoch);
     577        2542 :         cb_state->modified = true;
     578        2542 : }
     579             : 
     580      943529 : static NTSTATUS g_lock_lock_cb_run_and_store(struct g_lock_lock_cb_state *cb_state)
     581             : {
     582      943529 :         struct g_lock *lck = cb_state->lck;
     583      943529 :         NTSTATUS success_status = NT_STATUS_OK;
     584        2314 :         NTSTATUS status;
     585             : 
     586      943529 :         if (cb_state->cb_fn != NULL) {
     587             : 
     588      931673 :                 SMB_ASSERT(lck->num_shared == 0);
     589      931673 :                 SMB_ASSERT(cb_state->new_shared == NULL);
     590             : 
     591      931673 :                 if (cb_state->ctx->lock_order != DBWRAP_LOCK_ORDER_NONE) {
     592      931673 :                         const char *name = dbwrap_name(cb_state->ctx->db);
     593      931673 :                         dbwrap_lock_order_lock(name, cb_state->ctx->lock_order);
     594             :                 }
     595             : 
     596      931673 :                 cb_state->ctx->busy = true;
     597      931673 :                 cb_state->cb_fn(cb_state, cb_state->cb_private);
     598      931673 :                 cb_state->ctx->busy = false;
     599             : 
     600      931673 :                 if (cb_state->ctx->lock_order != DBWRAP_LOCK_ORDER_NONE) {
     601      931673 :                         const char *name = dbwrap_name(cb_state->ctx->db);
     602      931673 :                         dbwrap_lock_order_unlock(name, cb_state->ctx->lock_order);
     603             :                 }
     604             :         }
     605             : 
     606      943529 :         if (cb_state->unlock) {
     607             :                 /*
     608             :                  * Unlocked should wake up watchers.
     609             :                  *
     610             :                  * We no longer need the lock, so
     611             :                  * force a wakeup of the next watchers,
     612             :                  * even if we don't do any update.
     613             :                  */
     614      452420 :                 dbwrap_watched_watch_reset_alerting(cb_state->rec);
     615      452420 :                 dbwrap_watched_watch_force_alerting(cb_state->rec);
     616      452420 :                 if (!cb_state->modified) {
     617             :                         /*
     618             :                          * The record was not changed at
     619             :                          * all, so we can also avoid
     620             :                          * storing the lck.unique_lock_epoch
     621             :                          * change
     622             :                          */
     623       12924 :                         return NT_STATUS_WAS_UNLOCKED;
     624             :                 }
     625      439496 :                 lck->exclusive = (struct server_id) { .pid = 0 };
     626      439496 :                 cb_state->new_shared = NULL;
     627             : 
     628      439496 :                 if (lck->datalen == 0) {
     629      242299 :                         if (!cb_state->existed) {
     630           0 :                                 return NT_STATUS_WAS_UNLOCKED;
     631             :                         }
     632             : 
     633      242299 :                         status = dbwrap_record_delete(cb_state->rec);
     634      242299 :                         if (!NT_STATUS_IS_OK(status)) {
     635           0 :                                 DBG_WARNING("dbwrap_record_delete() failed: %s\n",
     636             :                                     nt_errstr(status));
     637           0 :                                 return status;
     638             :                         }
     639      242299 :                         return NT_STATUS_WAS_UNLOCKED;
     640             :                 }
     641             : 
     642      196393 :                 success_status = NT_STATUS_WAS_UNLOCKED;
     643             :         }
     644             : 
     645      688306 :         status = g_lock_store(cb_state->rec,
     646             :                               cb_state->lck,
     647             :                               cb_state->new_shared,
     648             :                               NULL, 0);
     649      688306 :         if (!NT_STATUS_IS_OK(status)) {
     650           0 :                 DBG_WARNING("g_lock_store() failed: %s\n",
     651             :                             nt_errstr(status));
     652           0 :                 return status;
     653             :         }
     654             : 
     655      688306 :         return success_status;
     656             : }
     657             : 
     658             : struct g_lock_lock_state {
     659             :         struct tevent_context *ev;
     660             :         struct g_lock_ctx *ctx;
     661             :         TDB_DATA key;
     662             :         enum g_lock_type type;
     663             :         bool retry;
     664             :         g_lock_lock_cb_fn_t cb_fn;
     665             :         void *cb_private;
     666             : };
     667             : 
     668             : struct g_lock_lock_fn_state {
     669             :         struct g_lock_lock_state *req_state;
     670             :         struct server_id *dead_blocker;
     671             : 
     672             :         struct tevent_req *watch_req;
     673             :         uint64_t watch_instance;
     674             :         NTSTATUS status;
     675             : };
     676             : 
     677             : static int g_lock_lock_state_destructor(struct g_lock_lock_state *s);
     678             : 
     679         321 : static NTSTATUS g_lock_trylock(
     680             :         struct db_record *rec,
     681             :         struct g_lock_lock_fn_state *state,
     682             :         TDB_DATA data,
     683             :         struct server_id *blocker)
     684             : {
     685         321 :         struct g_lock_lock_state *req_state = state->req_state;
     686         321 :         struct server_id self = messaging_server_id(req_state->ctx->msg);
     687         321 :         enum g_lock_type type = req_state->type;
     688         321 :         bool retry = req_state->retry;
     689         321 :         struct g_lock lck = { .exclusive.pid = 0 };
     690         642 :         struct g_lock_lock_cb_state cb_state = {
     691         321 :                 .ctx = req_state->ctx,
     692             :                 .rec = rec,
     693             :                 .lck = &lck,
     694         321 :                 .cb_fn = req_state->cb_fn,
     695         321 :                 .cb_private = req_state->cb_private,
     696         321 :                 .existed = data.dsize != 0,
     697         321 :                 .update_mem_ctx = talloc_tos(),
     698             :         };
     699          15 :         struct server_id_buf tmp;
     700          15 :         NTSTATUS status;
     701          15 :         bool ok;
     702             : 
     703         321 :         ok = g_lock_parse(data.dptr, data.dsize, &lck);
     704         321 :         if (!ok) {
     705           0 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     706           0 :                 DBG_DEBUG("g_lock_parse failed\n");
     707           0 :                 return NT_STATUS_INTERNAL_DB_CORRUPTION;
     708             :         }
     709             : 
     710         321 :         g_lock_cleanup_dead(&lck, state->dead_blocker);
     711             : 
     712         321 :         lck.unique_lock_epoch = generate_unique_u64(lck.unique_lock_epoch);
     713             : 
     714         321 :         if (lck.exclusive.pid != 0) {
     715          21 :                 bool self_exclusive = server_id_equal(&self, &lck.exclusive);
     716             : 
     717          21 :                 if (!self_exclusive) {
     718          17 :                         bool exists = serverid_exists(&lck.exclusive);
     719          17 :                         if (!exists) {
     720           0 :                                 lck.exclusive = (struct server_id) { .pid=0 };
     721           0 :                                 goto noexclusive;
     722             :                         }
     723             : 
     724          17 :                         DBG_DEBUG("%s has an exclusive lock\n",
     725             :                                   server_id_str_buf(lck.exclusive, &tmp));
     726             : 
     727          17 :                         if (type == G_LOCK_DOWNGRADE) {
     728           0 :                                 struct server_id_buf tmp2;
     729             : 
     730           0 :                                 dbwrap_watched_watch_remove_instance(rec,
     731             :                                                 state->watch_instance);
     732             : 
     733           0 :                                 DBG_DEBUG("%s: Trying to downgrade %s\n",
     734             :                                           server_id_str_buf(self, &tmp),
     735             :                                           server_id_str_buf(
     736             :                                                   lck.exclusive, &tmp2));
     737           0 :                                 return NT_STATUS_NOT_LOCKED;
     738             :                         }
     739             : 
     740          17 :                         if (type == G_LOCK_UPGRADE) {
     741           1 :                                 ssize_t shared_idx;
     742             : 
     743           1 :                                 dbwrap_watched_watch_remove_instance(rec,
     744             :                                                 state->watch_instance);
     745             : 
     746           1 :                                 shared_idx = g_lock_find_shared(&lck, &self);
     747             : 
     748           1 :                                 if (shared_idx == -1) {
     749           0 :                                         DBG_DEBUG("Trying to upgrade %s "
     750             :                                                   "without "
     751             :                                                   "existing shared lock\n",
     752             :                                                   server_id_str_buf(
     753             :                                                           self, &tmp));
     754           0 :                                         return NT_STATUS_NOT_LOCKED;
     755             :                                 }
     756             : 
     757             :                                 /*
     758             :                                  * We're trying to upgrade, and the
     759             :                                  * exclusive lock is taken by someone
     760             :                                  * else. This means that someone else
     761             :                                  * is waiting for us to give up our
     762             :                                  * shared lock. If we now also wait
     763             :                                  * for someone to give their shared
     764             :                                  * lock, we will deadlock.
     765             :                                  */
     766             : 
     767           1 :                                 DBG_DEBUG("Trying to upgrade %s while "
     768             :                                           "someone else is also "
     769             :                                           "trying to upgrade\n",
     770             :                                           server_id_str_buf(self, &tmp));
     771           1 :                                 return NT_STATUS_POSSIBLE_DEADLOCK;
     772             :                         }
     773             : 
     774          16 :                         DBG_DEBUG("Waiting for lck.exclusive=%s\n",
     775             :                                   server_id_str_buf(lck.exclusive, &tmp));
     776             : 
     777             :                         /*
     778             :                          * We will return NT_STATUS_LOCK_NOT_GRANTED
     779             :                          * and need to monitor the record.
     780             :                          *
     781             :                          * If we don't have a watcher instance yet,
     782             :                          * we should add one.
     783             :                          */
     784          16 :                         if (state->watch_instance == 0) {
     785          16 :                                 state->watch_instance =
     786          16 :                                         dbwrap_watched_watch_add_instance(rec);
     787             :                         }
     788             : 
     789          16 :                         *blocker = lck.exclusive;
     790          16 :                         return NT_STATUS_LOCK_NOT_GRANTED;
     791             :                 }
     792             : 
     793           4 :                 if (type == G_LOCK_DOWNGRADE) {
     794           0 :                         DBG_DEBUG("Downgrading %s from WRITE to READ\n",
     795             :                                   server_id_str_buf(self, &tmp));
     796             : 
     797           0 :                         lck.exclusive = (struct server_id) { .pid = 0 };
     798           0 :                         goto do_shared;
     799             :                 }
     800             : 
     801           4 :                 if (!retry) {
     802           1 :                         dbwrap_watched_watch_remove_instance(rec,
     803             :                                                 state->watch_instance);
     804             : 
     805           1 :                         DBG_DEBUG("%s already locked by self\n",
     806             :                                   server_id_str_buf(self, &tmp));
     807           1 :                         return NT_STATUS_WAS_LOCKED;
     808             :                 }
     809             : 
     810           3 :                 g_lock_cleanup_shared(&lck);
     811             : 
     812           3 :                 if (lck.num_shared != 0) {
     813           1 :                         g_lock_get_shared(&lck, 0, blocker);
     814             : 
     815           1 :                         DBG_DEBUG("Continue waiting for shared lock %s\n",
     816             :                                   server_id_str_buf(*blocker, &tmp));
     817             : 
     818             :                         /*
     819             :                          * We will return NT_STATUS_LOCK_NOT_GRANTED
     820             :                          * and need to monitor the record.
     821             :                          *
     822             :                          * If we don't have a watcher instance yet,
     823             :                          * we should add one.
     824             :                          */
     825           1 :                         if (state->watch_instance == 0) {
     826           0 :                                 state->watch_instance =
     827           0 :                                         dbwrap_watched_watch_add_instance(rec);
     828             :                         }
     829             : 
     830           1 :                         return NT_STATUS_LOCK_NOT_GRANTED;
     831             :                 }
     832             : 
     833             :                 /*
     834             :                  * Retry after a conflicting lock was released..
     835             :                  * All pending readers are gone so we got the lock...
     836             :                  */
     837           2 :                 goto got_lock;
     838             :         }
     839             : 
     840         300 : noexclusive:
     841             : 
     842         300 :         if (type == G_LOCK_UPGRADE) {
     843           3 :                 ssize_t shared_idx = g_lock_find_shared(&lck, &self);
     844             : 
     845           3 :                 if (shared_idx == -1) {
     846           0 :                         dbwrap_watched_watch_remove_instance(rec,
     847             :                                                 state->watch_instance);
     848             : 
     849           0 :                         DBG_DEBUG("Trying to upgrade %s without "
     850             :                                   "existing shared lock\n",
     851             :                                   server_id_str_buf(self, &tmp));
     852           0 :                         return NT_STATUS_NOT_LOCKED;
     853             :                 }
     854             : 
     855           3 :                 g_lock_del_shared(&lck, shared_idx);
     856           3 :                 type = G_LOCK_WRITE;
     857             :         }
     858             : 
     859         300 :         if (type == G_LOCK_WRITE) {
     860         298 :                 ssize_t shared_idx = g_lock_find_shared(&lck, &self);
     861             : 
     862         298 :                 if (shared_idx != -1) {
     863           0 :                         dbwrap_watched_watch_remove_instance(rec,
     864             :                                                 state->watch_instance);
     865           0 :                         DBG_DEBUG("Trying to writelock existing shared %s\n",
     866             :                                   server_id_str_buf(self, &tmp));
     867           0 :                         return NT_STATUS_WAS_LOCKED;
     868             :                 }
     869             : 
     870         298 :                 lck.exclusive = self;
     871             : 
     872         298 :                 g_lock_cleanup_shared(&lck);
     873             : 
     874         298 :                 if (lck.num_shared == 0) {
     875             :                         /*
     876             :                          * If we store ourself as exclusive writer,
     877             :                          * without any pending readers ...
     878             :                          */
     879         293 :                         goto got_lock;
     880             :                 }
     881             : 
     882           5 :                 if (state->watch_instance == 0) {
     883             :                         /*
     884             :                          * Here we have lck.num_shared != 0.
     885             :                          *
     886             :                          * We will return NT_STATUS_LOCK_NOT_GRANTED
     887             :                          * below.
     888             :                          *
     889             :                          * And don't have a watcher instance yet!
     890             :                          *
     891             :                          * We add it here before g_lock_store()
     892             :                          * in order to trigger just one
     893             :                          * low level dbwrap_do_locked() call.
     894             :                          */
     895           5 :                         state->watch_instance =
     896           5 :                                 dbwrap_watched_watch_add_instance(rec);
     897             :                 }
     898             : 
     899           5 :                 status = g_lock_store(rec, &lck, NULL, NULL, 0);
     900           5 :                 if (!NT_STATUS_IS_OK(status)) {
     901           0 :                         DBG_DEBUG("g_lock_store() failed: %s\n",
     902             :                                   nt_errstr(status));
     903           0 :                         return status;
     904             :                 }
     905             : 
     906           5 :                 talloc_set_destructor(
     907             :                         req_state, g_lock_lock_state_destructor);
     908             : 
     909           5 :                 g_lock_get_shared(&lck, 0, blocker);
     910             : 
     911           5 :                 DBG_DEBUG("Waiting for %zu shared locks, "
     912             :                           "picking blocker %s\n",
     913             :                           lck.num_shared,
     914             :                           server_id_str_buf(*blocker, &tmp));
     915             : 
     916           5 :                 return NT_STATUS_LOCK_NOT_GRANTED;
     917             :         }
     918             : 
     919           2 : do_shared:
     920             : 
     921           2 :         g_lock_cleanup_shared(&lck);
     922           2 :         cb_state.new_shared = &self;
     923           2 :         goto got_lock;
     924             : 
     925         297 : got_lock:
     926             :         /*
     927             :          * We got the lock we asked for, so we no
     928             :          * longer need to monitor the record.
     929             :          */
     930         297 :         dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
     931             : 
     932         297 :         status = g_lock_lock_cb_run_and_store(&cb_state);
     933         297 :         if (!NT_STATUS_IS_OK(status) &&
     934           6 :             !NT_STATUS_EQUAL(status, NT_STATUS_WAS_UNLOCKED))
     935             :         {
     936           0 :                 DBG_WARNING("g_lock_lock_cb_run_and_store() failed: %s\n",
     937             :                             nt_errstr(status));
     938           0 :                 return status;
     939             :         }
     940             : 
     941         297 :         talloc_set_destructor(req_state, NULL);
     942         297 :         return status;
     943             : }
     944             : 
     945         321 : static void g_lock_lock_fn(
     946             :         struct db_record *rec,
     947             :         TDB_DATA value,
     948             :         void *private_data)
     949             : {
     950         321 :         struct g_lock_lock_fn_state *state = private_data;
     951         321 :         struct server_id blocker = {0};
     952             : 
     953             :         /*
     954             :          * We're trying to get a lock and if we are
     955             :          * successful in doing that, we should not
     956             :          * wakeup any other waiters, all they would
     957             :          * find is that we're holding a lock they
     958             :          * are conflicting with.
     959             :          */
     960         321 :         dbwrap_watched_watch_skip_alerting(rec);
     961             : 
     962         321 :         state->status = g_lock_trylock(rec, state, value, &blocker);
     963         321 :         if (!NT_STATUS_IS_OK(state->status)) {
     964          30 :                 DBG_DEBUG("g_lock_trylock returned %s\n",
     965             :                           nt_errstr(state->status));
     966             :         }
     967         321 :         if (!NT_STATUS_EQUAL(state->status, NT_STATUS_LOCK_NOT_GRANTED)) {
     968         299 :                 return;
     969             :         }
     970             : 
     971          44 :         state->watch_req = dbwrap_watched_watch_send(
     972          22 :                 state->req_state, state->req_state->ev, rec, state->watch_instance, blocker);
     973          22 :         if (state->watch_req == NULL) {
     974           0 :                 state->status = NT_STATUS_NO_MEMORY;
     975             :         }
     976             : }
     977             : 
     978           2 : static int g_lock_lock_state_destructor(struct g_lock_lock_state *s)
     979             : {
     980           2 :         NTSTATUS status = g_lock_unlock(s->ctx, s->key);
     981           2 :         if (!NT_STATUS_IS_OK(status)) {
     982           0 :                 DBG_DEBUG("g_lock_unlock failed: %s\n", nt_errstr(status));
     983             :         }
     984           2 :         return 0;
     985             : }
     986             : 
     987             : static void g_lock_lock_retry(struct tevent_req *subreq);
     988             : 
     989         304 : struct tevent_req *g_lock_lock_send(TALLOC_CTX *mem_ctx,
     990             :                                     struct tevent_context *ev,
     991             :                                     struct g_lock_ctx *ctx,
     992             :                                     TDB_DATA key,
     993             :                                     enum g_lock_type type,
     994             :                                     g_lock_lock_cb_fn_t cb_fn,
     995             :                                     void *cb_private)
     996             : {
     997          11 :         struct tevent_req *req;
     998          11 :         struct g_lock_lock_state *state;
     999          11 :         struct g_lock_lock_fn_state fn_state;
    1000          11 :         NTSTATUS status;
    1001          11 :         bool ok;
    1002             : 
    1003         304 :         SMB_ASSERT(!ctx->busy);
    1004             : 
    1005         304 :         req = tevent_req_create(mem_ctx, &state, struct g_lock_lock_state);
    1006         304 :         if (req == NULL) {
    1007           0 :                 return NULL;
    1008             :         }
    1009         304 :         state->ev = ev;
    1010         304 :         state->ctx = ctx;
    1011         304 :         state->key = key;
    1012         304 :         state->type = type;
    1013         304 :         state->cb_fn = cb_fn;
    1014         304 :         state->cb_private = cb_private;
    1015             : 
    1016         304 :         fn_state = (struct g_lock_lock_fn_state) {
    1017             :                 .req_state = state,
    1018             :         };
    1019             : 
    1020             :         /*
    1021             :          * We allow a cn_fn only for G_LOCK_WRITE for now.
    1022             :          *
    1023             :          * It's all we currently need and it makes a few things
    1024             :          * easier to implement.
    1025             :          */
    1026         304 :         if (unlikely(cb_fn != NULL && type != G_LOCK_WRITE)) {
    1027           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_PARAMETER_6);
    1028           0 :                 return tevent_req_post(req, ev);
    1029             :         }
    1030             : 
    1031         304 :         status = dbwrap_do_locked(ctx->db, key, g_lock_lock_fn, &fn_state);
    1032         304 :         if (tevent_req_nterror(req, status)) {
    1033           0 :                 DBG_DEBUG("dbwrap_do_locked failed: %s\n",
    1034             :                           nt_errstr(status));
    1035           0 :                 return tevent_req_post(req, ev);
    1036             :         }
    1037             : 
    1038         304 :         if (NT_STATUS_IS_OK(fn_state.status)) {
    1039         281 :                 tevent_req_done(req);
    1040         281 :                 return tevent_req_post(req, ev);
    1041             :         }
    1042          23 :         if (!NT_STATUS_EQUAL(fn_state.status, NT_STATUS_LOCK_NOT_GRANTED)) {
    1043           2 :                 tevent_req_nterror(req, fn_state.status);
    1044           2 :                 return tevent_req_post(req, ev);
    1045             :         }
    1046             : 
    1047          21 :         if (tevent_req_nomem(fn_state.watch_req, req)) {
    1048           0 :                 return tevent_req_post(req, ev);
    1049             :         }
    1050             : 
    1051          58 :         ok = tevent_req_set_endtime(
    1052             :                 fn_state.watch_req,
    1053          21 :                 state->ev,
    1054          21 :                 timeval_current_ofs(5 + generate_random() % 5, 0));
    1055          21 :         if (!ok) {
    1056           0 :                 tevent_req_oom(req);
    1057           0 :                 return tevent_req_post(req, ev);
    1058             :         }
    1059          21 :         tevent_req_set_callback(fn_state.watch_req, g_lock_lock_retry, req);
    1060             : 
    1061          21 :         return req;
    1062             : }
    1063             : 
    1064          17 : static void g_lock_lock_retry(struct tevent_req *subreq)
    1065             : {
    1066          17 :         struct tevent_req *req = tevent_req_callback_data(
    1067             :                 subreq, struct tevent_req);
    1068          17 :         struct g_lock_lock_state *state = tevent_req_data(
    1069             :                 req, struct g_lock_lock_state);
    1070           4 :         struct g_lock_lock_fn_state fn_state;
    1071          17 :         struct server_id blocker = { .pid = 0 };
    1072          17 :         bool blockerdead = false;
    1073           4 :         NTSTATUS status;
    1074          17 :         uint64_t instance = 0;
    1075             : 
    1076          17 :         status = dbwrap_watched_watch_recv(subreq, &instance, &blockerdead, &blocker);
    1077          17 :         DBG_DEBUG("watch_recv returned %s\n", nt_errstr(status));
    1078          17 :         TALLOC_FREE(subreq);
    1079             : 
    1080          17 :         if (!NT_STATUS_IS_OK(status) &&
    1081           0 :             !NT_STATUS_EQUAL(status, NT_STATUS_IO_TIMEOUT)) {
    1082           0 :                 tevent_req_nterror(req, status);
    1083           0 :                 return;
    1084             :         }
    1085             : 
    1086          17 :         state->retry = true;
    1087             : 
    1088          21 :         fn_state = (struct g_lock_lock_fn_state) {
    1089             :                 .req_state = state,
    1090          17 :                 .dead_blocker = blockerdead ? &blocker : NULL,
    1091             :                 .watch_instance = instance,
    1092             :         };
    1093             : 
    1094          17 :         status = dbwrap_do_locked(state->ctx->db, state->key,
    1095             :                                   g_lock_lock_fn, &fn_state);
    1096          17 :         if (tevent_req_nterror(req, status)) {
    1097           0 :                 DBG_DEBUG("dbwrap_do_locked failed: %s\n",
    1098             :                           nt_errstr(status));
    1099           0 :                 return;
    1100             :         }
    1101             : 
    1102          17 :         if (NT_STATUS_IS_OK(fn_state.status)) {
    1103          10 :                 tevent_req_done(req);
    1104          10 :                 return;
    1105             :         }
    1106           7 :         if (!NT_STATUS_EQUAL(fn_state.status, NT_STATUS_LOCK_NOT_GRANTED)) {
    1107           6 :                 tevent_req_nterror(req, fn_state.status);
    1108           6 :                 return;
    1109             :         }
    1110             : 
    1111           1 :         if (tevent_req_nomem(fn_state.watch_req, req)) {
    1112           0 :                 return;
    1113             :         }
    1114             : 
    1115           1 :         if (!tevent_req_set_endtime(
    1116             :                     fn_state.watch_req, state->ev,
    1117           1 :                     timeval_current_ofs(5 + generate_random() % 5, 0))) {
    1118           0 :                 return;
    1119             :         }
    1120           1 :         tevent_req_set_callback(fn_state.watch_req, g_lock_lock_retry, req);
    1121             : }
    1122             : 
    1123         303 : NTSTATUS g_lock_lock_recv(struct tevent_req *req)
    1124             : {
    1125         303 :         struct g_lock_lock_state *state = tevent_req_data(
    1126             :                 req, struct g_lock_lock_state);
    1127         303 :         struct g_lock_ctx *ctx = state->ctx;
    1128          10 :         NTSTATUS status;
    1129             : 
    1130         303 :         if (tevent_req_is_nterror(req, &status)) {
    1131          12 :                 if (NT_STATUS_EQUAL(status, NT_STATUS_WAS_UNLOCKED)) {
    1132           6 :                         return NT_STATUS_OK;
    1133             :                 }
    1134           6 :                 return status;
    1135             :         }
    1136             : 
    1137         291 :         if ((ctx->lock_order != DBWRAP_LOCK_ORDER_NONE) &&
    1138           8 :             ((state->type == G_LOCK_READ) ||
    1139           8 :              (state->type == G_LOCK_WRITE))) {
    1140           8 :                 const char *name = dbwrap_name(ctx->db);
    1141           8 :                 dbwrap_lock_order_lock(name, ctx->lock_order);
    1142             :         }
    1143             : 
    1144         291 :         return NT_STATUS_OK;
    1145             : }
    1146             : 
    1147             : struct g_lock_lock_simple_state {
    1148             :         struct g_lock_ctx *ctx;
    1149             :         struct server_id me;
    1150             :         enum g_lock_type type;
    1151             :         NTSTATUS status;
    1152             :         g_lock_lock_cb_fn_t cb_fn;
    1153             :         void *cb_private;
    1154             : };
    1155             : 
    1156      943251 : static void g_lock_lock_simple_fn(
    1157             :         struct db_record *rec,
    1158             :         TDB_DATA value,
    1159             :         void *private_data)
    1160             : {
    1161      943251 :         struct g_lock_lock_simple_state *state = private_data;
    1162        2315 :         struct server_id_buf buf;
    1163      943251 :         struct g_lock lck = { .exclusive.pid = 0 };
    1164     1886502 :         struct g_lock_lock_cb_state cb_state = {
    1165      943251 :                 .ctx = state->ctx,
    1166             :                 .rec = rec,
    1167             :                 .lck = &lck,
    1168      943251 :                 .cb_fn = state->cb_fn,
    1169      943251 :                 .cb_private = state->cb_private,
    1170      943251 :                 .existed = value.dsize != 0,
    1171      943251 :                 .update_mem_ctx = talloc_tos(),
    1172             :         };
    1173        2315 :         bool ok;
    1174             : 
    1175      943251 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1176      943251 :         if (!ok) {
    1177           0 :                 DBG_DEBUG("g_lock_parse failed\n");
    1178           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1179           0 :                 return;
    1180             :         }
    1181             : 
    1182      943251 :         if (lck.exclusive.pid != 0) {
    1183          17 :                 DBG_DEBUG("locked by %s\n",
    1184             :                           server_id_str_buf(lck.exclusive, &buf));
    1185          17 :                 goto not_granted;
    1186             :         }
    1187             : 
    1188      943234 :         if (state->type == G_LOCK_WRITE) {
    1189      943216 :                 if (lck.num_shared != 0) {
    1190           2 :                         DBG_DEBUG("num_shared=%zu\n", lck.num_shared);
    1191           2 :                         goto not_granted;
    1192             :                 }
    1193      943214 :                 lck.exclusive = state->me;
    1194          18 :         } else if (state->type == G_LOCK_READ) {
    1195          18 :                 g_lock_cleanup_shared(&lck);
    1196          18 :                 cb_state.new_shared = &state->me;
    1197             :         } else {
    1198           0 :                 smb_panic(__location__);
    1199             :         }
    1200             : 
    1201      943232 :         lck.unique_lock_epoch = generate_unique_u64(lck.unique_lock_epoch);
    1202             : 
    1203             :         /*
    1204             :          * We are going to store us as owner,
    1205             :          * so we got what we were waiting for.
    1206             :          *
    1207             :          * So we no longer need to monitor the
    1208             :          * record.
    1209             :          */
    1210      943232 :         dbwrap_watched_watch_skip_alerting(rec);
    1211             : 
    1212      943232 :         state->status = g_lock_lock_cb_run_and_store(&cb_state);
    1213      943232 :         if (!NT_STATUS_IS_OK(state->status) &&
    1214      450945 :             !NT_STATUS_EQUAL(state->status, NT_STATUS_WAS_UNLOCKED))
    1215             :         {
    1216           0 :                 DBG_WARNING("g_lock_lock_cb_run_and_store() failed: %s\n",
    1217             :                             nt_errstr(state->status));
    1218           0 :                 return;
    1219             :         }
    1220             : 
    1221      940922 :         return;
    1222             : 
    1223          19 : not_granted:
    1224          19 :         state->status = NT_STATUS_LOCK_NOT_GRANTED;
    1225             : }
    1226             : 
    1227      943254 : NTSTATUS g_lock_lock(struct g_lock_ctx *ctx, TDB_DATA key,
    1228             :                      enum g_lock_type type, struct timeval timeout,
    1229             :                      g_lock_lock_cb_fn_t cb_fn,
    1230             :                      void *cb_private)
    1231             : {
    1232        2318 :         TALLOC_CTX *frame;
    1233        2318 :         struct tevent_context *ev;
    1234        2318 :         struct tevent_req *req;
    1235        2318 :         struct timeval end;
    1236        2318 :         NTSTATUS status;
    1237             : 
    1238      943254 :         SMB_ASSERT(!ctx->busy);
    1239             : 
    1240             :         /*
    1241             :          * We allow a cn_fn only for G_LOCK_WRITE for now.
    1242             :          *
    1243             :          * It's all we currently need and it makes a few things
    1244             :          * easier to implement.
    1245             :          */
    1246      943254 :         if (unlikely(cb_fn != NULL && type != G_LOCK_WRITE)) {
    1247           0 :                 return NT_STATUS_INVALID_PARAMETER_5;
    1248             :         }
    1249             : 
    1250      943254 :         if ((type == G_LOCK_READ) || (type == G_LOCK_WRITE)) {
    1251             :                 /*
    1252             :                  * This is an abstraction violation: Normally we do
    1253             :                  * the sync wrappers around async functions with full
    1254             :                  * nested event contexts. However, this is used in
    1255             :                  * very hot code paths, so avoid the event context
    1256             :                  * creation for the good path where there's no lock
    1257             :                  * contention. My benchmark gave a factor of 2
    1258             :                  * improvement for lock/unlock.
    1259             :                  */
    1260     1886502 :                 struct g_lock_lock_simple_state state = {
    1261             :                         .ctx = ctx,
    1262      943251 :                         .me = messaging_server_id(ctx->msg),
    1263             :                         .type = type,
    1264             :                         .cb_fn = cb_fn,
    1265             :                         .cb_private = cb_private,
    1266             :                 };
    1267      943251 :                 status = dbwrap_do_locked(
    1268             :                         ctx->db, key, g_lock_lock_simple_fn, &state);
    1269      943251 :                 if (!NT_STATUS_IS_OK(status)) {
    1270           0 :                         DBG_DEBUG("dbwrap_do_locked() failed: %s\n",
    1271             :                                   nt_errstr(status));
    1272      943232 :                         return status;
    1273             :                 }
    1274             : 
    1275      943251 :                 DBG_DEBUG("status=%s, state.status=%s\n",
    1276             :                           nt_errstr(status),
    1277             :                           nt_errstr(state.status));
    1278             : 
    1279      943251 :                 if (NT_STATUS_IS_OK(state.status)) {
    1280      490818 :                         if (ctx->lock_order != DBWRAP_LOCK_ORDER_NONE) {
    1281      490789 :                                 const char *name = dbwrap_name(ctx->db);
    1282      490789 :                                 dbwrap_lock_order_lock(name, ctx->lock_order);
    1283             :                         }
    1284      490818 :                         return NT_STATUS_OK;
    1285             :                 }
    1286      452433 :                 if (NT_STATUS_EQUAL(state.status, NT_STATUS_WAS_UNLOCKED)) {
    1287             :                         /* without dbwrap_lock_order_lock() */
    1288      452414 :                         return NT_STATUS_OK;
    1289             :                 }
    1290          19 :                 if (!NT_STATUS_EQUAL(
    1291             :                             state.status, NT_STATUS_LOCK_NOT_GRANTED)) {
    1292           0 :                         return state.status;
    1293             :                 }
    1294             : 
    1295          19 :                 if (timeval_is_zero(&timeout)) {
    1296           0 :                         return NT_STATUS_LOCK_NOT_GRANTED;
    1297             :                 }
    1298             : 
    1299             :                 /*
    1300             :                  * Fall back to the full g_lock_trylock logic,
    1301             :                  * g_lock_lock_simple_fn() called above only covers
    1302             :                  * the uncontended path.
    1303             :                  */
    1304             :         }
    1305             : 
    1306          22 :         frame = talloc_stackframe();
    1307          22 :         status = NT_STATUS_NO_MEMORY;
    1308             : 
    1309          22 :         ev = samba_tevent_context_init(frame);
    1310          22 :         if (ev == NULL) {
    1311           0 :                 goto fail;
    1312             :         }
    1313          22 :         req = g_lock_lock_send(frame, ev, ctx, key, type, cb_fn, cb_private);
    1314          22 :         if (req == NULL) {
    1315           0 :                 goto fail;
    1316             :         }
    1317          22 :         end = timeval_current_ofs(timeout.tv_sec, timeout.tv_usec);
    1318          22 :         if (!tevent_req_set_endtime(req, ev, end)) {
    1319           0 :                 goto fail;
    1320             :         }
    1321          22 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1322           0 :                 goto fail;
    1323             :         }
    1324          22 :         status = g_lock_lock_recv(req);
    1325          22 :  fail:
    1326          22 :         TALLOC_FREE(frame);
    1327          22 :         return status;
    1328             : }
    1329             : 
    1330             : struct g_lock_unlock_state {
    1331             :         struct server_id self;
    1332             :         NTSTATUS status;
    1333             : };
    1334             : 
    1335      491095 : static void g_lock_unlock_fn(
    1336             :         struct db_record *rec,
    1337             :         TDB_DATA value,
    1338             :         void *private_data)
    1339             : {
    1340      491095 :         struct g_lock_unlock_state *state = private_data;
    1341         831 :         struct server_id_buf tmp1, tmp2;
    1342         831 :         struct g_lock lck;
    1343         831 :         size_t i;
    1344         831 :         bool ok, exclusive;
    1345             : 
    1346      491095 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1347      491095 :         if (!ok) {
    1348           0 :                 DBG_DEBUG("g_lock_parse() failed\n");
    1349           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1350           0 :                 return;
    1351             :         }
    1352             : 
    1353      491095 :         exclusive = server_id_equal(&state->self, &lck.exclusive);
    1354             : 
    1355      491940 :         for (i=0; i<lck.num_shared; i++) {
    1356          20 :                 struct server_id shared;
    1357          22 :                 g_lock_get_shared(&lck, i, &shared);
    1358          22 :                 if (server_id_equal(&state->self, &shared)) {
    1359           2 :                         break;
    1360             :                 }
    1361             :         }
    1362             : 
    1363      491095 :         if (i < lck.num_shared) {
    1364           8 :                 if (exclusive) {
    1365           0 :                         DBG_DEBUG("%s both exclusive and shared (%zu)\n",
    1366             :                                   server_id_str_buf(state->self, &tmp1),
    1367             :                                   i);
    1368           0 :                         state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1369           0 :                         return;
    1370             :                 }
    1371           8 :                 g_lock_del_shared(&lck, i);
    1372             :         } else {
    1373      491087 :                 if (!exclusive) {
    1374           1 :                         DBG_DEBUG("Lock not found, self=%s, lck.exclusive=%s, "
    1375             :                                   "num_shared=%zu\n",
    1376             :                                   server_id_str_buf(state->self, &tmp1),
    1377             :                                   server_id_str_buf(lck.exclusive, &tmp2),
    1378             :                                   lck.num_shared);
    1379           1 :                         state->status = NT_STATUS_NOT_FOUND;
    1380           1 :                         return;
    1381             :                 }
    1382      491086 :                 lck.exclusive = (struct server_id) { .pid = 0 };
    1383             :         }
    1384             : 
    1385      491094 :         if ((lck.exclusive.pid == 0) &&
    1386      491094 :             (lck.num_shared == 0) &&
    1387      491087 :             (lck.datalen == 0)) {
    1388      170670 :                 state->status = dbwrap_record_delete(rec);
    1389      170670 :                 return;
    1390             :         }
    1391             : 
    1392      320424 :         if (!exclusive && lck.exclusive.pid != 0) {
    1393             :                 /*
    1394             :                  * We only had a read lock and there's
    1395             :                  * someone waiting for an exclusive lock.
    1396             :                  *
    1397             :                  * Don't alert the exclusive lock waiter
    1398             :                  * if there are still other read lock holders.
    1399             :                  */
    1400           0 :                 g_lock_cleanup_shared(&lck);
    1401           0 :                 if (lck.num_shared != 0) {
    1402           0 :                         dbwrap_watched_watch_skip_alerting(rec);
    1403             :                 }
    1404             :         }
    1405             : 
    1406      320424 :         lck.unique_lock_epoch = generate_unique_u64(lck.unique_lock_epoch);
    1407             : 
    1408      320424 :         state->status = g_lock_store(rec, &lck, NULL, NULL, 0);
    1409             : }
    1410             : 
    1411      491095 : NTSTATUS g_lock_unlock(struct g_lock_ctx *ctx, TDB_DATA key)
    1412             : {
    1413      491095 :         struct g_lock_unlock_state state = {
    1414      491095 :                 .self = messaging_server_id(ctx->msg),
    1415             :         };
    1416         831 :         NTSTATUS status;
    1417             : 
    1418      491095 :         SMB_ASSERT(!ctx->busy);
    1419             : 
    1420      491095 :         status = dbwrap_do_locked(ctx->db, key, g_lock_unlock_fn, &state);
    1421      491095 :         if (!NT_STATUS_IS_OK(status)) {
    1422           0 :                 DBG_WARNING("dbwrap_do_locked failed: %s\n",
    1423             :                             nt_errstr(status));
    1424           0 :                 return status;
    1425             :         }
    1426      491095 :         if (!NT_STATUS_IS_OK(state.status)) {
    1427           1 :                 DBG_WARNING("g_lock_unlock_fn failed: %s\n",
    1428             :                             nt_errstr(state.status));
    1429           1 :                 return state.status;
    1430             :         }
    1431             : 
    1432      491094 :         if (ctx->lock_order != DBWRAP_LOCK_ORDER_NONE) {
    1433      490797 :                 const char *name = dbwrap_name(ctx->db);
    1434      490797 :                 dbwrap_lock_order_unlock(name, ctx->lock_order);
    1435             :         }
    1436             : 
    1437      491094 :         return NT_STATUS_OK;
    1438             : }
    1439             : 
    1440             : struct g_lock_writev_data_state {
    1441             :         TDB_DATA key;
    1442             :         struct server_id self;
    1443             :         const TDB_DATA *dbufs;
    1444             :         size_t num_dbufs;
    1445             :         NTSTATUS status;
    1446             : };
    1447             : 
    1448      172267 : static void g_lock_writev_data_fn(
    1449             :         struct db_record *rec,
    1450             :         TDB_DATA value,
    1451             :         void *private_data)
    1452             : {
    1453      172267 :         struct g_lock_writev_data_state *state = private_data;
    1454         356 :         struct g_lock lck;
    1455         356 :         bool exclusive;
    1456         356 :         bool ok;
    1457             : 
    1458             :         /*
    1459             :          * We're holding an exclusive write lock.
    1460             :          *
    1461             :          * Now we're updating the content of the record.
    1462             :          *
    1463             :          * We should not wakeup any other waiters, all they
    1464             :          * would find is that we're still holding a lock they
    1465             :          * are conflicting with.
    1466             :          */
    1467      172267 :         dbwrap_watched_watch_skip_alerting(rec);
    1468             : 
    1469      172267 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1470      172267 :         if (!ok) {
    1471           0 :                 DBG_DEBUG("g_lock_parse for %s failed\n",
    1472             :                           tdb_data_dbg(state->key));
    1473           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1474           0 :                 return;
    1475             :         }
    1476             : 
    1477      172267 :         exclusive = server_id_equal(&state->self, &lck.exclusive);
    1478             : 
    1479             :         /*
    1480             :          * Make sure we're really exclusive. We are marked as
    1481             :          * exclusive when we are waiting for an exclusive lock
    1482             :          */
    1483      172267 :         exclusive &= (lck.num_shared == 0);
    1484             : 
    1485      172267 :         if (!exclusive) {
    1486           1 :                 struct server_id_buf buf1, buf2;
    1487           1 :                 DBG_DEBUG("Not locked by us: self=%s, lck.exclusive=%s, "
    1488             :                           "lck.num_shared=%zu\n",
    1489             :                           server_id_str_buf(state->self, &buf1),
    1490             :                           server_id_str_buf(lck.exclusive, &buf2),
    1491             :                           lck.num_shared);
    1492           1 :                 state->status = NT_STATUS_NOT_LOCKED;
    1493           1 :                 return;
    1494             :         }
    1495             : 
    1496      172266 :         lck.unique_data_epoch = generate_unique_u64(lck.unique_data_epoch);
    1497      172266 :         lck.data = NULL;
    1498      172266 :         lck.datalen = 0;
    1499      172266 :         state->status = g_lock_store(
    1500             :                 rec, &lck, NULL, state->dbufs, state->num_dbufs);
    1501             : }
    1502             : 
    1503      172267 : NTSTATUS g_lock_writev_data(
    1504             :         struct g_lock_ctx *ctx,
    1505             :         TDB_DATA key,
    1506             :         const TDB_DATA *dbufs,
    1507             :         size_t num_dbufs)
    1508             : {
    1509      344534 :         struct g_lock_writev_data_state state = {
    1510             :                 .key = key,
    1511      172267 :                 .self = messaging_server_id(ctx->msg),
    1512             :                 .dbufs = dbufs,
    1513             :                 .num_dbufs = num_dbufs,
    1514             :         };
    1515         356 :         NTSTATUS status;
    1516             : 
    1517      172267 :         SMB_ASSERT(!ctx->busy);
    1518             : 
    1519      172267 :         status = dbwrap_do_locked(
    1520             :                 ctx->db, key, g_lock_writev_data_fn, &state);
    1521      172267 :         if (!NT_STATUS_IS_OK(status)) {
    1522           0 :                 DBG_WARNING("dbwrap_do_locked failed: %s\n",
    1523             :                             nt_errstr(status));
    1524           0 :                 return status;
    1525             :         }
    1526      172267 :         if (!NT_STATUS_IS_OK(state.status)) {
    1527           1 :                 DBG_WARNING("g_lock_writev_data_fn failed: %s\n",
    1528             :                             nt_errstr(state.status));
    1529           1 :                 return state.status;
    1530             :         }
    1531             : 
    1532      172266 :         return NT_STATUS_OK;
    1533             : }
    1534             : 
    1535           4 : NTSTATUS g_lock_write_data(struct g_lock_ctx *ctx, TDB_DATA key,
    1536             :                            const uint8_t *buf, size_t buflen)
    1537             : {
    1538           4 :         TDB_DATA dbuf = {
    1539             :                 .dptr = discard_const_p(uint8_t, buf),
    1540             :                 .dsize = buflen,
    1541             :         };
    1542           4 :         return g_lock_writev_data(ctx, key, &dbuf, 1);
    1543             : }
    1544             : 
    1545             : struct g_lock_locks_state {
    1546             :         int (*fn)(TDB_DATA key, void *private_data);
    1547             :         void *private_data;
    1548             : };
    1549             : 
    1550       18777 : static int g_lock_locks_fn(struct db_record *rec, void *priv)
    1551             : {
    1552           0 :         TDB_DATA key;
    1553       18777 :         struct g_lock_locks_state *state = (struct g_lock_locks_state *)priv;
    1554             : 
    1555       18777 :         key = dbwrap_record_get_key(rec);
    1556       18777 :         return state->fn(key, state->private_data);
    1557             : }
    1558             : 
    1559        6333 : int g_lock_locks(struct g_lock_ctx *ctx,
    1560             :                  int (*fn)(TDB_DATA key, void *private_data),
    1561             :                  void *private_data)
    1562             : {
    1563           0 :         struct g_lock_locks_state state;
    1564           0 :         NTSTATUS status;
    1565           0 :         int count;
    1566             : 
    1567        6333 :         SMB_ASSERT(!ctx->busy);
    1568             : 
    1569        6333 :         state.fn = fn;
    1570        6333 :         state.private_data = private_data;
    1571             : 
    1572        6333 :         status = dbwrap_traverse_read(ctx->db, g_lock_locks_fn, &state, &count);
    1573        6333 :         if (!NT_STATUS_IS_OK(status)) {
    1574           0 :                 return -1;
    1575             :         }
    1576        6333 :         return count;
    1577             : }
    1578             : 
    1579             : struct g_lock_dump_state {
    1580             :         TALLOC_CTX *mem_ctx;
    1581             :         TDB_DATA key;
    1582             :         void (*fn)(struct server_id exclusive,
    1583             :                    size_t num_shared,
    1584             :                    const struct server_id *shared,
    1585             :                    const uint8_t *data,
    1586             :                    size_t datalen,
    1587             :                    void *private_data);
    1588             :         void *private_data;
    1589             :         NTSTATUS status;
    1590             :         enum dbwrap_req_state req_state;
    1591             : };
    1592             : 
    1593      262116 : static void g_lock_dump_fn(TDB_DATA key, TDB_DATA data,
    1594             :                            void *private_data)
    1595             : {
    1596      262116 :         struct g_lock_dump_state *state = private_data;
    1597      262116 :         struct g_lock lck = (struct g_lock) { .exclusive.pid = 0 };
    1598      262116 :         struct server_id *shared = NULL;
    1599        1377 :         size_t i;
    1600        1377 :         bool ok;
    1601             : 
    1602      262116 :         ok = g_lock_parse(data.dptr, data.dsize, &lck);
    1603      262116 :         if (!ok) {
    1604           0 :                 DBG_DEBUG("g_lock_parse failed for %s\n",
    1605             :                           tdb_data_dbg(state->key));
    1606           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1607           0 :                 return;
    1608             :         }
    1609             : 
    1610      262116 :         if (lck.num_shared > 0) {
    1611           7 :                 shared = talloc_array(
    1612             :                         state->mem_ctx, struct server_id, lck.num_shared);
    1613           7 :                 if (shared == NULL) {
    1614           0 :                         DBG_DEBUG("talloc failed\n");
    1615           0 :                         state->status = NT_STATUS_NO_MEMORY;
    1616           0 :                         return;
    1617             :                 }
    1618             :         }
    1619             : 
    1620      262134 :         for (i=0; i<lck.num_shared; i++) {
    1621          18 :                 g_lock_get_shared(&lck, i, &shared[i]);
    1622             :         }
    1623             : 
    1624      262116 :         state->fn(lck.exclusive,
    1625             :                   lck.num_shared,
    1626             :                   shared,
    1627      262116 :                   lck.data,
    1628             :                   lck.datalen,
    1629             :                   state->private_data);
    1630             : 
    1631      262116 :         TALLOC_FREE(shared);
    1632             : 
    1633      262116 :         state->status = NT_STATUS_OK;
    1634             : }
    1635             : 
    1636     1086150 : NTSTATUS g_lock_dump(struct g_lock_ctx *ctx, TDB_DATA key,
    1637             :                      void (*fn)(struct server_id exclusive,
    1638             :                                 size_t num_shared,
    1639             :                                 const struct server_id *shared,
    1640             :                                 const uint8_t *data,
    1641             :                                 size_t datalen,
    1642             :                                 void *private_data),
    1643             :                      void *private_data)
    1644             : {
    1645     1086150 :         struct g_lock_dump_state state = {
    1646             :                 .mem_ctx = ctx, .key = key,
    1647             :                 .fn = fn, .private_data = private_data
    1648             :         };
    1649        1812 :         NTSTATUS status;
    1650             : 
    1651     1086150 :         SMB_ASSERT(!ctx->busy);
    1652             : 
    1653     1086150 :         status = dbwrap_parse_record(ctx->db, key, g_lock_dump_fn, &state);
    1654     1086150 :         if (!NT_STATUS_IS_OK(status)) {
    1655      824034 :                 DBG_DEBUG("dbwrap_parse_record returned %s\n",
    1656             :                           nt_errstr(status));
    1657      824034 :                 return status;
    1658             :         }
    1659      262116 :         if (!NT_STATUS_IS_OK(state.status)) {
    1660           0 :                 DBG_DEBUG("g_lock_dump_fn returned %s\n",
    1661             :                           nt_errstr(state.status));
    1662           0 :                 return state.status;
    1663             :         }
    1664      262116 :         return NT_STATUS_OK;
    1665             : }
    1666             : 
    1667             : static void g_lock_dump_done(struct tevent_req *subreq);
    1668             : 
    1669           0 : struct tevent_req *g_lock_dump_send(
    1670             :         TALLOC_CTX *mem_ctx,
    1671             :         struct tevent_context *ev,
    1672             :         struct g_lock_ctx *ctx,
    1673             :         TDB_DATA key,
    1674             :         void (*fn)(struct server_id exclusive,
    1675             :                    size_t num_shared,
    1676             :                    const struct server_id *shared,
    1677             :                    const uint8_t *data,
    1678             :                    size_t datalen,
    1679             :                    void *private_data),
    1680             :         void *private_data)
    1681             : {
    1682           0 :         struct tevent_req *req = NULL, *subreq = NULL;
    1683           0 :         struct g_lock_dump_state *state = NULL;
    1684             : 
    1685           0 :         SMB_ASSERT(!ctx->busy);
    1686             : 
    1687           0 :         req = tevent_req_create(mem_ctx, &state, struct g_lock_dump_state);
    1688           0 :         if (req == NULL) {
    1689           0 :                 return NULL;
    1690             :         }
    1691           0 :         state->mem_ctx = state;
    1692           0 :         state->key = key;
    1693           0 :         state->fn = fn;
    1694           0 :         state->private_data = private_data;
    1695             : 
    1696           0 :         SMB_ASSERT(!ctx->busy);
    1697             : 
    1698           0 :         subreq = dbwrap_parse_record_send(
    1699             :                 state,
    1700             :                 ev,
    1701             :                 ctx->db,
    1702             :                 key,
    1703             :                 g_lock_dump_fn,
    1704             :                 state,
    1705           0 :                 &state->req_state);
    1706           0 :         if (tevent_req_nomem(subreq, req)) {
    1707           0 :                 return tevent_req_post(req, ev);
    1708             :         }
    1709           0 :         tevent_req_set_callback(subreq, g_lock_dump_done, req);
    1710           0 :         return req;
    1711             : }
    1712             : 
    1713           0 : static void g_lock_dump_done(struct tevent_req *subreq)
    1714             : {
    1715           0 :         struct tevent_req *req = tevent_req_callback_data(
    1716             :                 subreq, struct tevent_req);
    1717           0 :         struct g_lock_dump_state *state = tevent_req_data(
    1718             :                 req, struct g_lock_dump_state);
    1719           0 :         NTSTATUS status;
    1720             : 
    1721           0 :         status = dbwrap_parse_record_recv(subreq);
    1722           0 :         TALLOC_FREE(subreq);
    1723           0 :         if (tevent_req_nterror(req, status) ||
    1724           0 :             tevent_req_nterror(req, state->status)) {
    1725           0 :                 return;
    1726             :         }
    1727           0 :         tevent_req_done(req);
    1728             : }
    1729             : 
    1730           0 : NTSTATUS g_lock_dump_recv(struct tevent_req *req)
    1731             : {
    1732           0 :         return tevent_req_simple_recv_ntstatus(req);
    1733             : }
    1734             : 
    1735      193220 : int g_lock_seqnum(struct g_lock_ctx *ctx)
    1736             : {
    1737      193220 :         return dbwrap_get_seqnum(ctx->db);
    1738             : }
    1739             : 
    1740             : struct g_lock_watch_data_state {
    1741             :         struct tevent_context *ev;
    1742             :         struct g_lock_ctx *ctx;
    1743             :         TDB_DATA key;
    1744             :         struct server_id blocker;
    1745             :         bool blockerdead;
    1746             :         uint64_t unique_lock_epoch;
    1747             :         uint64_t unique_data_epoch;
    1748             :         uint64_t watch_instance;
    1749             :         NTSTATUS status;
    1750             : };
    1751             : 
    1752             : static void g_lock_watch_data_done(struct tevent_req *subreq);
    1753             : 
    1754         653 : static void g_lock_watch_data_send_fn(
    1755             :         struct db_record *rec,
    1756             :         TDB_DATA value,
    1757             :         void *private_data)
    1758             : {
    1759         653 :         struct tevent_req *req = talloc_get_type_abort(
    1760             :                 private_data, struct tevent_req);
    1761         653 :         struct g_lock_watch_data_state *state = tevent_req_data(
    1762             :                 req, struct g_lock_watch_data_state);
    1763         653 :         struct tevent_req *subreq = NULL;
    1764           3 :         struct g_lock lck;
    1765           3 :         bool ok;
    1766             : 
    1767         653 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1768         653 :         if (!ok) {
    1769           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1770           0 :                 return;
    1771             :         }
    1772         653 :         state->unique_lock_epoch = lck.unique_lock_epoch;
    1773         653 :         state->unique_data_epoch = lck.unique_data_epoch;
    1774             : 
    1775         653 :         DBG_DEBUG("state->unique_data_epoch=%"PRIu64"\n", state->unique_data_epoch);
    1776             : 
    1777         653 :         subreq = dbwrap_watched_watch_send(
    1778             :                 state, state->ev, rec, 0, state->blocker);
    1779         653 :         if (subreq == NULL) {
    1780           0 :                 state->status = NT_STATUS_NO_MEMORY;
    1781           0 :                 return;
    1782             :         }
    1783         653 :         tevent_req_set_callback(subreq, g_lock_watch_data_done, req);
    1784             : 
    1785         653 :         state->status = NT_STATUS_EVENT_PENDING;
    1786             : }
    1787             : 
    1788         653 : struct tevent_req *g_lock_watch_data_send(
    1789             :         TALLOC_CTX *mem_ctx,
    1790             :         struct tevent_context *ev,
    1791             :         struct g_lock_ctx *ctx,
    1792             :         TDB_DATA key,
    1793             :         struct server_id blocker)
    1794             : {
    1795         653 :         struct tevent_req *req = NULL;
    1796         653 :         struct g_lock_watch_data_state *state = NULL;
    1797           3 :         NTSTATUS status;
    1798             : 
    1799         653 :         SMB_ASSERT(!ctx->busy);
    1800             : 
    1801         653 :         req = tevent_req_create(
    1802             :                 mem_ctx, &state, struct g_lock_watch_data_state);
    1803         653 :         if (req == NULL) {
    1804           0 :                 return NULL;
    1805             :         }
    1806         653 :         state->ev = ev;
    1807         653 :         state->ctx = ctx;
    1808         653 :         state->blocker = blocker;
    1809             : 
    1810         653 :         state->key = tdb_data_talloc_copy(state, key);
    1811         653 :         if (tevent_req_nomem(state->key.dptr, req)) {
    1812           0 :                 return tevent_req_post(req, ev);
    1813             :         }
    1814             : 
    1815         653 :         status = dbwrap_do_locked(
    1816             :                 ctx->db, key, g_lock_watch_data_send_fn, req);
    1817         653 :         if (tevent_req_nterror(req, status)) {
    1818           0 :                 DBG_DEBUG("dbwrap_do_locked returned %s\n", nt_errstr(status));
    1819           0 :                 return tevent_req_post(req, ev);
    1820             :         }
    1821             : 
    1822         653 :         if (NT_STATUS_EQUAL(state->status, NT_STATUS_EVENT_PENDING)) {
    1823         650 :                 return req;
    1824             :         }
    1825           0 :         if (tevent_req_nterror(req, state->status)) {
    1826           0 :                 return tevent_req_post(req, ev);
    1827             :         }
    1828           0 :         tevent_req_done(req);
    1829           0 :         return tevent_req_post(req, ev);
    1830             : }
    1831             : 
    1832        1003 : static void g_lock_watch_data_done_fn(
    1833             :         struct db_record *rec,
    1834             :         TDB_DATA value,
    1835             :         void *private_data)
    1836             : {
    1837        1003 :         struct tevent_req *req = talloc_get_type_abort(
    1838             :                 private_data, struct tevent_req);
    1839        1003 :         struct g_lock_watch_data_state *state = tevent_req_data(
    1840             :                 req, struct g_lock_watch_data_state);
    1841        1003 :         struct tevent_req *subreq = NULL;
    1842           3 :         struct g_lock lck;
    1843           3 :         bool ok;
    1844             : 
    1845        1003 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1846        1003 :         if (!ok) {
    1847           0 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
    1848           0 :                 state->status = NT_STATUS_INTERNAL_DB_CORRUPTION;
    1849           0 :                 return;
    1850             :         }
    1851             : 
    1852        1003 :         if (lck.unique_data_epoch != state->unique_data_epoch) {
    1853          87 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
    1854          87 :                 DBG_DEBUG("lck.unique_data_epoch=%"PRIu64", "
    1855             :                           "state->unique_data_epoch=%"PRIu64"\n",
    1856             :                           lck.unique_data_epoch,
    1857             :                           state->unique_data_epoch);
    1858          87 :                 state->status = NT_STATUS_OK;
    1859          87 :                 return;
    1860             :         }
    1861             : 
    1862             :         /*
    1863             :          * The lock epoch changed, so we better
    1864             :          * remove ourself from the waiter list
    1865             :          * (most likely the first position)
    1866             :          * and re-add us at the end of the list.
    1867             :          *
    1868             :          * This gives other lock waiters a change
    1869             :          * to make progress.
    1870             :          *
    1871             :          * Otherwise we'll keep our waiter instance alive,
    1872             :          * keep waiting (most likely at first position).
    1873             :          */
    1874         916 :         if (lck.unique_lock_epoch != state->unique_lock_epoch) {
    1875         830 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
    1876         830 :                 state->watch_instance = dbwrap_watched_watch_add_instance(rec);
    1877         830 :                 state->unique_lock_epoch = lck.unique_lock_epoch;
    1878             :         }
    1879             : 
    1880         916 :         subreq = dbwrap_watched_watch_send(
    1881             :                 state, state->ev, rec, state->watch_instance, state->blocker);
    1882         916 :         if (subreq == NULL) {
    1883           0 :                 dbwrap_watched_watch_remove_instance(rec, state->watch_instance);
    1884           0 :                 state->status = NT_STATUS_NO_MEMORY;
    1885           0 :                 return;
    1886             :         }
    1887         916 :         tevent_req_set_callback(subreq, g_lock_watch_data_done, req);
    1888             : 
    1889         916 :         state->status = NT_STATUS_EVENT_PENDING;
    1890             : }
    1891             : 
    1892        1003 : static void g_lock_watch_data_done(struct tevent_req *subreq)
    1893             : {
    1894        1003 :         struct tevent_req *req = tevent_req_callback_data(
    1895             :                 subreq, struct tevent_req);
    1896        1003 :         struct g_lock_watch_data_state *state = tevent_req_data(
    1897             :                 req, struct g_lock_watch_data_state);
    1898           3 :         NTSTATUS status;
    1899        1003 :         uint64_t instance = 0;
    1900             : 
    1901        1003 :         status = dbwrap_watched_watch_recv(
    1902             :                 subreq, &instance, &state->blockerdead, &state->blocker);
    1903        1003 :         TALLOC_FREE(subreq);
    1904        1003 :         if (tevent_req_nterror(req, status)) {
    1905           0 :                 DBG_DEBUG("dbwrap_watched_watch_recv returned %s\n",
    1906             :                           nt_errstr(status));
    1907         916 :                 return;
    1908             :         }
    1909             : 
    1910        1003 :         state->watch_instance = instance;
    1911             : 
    1912        1003 :         status = dbwrap_do_locked(
    1913        1003 :                 state->ctx->db, state->key, g_lock_watch_data_done_fn, req);
    1914        1003 :         if (tevent_req_nterror(req, status)) {
    1915           0 :                 DBG_DEBUG("dbwrap_do_locked returned %s\n", nt_errstr(status));
    1916           0 :                 return;
    1917             :         }
    1918        1003 :         if (NT_STATUS_EQUAL(state->status, NT_STATUS_EVENT_PENDING)) {
    1919         914 :                 return;
    1920             :         }
    1921          87 :         if (tevent_req_nterror(req, state->status)) {
    1922           0 :                 return;
    1923             :         }
    1924          87 :         tevent_req_done(req);
    1925             : }
    1926             : 
    1927          86 : NTSTATUS g_lock_watch_data_recv(
    1928             :         struct tevent_req *req,
    1929             :         bool *blockerdead,
    1930             :         struct server_id *blocker)
    1931             : {
    1932          86 :         struct g_lock_watch_data_state *state = tevent_req_data(
    1933             :                 req, struct g_lock_watch_data_state);
    1934           0 :         NTSTATUS status;
    1935             : 
    1936          86 :         if (tevent_req_is_nterror(req, &status)) {
    1937           0 :                 return status;
    1938             :         }
    1939          86 :         if (blockerdead != NULL) {
    1940          86 :                 *blockerdead = state->blockerdead;
    1941             :         }
    1942          86 :         if (blocker != NULL) {
    1943          86 :                 *blocker = state->blocker;
    1944             :         }
    1945             : 
    1946          86 :         return NT_STATUS_OK;
    1947             : }
    1948             : 
    1949        2328 : static void g_lock_wake_watchers_fn(
    1950             :         struct db_record *rec,
    1951             :         TDB_DATA value,
    1952             :         void *private_data)
    1953             : {
    1954        2328 :         struct g_lock lck = { .exclusive.pid = 0 };
    1955          10 :         NTSTATUS status;
    1956          10 :         bool ok;
    1957             : 
    1958        2328 :         ok = g_lock_parse(value.dptr, value.dsize, &lck);
    1959        2328 :         if (!ok) {
    1960           0 :                 DBG_WARNING("g_lock_parse failed\n");
    1961           0 :                 return;
    1962             :         }
    1963             : 
    1964        2328 :         lck.unique_data_epoch = generate_unique_u64(lck.unique_data_epoch);
    1965             : 
    1966        2328 :         status = g_lock_store(rec, &lck, NULL, NULL, 0);
    1967        2328 :         if (!NT_STATUS_IS_OK(status)) {
    1968           0 :                 DBG_WARNING("g_lock_store failed: %s\n", nt_errstr(status));
    1969           0 :                 return;
    1970             :         }
    1971             : }
    1972             : 
    1973        2328 : void g_lock_wake_watchers(struct g_lock_ctx *ctx, TDB_DATA key)
    1974             : {
    1975          10 :         NTSTATUS status;
    1976             : 
    1977        2328 :         SMB_ASSERT(!ctx->busy);
    1978             : 
    1979        2328 :         status = dbwrap_do_locked(ctx->db, key, g_lock_wake_watchers_fn, NULL);
    1980        2328 :         if (!NT_STATUS_IS_OK(status)) {
    1981           0 :                 DBG_DEBUG("dbwrap_do_locked returned %s\n",
    1982             :                           nt_errstr(status));
    1983             :         }
    1984        2328 : }

Generated by: LCOV version 1.14