source: branches/samba-3.0/source/lib/messages.c@ 726

Last change on this file since 726 was 165, checked in by Paul Smedley, 17 years ago

Add 'missing' 3.0.34 diffs

File size: 21.6 KB
Line 
1/*
2 Unix SMB/CIFS implementation.
3 Samba internal messaging functions
4 Copyright (C) Andrew Tridgell 2000
5 Copyright (C) 2001 by Martin Pool
6 Copyright (C) 2002 by Jeremy Allison
7
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21*/
22
23/**
24 @defgroup messages Internal messaging framework
25 @{
26 @file messages.c
27
28 @brief Module for internal messaging between Samba daemons.
29
30 The idea is that if a part of Samba wants to do communication with
31 another Samba process then it will do a message_register() of a
32 dispatch function, and use message_send_pid() to send messages to
33 that process.
34
35 The dispatch function is given the pid of the sender, and it can
36 use that to reply by message_send_pid(). See ping_message() for a
37 simple example.
38
39 @caution Dispatch functions must be able to cope with incoming
40 messages on an *odd* byte boundary.
41
42 This system doesn't have any inherent size limitations but is not
43 very efficient for large messages or when messages are sent in very
44 quick succession.
45
46*/
47
48#include "includes.h"
49
50/* the locking database handle */
51static TDB_CONTEXT *tdb;
52static int received_signal;
53
54/* change the message version with any incompatible changes in the protocol */
55#define MESSAGE_VERSION 1
56
57struct message_rec {
58 int msg_version;
59 int msg_type;
60 struct process_id dest;
61 struct process_id src;
62 size_t len;
63};
64
65/* we have a linked list of dispatch handlers */
66static struct dispatch_fns {
67 struct dispatch_fns *next, *prev;
68 int msg_type;
69 void (*fn)(int msg_type, struct process_id pid, void *buf, size_t len,
70 void *private_data);
71 void *private_data;
72} *dispatch_fns;
73
74/****************************************************************************
75 Free global objects.
76****************************************************************************/
77
78void gfree_messages(void)
79{
80 struct dispatch_fns *dfn, *next;
81
82 /* delete the dispatch_fns list */
83 dfn = dispatch_fns;
84 while( dfn ) {
85 next = dfn->next;
86 DLIST_REMOVE(dispatch_fns, dfn);
87 SAFE_FREE(dfn);
88 dfn = next;
89 }
90}
91
92/****************************************************************************
93 Notifications come in as signals.
94****************************************************************************/
95
96static void sig_usr1(void)
97{
98 received_signal = 1;
99 sys_select_signal(SIGUSR1);
100}
101
102/****************************************************************************
103 A useful function for testing the message system.
104****************************************************************************/
105
106static void ping_message(int msg_type, struct process_id src,
107 void *buf, size_t len, void *private_data)
108{
109 const char *msg = buf ? (const char *)buf : "none";
110
111 DEBUG(1,("INFO: Received PING message from PID %s [%s]\n",
112 procid_str_static(&src), msg));
113 message_send_pid(src, MSG_PONG, buf, len, True);
114}
115
116/****************************************************************************
117 Initialise the messaging functions.
118****************************************************************************/
119
120BOOL message_init(void)
121{
122 sec_init();
123
124 if (tdb)
125 return True;
126
127 tdb = tdb_open_log(lock_path("messages.tdb"),
128 0, TDB_CLEAR_IF_FIRST|TDB_DEFAULT,
129 O_RDWR|O_CREAT,0600);
130
131 if (!tdb) {
132 DEBUG(0,("ERROR: Failed to initialise messages database\n"));
133 return False;
134 }
135
136 /* Activate the per-hashchain freelist */
137 tdb_set_max_dead(tdb, 5);
138
139 CatchSignal(SIGUSR1, SIGNAL_CAST sig_usr1);
140
141 message_register(MSG_PING, ping_message, NULL);
142
143 /* Register some debugging related messages */
144
145 register_msg_pool_usage();
146 register_dmalloc_msgs();
147
148 return True;
149}
150
151/*******************************************************************
152 Form a static tdb key from a pid.
153******************************************************************/
154
155static TDB_DATA message_key_pid(struct process_id pid)
156{
157 static char key[20];
158 TDB_DATA kbuf;
159
160 slprintf(key, sizeof(key)-1, "PID/%s", procid_str_static(&pid));
161
162 kbuf.dptr = (char *)key;
163 kbuf.dsize = strlen(key)+1;
164 return kbuf;
165}
166
167/****************************************************************************
168 Notify a process that it has a message. If the process doesn't exist
169 then delete its record in the database.
170****************************************************************************/
171
172static NTSTATUS message_notify(struct process_id procid)
173{
174 pid_t pid = procid.pid;
175 int ret;
176 uid_t euid = geteuid();
177
178 /*
179 * Doing kill with a non-positive pid causes messages to be
180 * sent to places we don't want.
181 */
182
183 SMB_ASSERT(pid > 0);
184
185 if (euid != 0) {
186 /* If we're not root become so to send the message. */
187 save_re_uid();
188 set_effective_uid(0);
189 }
190
191 ret = kill(pid, SIGUSR1);
192
193 if (euid != 0) {
194 /* Go back to who we were. */
195 int saved_errno = errno;
196 restore_re_uid_fromroot();
197 errno = saved_errno;
198 }
199
200 if (ret == -1) {
201 if (errno == ESRCH) {
202 DEBUG(2,("pid %d doesn't exist - deleting messages record\n",
203 (int)pid));
204 tdb_delete(tdb, message_key_pid(procid));
205
206 /*
207 * INVALID_HANDLE is the closest I can think of -- vl
208 */
209 return NT_STATUS_INVALID_HANDLE;
210 }
211
212 DEBUG(2,("message to process %d failed - %s\n", (int)pid,
213 strerror(errno)));
214
215 /*
216 * No call to map_nt_error_from_unix -- don't want to link in
217 * errormap.o into lots of utils.
218 */
219
220 if (errno == EINVAL) return NT_STATUS_INVALID_PARAMETER;
221 if (errno == EPERM) return NT_STATUS_ACCESS_DENIED;
222 return NT_STATUS_UNSUCCESSFUL;
223 }
224
225 return NT_STATUS_OK;
226}
227
228/****************************************************************************
229 Send a message to a particular pid.
230****************************************************************************/
231
232static NTSTATUS message_send_pid_internal(struct process_id pid, int msg_type,
233 const void *buf, size_t len,
234 BOOL duplicates_allowed,
235 unsigned int timeout)
236{
237 TDB_DATA kbuf;
238 TDB_DATA dbuf;
239 TDB_DATA old_dbuf;
240 struct message_rec rec;
241 char *ptr;
242 struct message_rec prec;
243
244 /* NULL pointer means implicit length zero. */
245 if (!buf) {
246 SMB_ASSERT(len == 0);
247 }
248
249 /*
250 * Doing kill with a non-positive pid causes messages to be
251 * sent to places we don't want.
252 */
253
254 SMB_ASSERT(procid_to_pid(&pid) > 0);
255
256 rec.msg_version = MESSAGE_VERSION;
257 rec.msg_type = msg_type;
258 rec.dest = pid;
259 rec.src = procid_self();
260 rec.len = buf ? len : 0;
261
262 kbuf = message_key_pid(pid);
263
264 dbuf.dptr = (char *)SMB_MALLOC(len + sizeof(rec));
265 if (!dbuf.dptr) {
266 return NT_STATUS_NO_MEMORY;
267 }
268
269 memcpy(dbuf.dptr, &rec, sizeof(rec));
270 if (len > 0 && buf)
271 memcpy((void *)((char*)dbuf.dptr+sizeof(rec)), buf, len);
272
273 dbuf.dsize = len + sizeof(rec);
274
275 if (duplicates_allowed) {
276
277 /* If duplicates are allowed we can just append the message and return. */
278
279 /* lock the record for the destination */
280 if (timeout) {
281 if (tdb_chainlock_with_timeout(tdb, kbuf, timeout) == -1) {
282 DEBUG(0,("message_send_pid_internal: failed to get "
283 "chainlock with timeout %ul.\n", timeout));
284 SAFE_FREE(dbuf.dptr);
285 return NT_STATUS_IO_TIMEOUT;
286 }
287 } else {
288 if (tdb_chainlock(tdb, kbuf) == -1) {
289 DEBUG(0,("message_send_pid_internal: failed to get "
290 "chainlock.\n"));
291 SAFE_FREE(dbuf.dptr);
292 return NT_STATUS_LOCK_NOT_GRANTED;
293 }
294 }
295 tdb_append(tdb, kbuf, dbuf);
296 tdb_chainunlock(tdb, kbuf);
297
298 SAFE_FREE(dbuf.dptr);
299 errno = 0; /* paranoia */
300 return message_notify(pid);
301 }
302
303 /* lock the record for the destination */
304 if (timeout) {
305 if (tdb_chainlock_with_timeout(tdb, kbuf, timeout) == -1) {
306 DEBUG(0,("message_send_pid_internal: failed to get chainlock "
307 "with timeout %ul.\n", timeout));
308 SAFE_FREE(dbuf.dptr);
309 return NT_STATUS_IO_TIMEOUT;
310 }
311 } else {
312 if (tdb_chainlock(tdb, kbuf) == -1) {
313 DEBUG(0,("message_send_pid_internal: failed to get "
314 "chainlock.\n"));
315 SAFE_FREE(dbuf.dptr);
316 return NT_STATUS_LOCK_NOT_GRANTED;
317 }
318 }
319
320 old_dbuf = tdb_fetch(tdb, kbuf);
321
322 if (!old_dbuf.dptr) {
323 /* its a new record */
324
325 tdb_store(tdb, kbuf, dbuf, TDB_REPLACE);
326 tdb_chainunlock(tdb, kbuf);
327
328 SAFE_FREE(dbuf.dptr);
329 errno = 0; /* paranoia */
330 return message_notify(pid);
331 }
332
333 /* Not a new record. Check for duplicates. */
334
335 for(ptr = (char *)old_dbuf.dptr; ptr < old_dbuf.dptr + old_dbuf.dsize; ) {
336 /*
337 * First check if the message header matches, then, if it's a non-zero
338 * sized message, check if the data matches. If so it's a duplicate and
339 * we can discard it. JRA.
340 */
341
342 if (!memcmp(ptr, &rec, sizeof(rec))) {
343 if (!len || (len && !memcmp( ptr + sizeof(rec), buf, len))) {
344 tdb_chainunlock(tdb, kbuf);
345 DEBUG(10,("message_send_pid_internal: discarding "
346 "duplicate message.\n"));
347 SAFE_FREE(dbuf.dptr);
348 SAFE_FREE(old_dbuf.dptr);
349 return NT_STATUS_OK;
350 }
351 }
352 memcpy(&prec, ptr, sizeof(prec));
353 ptr += sizeof(rec) + prec.len;
354 }
355
356 /* we're adding to an existing entry */
357
358 tdb_append(tdb, kbuf, dbuf);
359 tdb_chainunlock(tdb, kbuf);
360
361 SAFE_FREE(old_dbuf.dptr);
362 SAFE_FREE(dbuf.dptr);
363
364 errno = 0; /* paranoia */
365 return message_notify(pid);
366}
367
368/****************************************************************************
369 Send a message to a particular pid - no timeout.
370****************************************************************************/
371
372NTSTATUS message_send_pid(struct process_id pid, int msg_type, const void *buf,
373 size_t len, BOOL duplicates_allowed)
374{
375 return message_send_pid_internal(pid, msg_type, buf, len,
376 duplicates_allowed, 0);
377}
378
379/****************************************************************************
380 Send a message to a particular pid, with timeout in seconds.
381****************************************************************************/
382
383NTSTATUS message_send_pid_with_timeout(struct process_id pid, int msg_type,
384 const void *buf, size_t len,
385 BOOL duplicates_allowed, unsigned int timeout)
386{
387 return message_send_pid_internal(pid, msg_type, buf, len, duplicates_allowed,
388 timeout);
389}
390
391/****************************************************************************
392 Count the messages pending for a particular pid. Expensive....
393****************************************************************************/
394
395unsigned int messages_pending_for_pid(struct process_id pid)
396{
397 TDB_DATA kbuf;
398 TDB_DATA dbuf;
399 char *buf;
400 unsigned int message_count = 0;
401
402 kbuf = message_key_pid(pid);
403
404 dbuf = tdb_fetch(tdb, kbuf);
405 if (dbuf.dptr == NULL || dbuf.dsize == 0) {
406 SAFE_FREE(dbuf.dptr);
407 return 0;
408 }
409
410 for (buf = dbuf.dptr; dbuf.dsize > sizeof(struct message_rec);) {
411 struct message_rec rec;
412 memcpy(&rec, buf, sizeof(rec));
413 buf += (sizeof(rec) + rec.len);
414 dbuf.dsize -= (sizeof(rec) + rec.len);
415 message_count++;
416 }
417
418 SAFE_FREE(dbuf.dptr);
419 return message_count;
420}
421
422/****************************************************************************
423 Retrieve all messages for the current process.
424****************************************************************************/
425
426static BOOL retrieve_all_messages(char **msgs_buf, size_t *total_len)
427{
428 TDB_DATA kbuf;
429 TDB_DATA dbuf;
430 TDB_DATA null_dbuf;
431
432 ZERO_STRUCT(null_dbuf);
433
434 *msgs_buf = NULL;
435 *total_len = 0;
436
437 kbuf = message_key_pid(pid_to_procid(sys_getpid()));
438
439 if (tdb_chainlock(tdb, kbuf) == -1)
440 return False;
441
442 dbuf = tdb_fetch(tdb, kbuf);
443 /*
444 * Replace with an empty record to keep the allocated
445 * space in the tdb.
446 */
447 tdb_store(tdb, kbuf, null_dbuf, TDB_REPLACE);
448 tdb_chainunlock(tdb, kbuf);
449
450 if (dbuf.dptr == NULL || dbuf.dsize == 0) {
451 SAFE_FREE(dbuf.dptr);
452 return False;
453 }
454
455 *msgs_buf = dbuf.dptr;
456 *total_len = dbuf.dsize;
457
458 return True;
459}
460
461/****************************************************************************
462 Parse out the next message for the current process.
463****************************************************************************/
464
465static BOOL message_recv(char *msgs_buf, size_t total_len, int *msg_type,
466 struct process_id *src, char **buf, size_t *len)
467{
468 struct message_rec rec;
469 char *ret_buf = *buf;
470
471 *buf = NULL;
472 *len = 0;
473
474 if (total_len - (ret_buf - msgs_buf) < sizeof(rec))
475 return False;
476
477 memcpy(&rec, ret_buf, sizeof(rec));
478 ret_buf += sizeof(rec);
479
480 if (rec.msg_version != MESSAGE_VERSION) {
481 DEBUG(0,("message version %d received (expected %d)\n", rec.msg_version, MESSAGE_VERSION));
482 return False;
483 }
484
485 if (rec.len > 0) {
486 if (total_len - (ret_buf - msgs_buf) < rec.len)
487 return False;
488 }
489
490 *len = rec.len;
491 *msg_type = rec.msg_type;
492 *src = rec.src;
493 *buf = ret_buf;
494
495 return True;
496}
497
498/****************************************************************************
499 Receive and dispatch any messages pending for this process.
500 JRA changed Dec 13 2006. Only one message handler now permitted per type.
501 *NOTE*: Dispatch functions must be able to cope with incoming
502 messages on an *odd* byte boundary.
503****************************************************************************/
504
505void message_dispatch(void)
506{
507 int msg_type;
508 struct process_id src;
509 char *buf;
510 char *msgs_buf;
511 size_t len, total_len;
512 int n_handled;
513
514 if (!received_signal)
515 return;
516
517 DEBUG(10,("message_dispatch: received_signal = %d\n", received_signal));
518
519 received_signal = 0;
520
521 if (!retrieve_all_messages(&msgs_buf, &total_len))
522 return;
523
524 for (buf = msgs_buf; message_recv(msgs_buf, total_len, &msg_type, &src, &buf, &len); buf += len) {
525 struct dispatch_fns *dfn;
526
527 DEBUG(10,("message_dispatch: received msg_type=%d "
528 "src_pid=%u\n", msg_type,
529 (unsigned int) procid_to_pid(&src)));
530
531 n_handled = 0;
532 for (dfn = dispatch_fns; dfn; dfn = dfn->next) {
533 if (dfn->msg_type == msg_type) {
534 DEBUG(10,("message_dispatch: processing message of type %d.\n", msg_type));
535 dfn->fn(msg_type, src,
536 len ? (void *)buf : NULL, len,
537 dfn->private_data);
538 n_handled++;
539 break;
540 }
541 }
542 if (!n_handled) {
543 DEBUG(5,("message_dispatch: warning: no handler registed for "
544 "msg_type %d in pid %u\n",
545 msg_type, (unsigned int)sys_getpid()));
546 }
547 }
548 SAFE_FREE(msgs_buf);
549}
550
551/****************************************************************************
552 Register/replace a dispatch function for a particular message type.
553 JRA changed Dec 13 2006. Only one message handler now permitted per type.
554 *NOTE*: Dispatch functions must be able to cope with incoming
555 messages on an *odd* byte boundary.
556****************************************************************************/
557
558void message_register(int msg_type,
559 void (*fn)(int msg_type, struct process_id pid,
560 void *buf, size_t len,
561 void *private_data),
562 void *private_data)
563{
564 struct dispatch_fns *dfn;
565
566 for (dfn = dispatch_fns; dfn; dfn = dfn->next) {
567 if (dfn->msg_type == msg_type) {
568 dfn->fn = fn;
569 return;
570 }
571 }
572
573 dfn = SMB_MALLOC_P(struct dispatch_fns);
574
575 if (dfn != NULL) {
576
577 ZERO_STRUCTPN(dfn);
578
579 dfn->msg_type = msg_type;
580 dfn->fn = fn;
581 dfn->private_data = private_data;
582
583 DLIST_ADD(dispatch_fns, dfn);
584 }
585 else {
586
587 DEBUG(0,("message_register: Not enough memory. malloc failed!\n"));
588 }
589}
590
591/****************************************************************************
592 De-register the function for a particular message type.
593****************************************************************************/
594
595void message_deregister(int msg_type)
596{
597 struct dispatch_fns *dfn, *next;
598
599 for (dfn = dispatch_fns; dfn; dfn = next) {
600 next = dfn->next;
601 if (dfn->msg_type == msg_type) {
602 DLIST_REMOVE(dispatch_fns, dfn);
603 SAFE_FREE(dfn);
604 return;
605 }
606 }
607}
608
609struct msg_all {
610 int msg_type;
611 uint32 msg_flag;
612 const void *buf;
613 size_t len;
614 BOOL duplicates;
615 int n_sent;
616};
617
618/****************************************************************************
619 Send one of the messages for the broadcast.
620****************************************************************************/
621
622static int traverse_fn(TDB_CONTEXT *the_tdb, TDB_DATA kbuf, TDB_DATA dbuf, void *state)
623{
624 struct connections_data crec;
625 struct msg_all *msg_all = (struct msg_all *)state;
626 NTSTATUS status;
627
628 if (dbuf.dsize != sizeof(crec))
629 return 0;
630
631 memcpy(&crec, dbuf.dptr, sizeof(crec));
632
633 if (crec.cnum != -1)
634 return 0;
635
636 /* Don't send if the receiver hasn't registered an interest. */
637
638 if(!(crec.bcast_msg_flags & msg_all->msg_flag))
639 return 0;
640
641 /* If the msg send fails because the pid was not found (i.e. smbd died),
642 * the msg has already been deleted from the messages.tdb.*/
643
644 status = message_send_pid(crec.pid, msg_all->msg_type,
645 msg_all->buf, msg_all->len,
646 msg_all->duplicates);
647
648 if (NT_STATUS_EQUAL(status, NT_STATUS_INVALID_HANDLE)) {
649
650 /* If the pid was not found delete the entry from connections.tdb */
651
652 DEBUG(2,("pid %s doesn't exist - deleting connections %d [%s]\n",
653 procid_str_static(&crec.pid), crec.cnum, crec.servicename));
654 tdb_delete(the_tdb, kbuf);
655 }
656 msg_all->n_sent++;
657 return 0;
658}
659
660/**
661 * Send a message to all smbd processes.
662 *
663 * It isn't very efficient, but should be OK for the sorts of
664 * applications that use it. When we need efficient broadcast we can add
665 * it.
666 *
667 * @param n_sent Set to the number of messages sent. This should be
668 * equal to the number of processes, but be careful for races.
669 *
670 * @retval True for success.
671 **/
672BOOL message_send_all(TDB_CONTEXT *conn_tdb, int msg_type,
673 const void *buf, size_t len,
674 BOOL duplicates_allowed,
675 int *n_sent)
676{
677 struct msg_all msg_all;
678
679 msg_all.msg_type = msg_type;
680 if (msg_type < 1000)
681 msg_all.msg_flag = FLAG_MSG_GENERAL;
682 else if (msg_type > 1000 && msg_type < 2000)
683 msg_all.msg_flag = FLAG_MSG_NMBD;
684 else if (msg_type > 2000 && msg_type < 2100)
685 msg_all.msg_flag = FLAG_MSG_PRINT_NOTIFY;
686 else if (msg_type > 2100 && msg_type < 3000)
687 msg_all.msg_flag = FLAG_MSG_PRINT_GENERAL;
688 else if (msg_type > 3000 && msg_type < 4000)
689 msg_all.msg_flag = FLAG_MSG_SMBD;
690 else
691 return False;
692
693 msg_all.buf = buf;
694 msg_all.len = len;
695 msg_all.duplicates = duplicates_allowed;
696 msg_all.n_sent = 0;
697
698 tdb_traverse(conn_tdb, traverse_fn, &msg_all);
699 if (n_sent)
700 *n_sent = msg_all.n_sent;
701 return True;
702}
703
704/*
705 * Block and unblock receiving of messages. Allows removal of race conditions
706 * when doing a fork and changing message disposition.
707 */
708
709void message_block(void)
710{
711 BlockSignals(True, SIGUSR1);
712}
713
714void message_unblock(void)
715{
716 BlockSignals(False, SIGUSR1);
717}
718
719/*
720 * Samba4 API wrapper around the Samba3 implementation. Yes, I know, we could
721 * import the whole Samba4 thing, but I want notify.c from Samba4 in first.
722 */
723
724struct messaging_callback {
725 struct messaging_callback *prev, *next;
726 uint32 msg_type;
727 void (*fn)(struct messaging_context *msg, void *private_data,
728 uint32_t msg_type,
729 struct server_id server_id, DATA_BLOB *data);
730 void *private_data;
731};
732
733struct messaging_context {
734 struct server_id id;
735 struct messaging_callback *callbacks;
736};
737
738static int messaging_context_destructor(struct messaging_context *ctx)
739{
740 struct messaging_callback *cb;
741
742 for (cb = ctx->callbacks; cb; cb = cb->next) {
743 /*
744 * We unconditionally remove all instances of our callback
745 * from the tdb basis.
746 */
747 message_deregister(cb->msg_type);
748 }
749 return 0;
750}
751
752struct messaging_context *messaging_init(TALLOC_CTX *mem_ctx,
753 struct server_id server_id,
754 struct event_context *ev)
755{
756 struct messaging_context *ctx;
757
758 if (!(ctx = TALLOC_ZERO_P(mem_ctx, struct messaging_context))) {
759 return NULL;
760 }
761
762 ctx->id = server_id;
763 talloc_set_destructor(ctx, messaging_context_destructor);
764 return ctx;
765}
766
767static void messaging_callback(int msg_type, struct process_id pid,
768 void *buf, size_t len, void *private_data)
769{
770 struct messaging_context *ctx = talloc_get_type_abort(
771 private_data, struct messaging_context);
772 struct messaging_callback *cb, *next;
773
774 for (cb = ctx->callbacks; cb; cb = next) {
775 /*
776 * Allow a callback to remove itself
777 */
778 next = cb->next;
779
780 if (msg_type == cb->msg_type) {
781 DATA_BLOB blob;
782 struct server_id id;
783
784 blob.data = (uint8 *)buf;
785 blob.length = len;
786 id.id = pid;
787
788 cb->fn(ctx, cb->private_data, msg_type, id, &blob);
789 }
790 }
791}
792
793/*
794 * Register a dispatch function for a particular message type. Allow multiple
795 * registrants
796*/
797NTSTATUS messaging_register(struct messaging_context *ctx, void *private_data,
798 uint32_t msg_type,
799 void (*fn)(struct messaging_context *msg,
800 void *private_data,
801 uint32_t msg_type,
802 struct server_id server_id,
803 DATA_BLOB *data))
804{
805 struct messaging_callback *cb;
806
807 if (!(cb = talloc(ctx, struct messaging_callback))) {
808 return NT_STATUS_NO_MEMORY;
809 }
810
811 cb->msg_type = msg_type;
812 cb->fn = fn;
813 cb->private_data = private_data;
814
815 DLIST_ADD(ctx->callbacks, cb);
816 message_register(msg_type, messaging_callback, ctx);
817 return NT_STATUS_OK;
818}
819
820/*
821 De-register the function for a particular message type.
822*/
823void messaging_deregister(struct messaging_context *ctx, uint32_t msg_type,
824 void *private_data)
825{
826 struct messaging_callback *cb, *next;
827
828 for (cb = ctx->callbacks; cb; cb = next) {
829 next = cb->next;
830 if ((cb->msg_type == msg_type)
831 && (cb->private_data == private_data)) {
832 DLIST_REMOVE(ctx->callbacks, cb);
833 TALLOC_FREE(cb);
834 }
835 }
836}
837
838/*
839 Send a message to a particular server
840*/
841NTSTATUS messaging_send(struct messaging_context *msg,
842 struct server_id server,
843 uint32_t msg_type, DATA_BLOB *data)
844{
845 return message_send_pid_internal(server.id, msg_type, data->data,
846 data->length, True, 0);
847}
848
849/** @} **/
Note: See TracBrowser for help on using the repository browser.