ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/trunk/src/send.c
Revision: 3136
Committed: Tue Mar 11 18:24:03 2014 UTC (11 years, 5 months ago) by michael
Content type: text/x-csrc
File size: 27670 byte(s)
Log Message:
- Various fixes to previous ts5 removal changes

File Contents

# User Rev Content
1 adx 30 /*
2 michael 2916 * ircd-hybrid: an advanced, lightweight Internet Relay Chat Daemon (ircd)
3 adx 30 *
4 michael 2916 * Copyright (c) 1997-2014 ircd-hybrid development team
5 adx 30 *
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 2 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, write to the Free Software
18     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19     * USA
20     */
21    
22 michael 2916 /*! \file send.c
23     * \brief Functions for sending messages.
24     * \version $Id$
25     */
26    
27 adx 30 #include "stdinc.h"
28 michael 1011 #include "list.h"
29 adx 30 #include "send.h"
30     #include "channel.h"
31     #include "client.h"
32     #include "dbuf.h"
33     #include "irc_string.h"
34     #include "ircd.h"
35     #include "numeric.h"
36     #include "fdlist.h"
37     #include "s_bsd.h"
38     #include "s_serv.h"
39 michael 1309 #include "conf.h"
40     #include "log.h"
41 adx 30 #include "memory.h"
42     #include "packet.h"
43    
44    
45 michael 1078 static unsigned int current_serial = 0;
46 adx 30
47    
48     /* send_format()
49     *
50 michael 3120 * inputs
51     * - buffer
52 adx 30 * - format pattern to use
53     * - var args
54     * output - number of bytes formatted output
55     * side effects - modifies sendbuf
56     */
57 michael 3107 static void
58     send_format(struct dbuf_block *buffer, const char *pattern, va_list args)
59 adx 30 {
60     /*
61     * from rfc1459
62     *
63     * IRC messages are always lines of characters terminated with a CR-LF
64     * (Carriage Return - Line Feed) pair, and these messages shall not
65 michael 2638 * exceed 512 characters in length, counting all characters
66 adx 30 * including the trailing CR-LF.
67     * Thus, there are 510 characters maximum allowed
68     * for the command and its parameters. There is no provision for
69     * continuation message lines. See section 7 for more details about
70     * current implementations.
71     */
72 michael 3107 dbuf_put_args(buffer, pattern, args);
73 adx 30
74 michael 3107 if (buffer->size > sizeof(buffer->data) - 2)
75     buffer->size = sizeof(buffer->data) - 2;
76    
77     buffer->data[buffer->size++] = '\r';
78     buffer->data[buffer->size++] = '\n';
79 adx 30 }
80    
81     /*
82     ** send_message
83     ** Internal utility which appends given buffer to the sockets
84     ** sendq.
85     */
86     static void
87 michael 3107 send_message(struct Client *to, struct dbuf_block *buf)
88 adx 30 {
89 michael 439 assert(!IsMe(to));
90 michael 438 assert(to != &me);
91 adx 30
92 michael 3107 if (dbuf_length(&to->localClient->buf_sendq) + buf->size > get_sendq(&to->localClient->confs))
93 adx 30 {
94     if (IsServer(to))
95 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
96 michael 1761 "Max SendQ limit exceeded for %s: %lu > %u",
97 adx 30 get_client_name(to, HIDE_IP),
98 michael 3107 (unsigned long)(dbuf_length(&to->localClient->buf_sendq) + buf->size),
99 michael 1632 get_sendq(&to->localClient->confs));
100 adx 30 if (IsClient(to))
101     SetSendQExceeded(to);
102     dead_link_on_write(to, 0);
103     return;
104     }
105 michael 2638
106 michael 3107 dbuf_add(&to->localClient->buf_sendq, buf);
107 adx 30
108     /*
109     ** Update statistics. The following is slightly incorrect
110     ** because it counts messages even if queued, but bytes
111     ** only really sent. Queued bytes get updated in SendQueued.
112     */
113     ++to->localClient->send.messages;
114     ++me.localClient->send.messages;
115    
116 michael 2881 send_queued_write(to);
117 adx 30 }
118    
119     /* send_message_remote()
120     *
121     * inputs - pointer to client from message is being sent
122     * - pointer to client to send to
123 michael 3120 * - pointer to buffer
124 adx 30 * output - none
125     * side effects - Despite the function name, this only sends to directly
126     * connected clients.
127 michael 2916 *
128 adx 30 */
129     static void
130 michael 3107 send_message_remote(struct Client *to, struct Client *from, struct dbuf_block *buf)
131 adx 30 {
132 michael 3112 to = to->from;
133 michael 2793
134 adx 30 if (!MyConnect(to))
135     {
136 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
137 michael 2638 "Server send message to %s [%s] dropped from %s(Not local server)",
138     to->name, to->from->name, from->name);
139 adx 30 return;
140     }
141    
142     /* Optimize by checking if (from && to) before everything */
143     /* we set to->from up there.. */
144    
145     if (!MyClient(from) && IsClient(to) && (to == from->from))
146     {
147     if (IsServer(from))
148     {
149 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
150 adx 30 "Send message to %s [%s] dropped from %s(Fake Dir)",
151     to->name, to->from->name, from->name);
152     return;
153     }
154    
155 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
156 adx 30 "Ghosted: %s[%s@%s] from %s[%s@%s] (%s)",
157     to->name, to->username, to->host,
158     from->name, from->username, from->host,
159     to->from->name);
160    
161 michael 3135 sendto_server(NULL, NOCAPS, NOCAPS,
162 adx 30 ":%s KILL %s :%s (%s[%s@%s] Ghosted %s)",
163     me.id, to->name, me.name, to->name,
164     to->username, to->host, to->from->name);
165    
166 michael 1219 AddFlag(to, FLAGS_KILLED);
167 adx 30
168     if (IsClient(from))
169 michael 3109 sendto_one_numeric(from, &me, ERR_GHOSTEDCLIENT, to->name,
170     to->username, to->host, to->from);
171 adx 30
172     exit_client(to, &me, "Ghosted client");
173     return;
174 michael 2638 }
175 adx 30
176 michael 3107 send_message(to, buf);
177 adx 30 }
178    
179     /*
180     ** sendq_unblocked
181     ** Called when a socket is ready for writing.
182     */
183     void
184     sendq_unblocked(fde_t *fd, struct Client *client_p)
185     {
186 michael 2881 assert(fd == &client_p->localClient->fd);
187     send_queued_write(client_p);
188 adx 30 }
189    
190     /*
191     ** send_queued_write
192     ** This is called when there is a chance that some output would
193     ** be possible. This attempts to empty the send queue as far as
194     ** possible, and then if any data is left, a write is rescheduled.
195     */
196     void
197     send_queued_write(struct Client *to)
198     {
199 michael 2638 int retlen = 0;
200 adx 30
201     /*
202     ** Once socket is marked dead, we cannot start writing to it,
203     ** even if the error is removed...
204     */
205 michael 2881 if (IsDead(to))
206 adx 30 return; /* no use calling send() now */
207    
208     /* Next, lets try to write some data */
209 michael 3113 if (dbuf_length(&to->localClient->buf_sendq))
210 adx 30 {
211 michael 2638 do
212     {
213 michael 3113 struct dbuf_block *first = to->localClient->buf_sendq.blocks.head->data;
214 adx 30
215     #ifdef HAVE_LIBCRYPTO
216     if (to->localClient->fd.ssl)
217     {
218 michael 3113 retlen = SSL_write(to->localClient->fd.ssl, first->data + to->localClient->buf_sendq.pos, first->size - to->localClient->buf_sendq.pos);
219 adx 30
220     /* translate openssl error codes, sigh */
221 michael 2638 if (retlen < 0)
222     {
223     switch (SSL_get_error(to->localClient->fd.ssl, retlen))
224     {
225 adx 30 case SSL_ERROR_WANT_READ:
226 michael 2638 return; /* retry later, don't register for write events */
227     case SSL_ERROR_WANT_WRITE:
228     errno = EWOULDBLOCK;
229 adx 30 case SSL_ERROR_SYSCALL:
230 michael 2638 break;
231 michael 430 case SSL_ERROR_SSL:
232     if (errno == EAGAIN)
233     break;
234 adx 30 default:
235 michael 2638 retlen = errno = 0; /* either an SSL-specific error or EOF */
236     }
237     }
238 adx 30 }
239     else
240     #endif
241 michael 3113 retlen = send(to->localClient->fd.fd, first->data + to->localClient->buf_sendq.pos, first->size - to->localClient->buf_sendq.pos, 0);
242 adx 30
243     if (retlen <= 0)
244     break;
245    
246     dbuf_delete(&to->localClient->buf_sendq, retlen);
247    
248     /* We have some data written .. update counters */
249     to->localClient->send.bytes += retlen;
250     me.localClient->send.bytes += retlen;
251     } while (dbuf_length(&to->localClient->buf_sendq));
252    
253     if ((retlen < 0) && (ignoreErrno(errno)))
254     {
255     /* we have a non-fatal error, reschedule a write */
256 michael 3113 comm_setselect(&to->localClient->fd, COMM_SELECT_WRITE,
257     (PF *)sendq_unblocked, to, 0);
258 adx 30 }
259     else if (retlen <= 0)
260     {
261     dead_link_on_write(to, errno);
262     return;
263     }
264     }
265     }
266    
267     /* send_queued_all()
268     *
269     * input - NONE
270     * output - NONE
271     * side effects - try to flush sendq of each client
272     */
273     void
274     send_queued_all(void)
275     {
276     dlink_node *ptr;
277    
278     /* Servers are processed first, mainly because this can generate
279     * a notice to opers, which is to be delivered by this function.
280     */
281     DLINK_FOREACH(ptr, serv_list.head)
282 michael 2638 send_queued_write(ptr->data);
283 adx 30
284     DLINK_FOREACH(ptr, unknown_list.head)
285 michael 2638 send_queued_write(ptr->data);
286 adx 30
287     DLINK_FOREACH(ptr, local_client_list.head)
288 michael 2638 send_queued_write(ptr->data);
289 adx 30
290     /* NOTE: This can still put clients on aborted_list; unfortunately,
291     * exit_aborted_clients takes precedence over send_queued_all,
292     * because exiting clients can generate server/oper traffic.
293     * The easiest approach is dealing with aborted clients in the next I/O lap.
294     * -adx
295     */
296     }
297    
298     /* sendto_one()
299     *
300     * inputs - pointer to destination client
301     * - var args message
302     * output - NONE
303     * side effects - send message to single client
304     */
305     void
306     sendto_one(struct Client *to, const char *pattern, ...)
307     {
308     va_list args;
309 michael 3107 struct dbuf_block *buffer;
310 adx 30
311 michael 3112 to = to->from;
312    
313 adx 30 if (IsDead(to))
314 michael 3107 return; /* This socket has already been marked as dead */
315 adx 30
316 michael 3107 buffer = dbuf_alloc();
317    
318 adx 30 va_start(args, pattern);
319 michael 3107 send_format(buffer, pattern, args);
320 adx 30 va_end(args);
321    
322 michael 3107 send_message(to, buffer);
323    
324     dbuf_ref_free(buffer);
325 adx 30 }
326    
327 michael 3109 void
328     sendto_one_numeric(struct Client *to, struct Client *from, enum irc_numerics numeric, ...)
329     {
330     struct dbuf_block *buffer;
331     va_list args;
332     const char *dest;
333    
334 michael 3112 to = to->from;
335    
336 michael 3109 if (IsDead(to))
337     return;
338    
339     dest = ID_or_name(to, to);
340     if (EmptyString(dest))
341     dest = "*";
342    
343     buffer = dbuf_alloc();
344    
345 michael 3113 dbuf_put_fmt(buffer, ":%s %03d %s ", ID_or_name(from, to), numeric, dest);
346 michael 3109
347     va_start(args, numeric);
348     send_format(buffer, numeric_form(numeric), args);
349     va_end(args);
350    
351     send_message(to, buffer);
352    
353     dbuf_ref_free(buffer);
354     }
355    
356 michael 3110 void
357     sendto_one_notice(struct Client *to, struct Client *from, const char *pattern, ...)
358     {
359     struct dbuf_block *buffer;
360     va_list args;
361     const char *dest;
362    
363 michael 3112 to = to->from;
364    
365 michael 3110 if (IsDead(to))
366     return;
367    
368     dest = ID_or_name(to, to);
369     if (EmptyString(dest))
370     dest = "*";
371    
372     buffer = dbuf_alloc();
373    
374 michael 3113 dbuf_put_fmt(buffer, ":%s NOTICE %s ", ID_or_name(from, to), dest);
375 michael 3110
376     va_start(args, pattern);
377     send_format(buffer, pattern, args);
378     va_end(args);
379    
380     send_message(to, buffer);
381    
382     dbuf_ref_free(buffer);
383     }
384    
385 adx 30 /* sendto_channel_butone()
386     *
387     * inputs - pointer to client(server) to NOT send message to
388     * - pointer to client that is sending this message
389     * - pointer to channel being sent to
390     * - vargs message
391     * output - NONE
392     * side effects - message as given is sent to given channel members.
393     *
394     * WARNING - +D clients are ignored
395     */
396     void
397     sendto_channel_butone(struct Client *one, struct Client *from,
398 michael 1479 struct Channel *chptr, unsigned int type,
399 adx 30 const char *pattern, ...)
400     {
401 michael 3136 va_list alocal, aremote;
402     struct dbuf_block *local_buf, *remote_buf;
403 michael 1078 dlink_node *ptr = NULL, *ptr_next = NULL;
404 adx 30
405 michael 3136 local_buf = dbuf_alloc(), remote_buf = dbuf_alloc();
406 michael 3107
407 adx 30 if (IsServer(from))
408 michael 3113 dbuf_put_fmt(local_buf, ":%s ", from->name);
409 adx 30 else
410 michael 3113 dbuf_put_fmt(local_buf, ":%s!%s@%s ", from->name, from->username, from->host);
411 adx 30
412 michael 3136 dbuf_put_fmt(remote_buf, ":%s ", ID(from));
413 michael 3107
414 adx 285 va_start(alocal, pattern);
415     va_start(aremote, pattern);
416 michael 3107 send_format(local_buf, pattern, alocal);
417     send_format(remote_buf, pattern, aremote);
418 michael 3136
419 adx 285 va_end(aremote);
420     va_end(alocal);
421 adx 30
422     ++current_serial;
423    
424     DLINK_FOREACH_SAFE(ptr, ptr_next, chptr->members.head)
425     {
426 michael 1479 struct Membership *ms = ptr->data;
427     struct Client *target_p = ms->client_p;
428 adx 30
429 michael 1078 assert(IsClient(target_p));
430    
431 michael 1219 if (IsDefunct(target_p) || HasUMode(target_p, UMODE_DEAF) || target_p->from == one)
432 adx 30 continue;
433    
434 michael 1479 if (type != 0 && (ms->flags & type) == 0)
435     continue;
436    
437 michael 1078 if (MyConnect(target_p))
438 michael 3103 send_message(target_p, local_buf);
439     else if (target_p->from->localClient->serial != current_serial)
440     send_message_remote(target_p->from, from, remote_buf);
441     target_p->from->localClient->serial = current_serial;
442 adx 30 }
443 michael 3107
444     dbuf_ref_free(local_buf);
445     dbuf_ref_free(remote_buf);
446 adx 30 }
447    
448     /* sendto_server()
449 michael 2916 *
450 adx 30 * inputs - pointer to client to NOT send to
451 db 939 * - pointer to channel
452 adx 30 * - caps or'd together which must ALL be present
453     * - caps or'd together which must ALL NOT be present
454     * - printf style format string
455     * - args to format string
456     * output - NONE
457     * side effects - Send a message to all connected servers, except the
458     * client 'one' (if non-NULL), as long as the servers
459     * support ALL capabs in 'caps', and NO capabs in 'nocaps'.
460 michael 2916 *
461 adx 30 * This function was written in an attempt to merge together the other
462     * billion sendto_*serv*() functions, which sprung up with capabs,
463     * lazylinks, uids, etc.
464     * -davidt
465     */
466 michael 2638 void
467 michael 1474 sendto_server(struct Client *one,
468 michael 1015 const unsigned int caps,
469     const unsigned int nocaps,
470 adx 30 const char *format, ...)
471     {
472     va_list args;
473 michael 885 dlink_node *ptr = NULL;
474 michael 3107 struct dbuf_block *buffer;
475 adx 30
476 michael 3107 buffer = dbuf_alloc();
477    
478 adx 30 va_start(args, format);
479 michael 3107 send_format(buffer, format, args);
480 adx 30 va_end(args);
481    
482     DLINK_FOREACH(ptr, serv_list.head)
483     {
484 michael 885 struct Client *client_p = ptr->data;
485 adx 30
486     /* If dead already skip */
487     if (IsDead(client_p))
488     continue;
489     /* check against 'one' */
490     if (one != NULL && (client_p == one->from))
491     continue;
492     /* check we have required capabs */
493     if ((client_p->localClient->caps & caps) != caps)
494     continue;
495     /* check we don't have any forbidden capabs */
496     if ((client_p->localClient->caps & nocaps) != 0)
497     continue;
498    
499 michael 3107 send_message(client_p, buffer);
500 adx 30 }
501 michael 3107
502     dbuf_ref_free(buffer);
503 adx 30 }
504    
505     /* sendto_common_channels_local()
506     *
507     * inputs - pointer to client
508     * - pattern to send
509     * output - NONE
510     * side effects - Sends a message to all people on local server who are
511 michael 2916 * in same channel with user.
512 adx 30 * used by m_nick.c and exit_one_client.
513     */
514     void
515 michael 1734 sendto_common_channels_local(struct Client *user, int touser, unsigned int cap,
516 adx 30 const char *pattern, ...)
517     {
518     va_list args;
519     dlink_node *uptr;
520     dlink_node *cptr;
521     struct Channel *chptr;
522     struct Membership *ms;
523     struct Client *target_p;
524 michael 3107 struct dbuf_block *buffer;
525 adx 30
526 michael 3107 buffer = dbuf_alloc();
527    
528 adx 30 va_start(args, pattern);
529 michael 3107 send_format(buffer, pattern, args);
530 adx 30 va_end(args);
531    
532     ++current_serial;
533    
534     DLINK_FOREACH(cptr, user->channel.head)
535     {
536 michael 2638 chptr = ((struct Membership *)cptr->data)->chptr;
537 adx 30 assert(chptr != NULL);
538    
539 michael 572 DLINK_FOREACH(uptr, chptr->members.head)
540 adx 30 {
541     ms = uptr->data;
542     target_p = ms->client_p;
543     assert(target_p != NULL);
544    
545 michael 572 if (!MyConnect(target_p) || target_p == user || IsDefunct(target_p) ||
546 michael 1078 target_p->localClient->serial == current_serial)
547 adx 30 continue;
548    
549 michael 1734 if (HasCap(target_p, cap) != cap)
550     continue;
551    
552 michael 1078 target_p->localClient->serial = current_serial;
553 michael 3107 send_message(target_p, buffer);
554 adx 30 }
555     }
556    
557     if (touser && MyConnect(user) && !IsDead(user) &&
558 michael 1078 user->localClient->serial != current_serial)
559 michael 1844 if (HasCap(user, cap) == cap)
560 michael 3107 send_message(user, buffer);
561    
562     dbuf_ref_free(buffer);
563 adx 30 }
564    
565     /* sendto_channel_local()
566     *
567     * inputs - member status mask, e.g. CHFL_CHANOP | CHFL_VOICE
568     * - whether to ignore +D clients (YES/NO)
569     * - pointer to channel to send to
570     * - var args pattern
571     * output - NONE
572     * side effects - Send a message to all members of a channel that are
573     * locally connected to this server.
574     */
575     void
576 michael 2543 sendto_channel_local(unsigned int type, int nodeaf, struct Channel *chptr,
577 adx 30 const char *pattern, ...)
578     {
579     va_list args;
580 michael 2638 dlink_node *ptr = NULL;
581 michael 3107 struct dbuf_block *buffer;
582 adx 30
583 michael 3107 buffer = dbuf_alloc();
584    
585 adx 30 va_start(args, pattern);
586 michael 3107 send_format(buffer, pattern, args);
587 adx 30 va_end(args);
588    
589 michael 572 DLINK_FOREACH(ptr, chptr->members.head)
590 adx 30 {
591 michael 2638 struct Membership *ms = ptr->data;
592     struct Client *target_p = ms->client_p;
593 adx 30
594     if (type != 0 && (ms->flags & type) == 0)
595     continue;
596    
597 michael 572 if (!MyConnect(target_p) || IsDefunct(target_p) ||
598 michael 1219 (nodeaf && HasUMode(target_p, UMODE_DEAF)))
599 adx 30 continue;
600    
601 michael 3107 send_message(target_p, buffer);
602 adx 30 }
603 michael 3107
604     dbuf_ref_free(buffer);
605 adx 30 }
606    
607     /* sendto_channel_local_butone()
608     *
609     * inputs - pointer to client to NOT send message to
610     * - member status mask, e.g. CHFL_CHANOP | CHFL_VOICE
611     * - pointer to channel to send to
612     * - var args pattern
613     * output - NONE
614     * side effects - Send a message to all members of a channel that are
615     * locally connected to this server except one.
616     *
617     * WARNING - +D clients are omitted
618     */
619 michael 2638 void
620 michael 2543 sendto_channel_local_butone(struct Client *one, unsigned int type, unsigned int cap,
621 michael 2638 struct Channel *chptr, const char *pattern, ...)
622 adx 30 {
623     va_list args;
624 michael 2638 dlink_node *ptr = NULL;
625 michael 3107 struct dbuf_block *buffer;
626 adx 30
627 michael 3107 buffer = dbuf_alloc();
628    
629 michael 2638 va_start(args, pattern);
630 michael 3107 send_format(buffer, pattern, args);
631 adx 30 va_end(args);
632    
633 michael 2638 DLINK_FOREACH(ptr, chptr->members.head)
634     {
635     struct Membership *ms = ptr->data;
636     struct Client *target_p = ms->client_p;
637 adx 30
638     if (type != 0 && (ms->flags & type) == 0)
639     continue;
640    
641 michael 572 if (!MyConnect(target_p) || target_p == one ||
642 michael 1219 IsDefunct(target_p) || HasUMode(target_p, UMODE_DEAF))
643 adx 30 continue;
644 michael 1734
645     if (HasCap(target_p, cap) != cap)
646 michael 2638 continue;
647    
648 michael 3107 send_message(target_p, buffer);
649 adx 30 }
650 michael 3107
651     dbuf_ref_free(buffer);
652 adx 30 }
653    
654     /*
655     ** match_it() and sendto_match_butone() ARE only used
656     ** to send a msg to all ppl on servers/hosts that match a specified mask
657     ** (used for enhanced PRIVMSGs) for opers
658     **
659     ** addition -- Armin, 8jun90 (gruner@informatik.tu-muenchen.de)
660     **
661     */
662    
663     /* match_it()
664     *
665     * inputs - client pointer to match on
666     * - actual mask to match
667     * - what to match on, HOST or SERVER
668     * output - 1 or 0 if match or not
669     * side effects - NONE
670     */
671     static int
672 michael 2867 match_it(const struct Client *one, const char *mask, unsigned int what)
673 adx 30 {
674     if (what == MATCH_HOST)
675 michael 1652 return !match(mask, one->host);
676 adx 30
677 michael 1652 return !match(mask, one->servptr->name);
678 adx 30 }
679    
680     /* sendto_match_butone()
681     *
682     * Send to all clients which match the mask in a way defined on 'what';
683     * either by user hostname or user servername.
684     *
685     * ugh. ONLY used by m_message.c to send an "oper magic" message. ugh.
686     */
687     void
688     sendto_match_butone(struct Client *one, struct Client *from, char *mask,
689     int what, const char *pattern, ...)
690     {
691 adx 285 va_list alocal, aremote;
692 michael 3107 dlink_node *ptr = NULL, *ptr_next = NULL;
693     struct dbuf_block *local_buf, *remote_buf;
694 adx 30
695 michael 3107 local_buf = dbuf_alloc(), remote_buf = dbuf_alloc();
696    
697 michael 3113 dbuf_put_fmt(local_buf, ":%s!%s@%s ", from->name, from->username, from->host);
698 michael 3136 dbuf_put_fmt(remote_buf, ":%s ", ID(from));
699 michael 3107
700 adx 285 va_start(alocal, pattern);
701     va_start(aremote, pattern);
702 michael 3107 send_format(local_buf, pattern, alocal);
703     send_format(remote_buf, pattern, aremote);
704 adx 285 va_end(aremote);
705     va_end(alocal);
706 adx 30
707     /* scan the local clients */
708     DLINK_FOREACH(ptr, local_client_list.head)
709     {
710 michael 3107 struct Client *client_p = ptr->data;
711 adx 30
712     if (client_p != one && !IsDefunct(client_p) &&
713     match_it(client_p, mask, what))
714 michael 3107 send_message(client_p, local_buf);
715 adx 30 }
716    
717     /* Now scan servers */
718     DLINK_FOREACH_SAFE(ptr, ptr_next, serv_list.head)
719     {
720 michael 3107 struct Client *client_p = ptr->data;
721 adx 30
722     /*
723     * The old code looped through every client on the
724     * network for each server to check if the
725     * server (client_p) has at least 1 client matching
726     * the mask, using something like:
727     *
728     * for (target_p = GlobalClientList; target_p; target_p = target_p->next)
729     * if (IsRegisteredUser(target_p) &&
730     * match_it(target_p, mask, what) &&
731     * (target_p->from == client_p))
732     * vsendto_prefix_one(client_p, from, pattern, args);
733     *
734     * That way, we wouldn't send the message to
735     * a server who didn't have a matching client.
736     * However, on a network such as EFNet, that
737     * code would have looped through about 50
738     * servers, and in each loop, loop through
739     * about 50k clients as well, calling match()
740     * in each nested loop. That is a very bad
741     * thing cpu wise - just send the message
742     * to every connected server and let that
743     * server deal with it.
744     * -wnder
745     */
746     if (client_p != one && !IsDefunct(client_p))
747 michael 3107 send_message_remote(client_p, from, remote_buf);
748 adx 30 }
749 michael 3107
750     dbuf_ref_free(local_buf);
751     dbuf_ref_free(remote_buf);
752 adx 30 }
753    
754     /* sendto_match_servs()
755     *
756     * inputs - source client
757     * - mask to send to
758     * - capab needed
759     * - data
760     * outputs - none
761     * side effects - data sent to servers matching with capab
762     */
763     void
764 michael 2543 sendto_match_servs(struct Client *source_p, const char *mask, unsigned int cap,
765 adx 30 const char *pattern, ...)
766     {
767     va_list args;
768 michael 2638 dlink_node *ptr = NULL;
769 michael 3135 struct dbuf_block *buff_suid;
770 adx 30
771 michael 3135 buff_suid = dbuf_alloc();
772 michael 3107
773 adx 30 va_start(args, pattern);
774 michael 3113 dbuf_put_fmt(buff_suid, ":%s ", ID(source_p));
775 michael 3107 dbuf_put_args(buff_suid, pattern, args);
776 adx 30 va_end(args);
777    
778 michael 948 ++current_serial;
779 adx 30
780     DLINK_FOREACH(ptr, global_serv_list.head)
781     {
782 michael 2638 struct Client *target_p = ptr->data;
783 adx 30
784     /* Do not attempt to send to ourselves, or the source */
785     if (IsMe(target_p) || target_p->from == source_p->from)
786     continue;
787    
788 michael 1078 if (target_p->from->localClient->serial == current_serial)
789 adx 30 continue;
790    
791 michael 1652 if (!match(mask, target_p->name))
792 adx 30 {
793     /*
794     * if we set the serial here, then we'll never do a
795     * match() again, if !IsCapable()
796     */
797 michael 1078 target_p->from->localClient->serial = current_serial;
798 adx 30
799     if (!IsCapable(target_p->from, cap))
800     continue;
801    
802 michael 3135 send_message_remote(target_p->from, source_p, buff_suid);
803 adx 30 }
804     }
805 michael 3107
806     dbuf_ref_free(buff_suid);
807 adx 30 }
808    
809     /* sendto_anywhere()
810     *
811     * inputs - pointer to dest client
812     * - pointer to from client
813     * - varags
814     * output - NONE
815     * side effects - less efficient than sendto_remote and sendto_one
816     * but useful when one does not know where target "lives"
817     */
818     void
819     sendto_anywhere(struct Client *to, struct Client *from,
820 michael 2793 const char *command,
821 adx 30 const char *pattern, ...)
822     {
823     va_list args;
824 michael 3107 struct dbuf_block *buffer;
825 adx 30
826 michael 2793 if (IsDead(to->from))
827 adx 30 return;
828    
829 michael 3107 buffer = dbuf_alloc();
830    
831 adx 30 if (MyClient(to))
832     {
833     if (IsServer(from))
834 michael 3113 dbuf_put_fmt(buffer, ":%s %s %s ", from->name, command, to->name);
835 adx 30 else
836 michael 3113 dbuf_put_fmt(buffer, ":%s!%s@%s %s %s ", from->name, from->username, from->host, command, to->name);
837 adx 30 }
838 michael 2634 else
839 michael 3113 dbuf_put_fmt(buffer, ":%s %s %s ", ID_or_name(from, to), command, ID_or_name(to, to));
840 adx 30
841     va_start(args, pattern);
842 michael 3107 send_format(buffer, pattern, args);
843 adx 30 va_end(args);
844    
845 michael 2518 if (MyClient(to))
846 michael 3107 send_message(to, buffer);
847 adx 30 else
848 michael 3107 send_message_remote(to, from, buffer);
849    
850     dbuf_ref_free(buffer);
851 adx 30 }
852    
853     /* sendto_realops_flags()
854     *
855     * inputs - flag types of messages to show to real opers
856     * - flag indicating opers/admins
857     * - var args input message
858     * output - NONE
859     * side effects - Send to *local* ops only but NOT +s nonopers.
860     */
861     void
862 michael 1618 sendto_realops_flags(unsigned int flags, int level, int type, const char *pattern, ...)
863 adx 30 {
864 michael 1618 const char *ntype = NULL;
865 michael 1078 dlink_node *ptr = NULL;
866 adx 30 char nbuf[IRCD_BUFSIZE];
867     va_list args;
868    
869     va_start(args, pattern);
870 michael 1618 vsnprintf(nbuf, sizeof(nbuf), pattern, args);
871 adx 30 va_end(args);
872    
873 michael 1618 switch (type)
874 adx 30 {
875 michael 1618 case SEND_NOTICE:
876     ntype = "Notice";
877     break;
878     case SEND_GLOBAL:
879     ntype = "Global";
880     break;
881     case SEND_LOCOPS:
882     ntype = "LocOps";
883     break;
884     default:
885     assert(0);
886 adx 30 }
887    
888 michael 1206 DLINK_FOREACH(ptr, oper_list.head)
889     {
890     struct Client *client_p = ptr->data;
891 michael 1618 assert(HasUMode(client_p, UMODE_OPER));
892 michael 1206
893 michael 1618 /*
894     * If we're sending it to opers and they're an admin, skip.
895     * If we're sending it to admins, and they're not, skip.
896 michael 1206 */
897 michael 1219 if (((level == L_ADMIN) && !HasUMode(client_p, UMODE_ADMIN)) ||
898 michael 2638 ((level == L_OPER) && HasUMode(client_p, UMODE_ADMIN)))
899 michael 1206 continue;
900    
901 michael 1219 if (HasUMode(client_p, flags))
902 michael 1618 sendto_one(client_p, ":%s NOTICE %s :*** %s -- %s",
903     me.name, client_p->name, ntype, nbuf);
904 michael 1206 }
905     }
906    
907 adx 30 /* sendto_wallops_flags()
908     *
909     * inputs - flag types of messages to show to real opers
910     * - client sending request
911     * - var args input message
912     * output - NONE
913     * side effects - Send a wallops to local opers
914     */
915     void
916     sendto_wallops_flags(unsigned int flags, struct Client *source_p,
917     const char *pattern, ...)
918     {
919 michael 1078 dlink_node *ptr = NULL;
920 adx 30 va_list args;
921 michael 3107 struct dbuf_block *buffer;
922 adx 30
923 michael 3107 buffer = dbuf_alloc();
924    
925 adx 30 if (IsClient(source_p))
926 michael 3113 dbuf_put_fmt(buffer, ":%s!%s@%s WALLOPS :", source_p->name, source_p->username, source_p->host);
927 adx 30 else
928 michael 3113 dbuf_put_fmt(buffer, ":%s WALLOPS :", source_p->name);
929 adx 30
930     va_start(args, pattern);
931 michael 3107 send_format(buffer, pattern, args);
932 adx 30 va_end(args);
933    
934     DLINK_FOREACH(ptr, oper_list.head)
935     {
936 michael 1078 struct Client *client_p = ptr->data;
937 adx 30 assert(client_p->umodes & UMODE_OPER);
938    
939 michael 1219 if (HasUMode(client_p, flags) && !IsDefunct(client_p))
940 michael 3107 send_message(client_p, buffer);
941 adx 30 }
942 michael 3107
943     dbuf_ref_free(buffer);
944 adx 30 }
945    
946     /* ts_warn()
947     *
948     * inputs - var args message
949     * output - NONE
950     * side effects - Call sendto_realops_flags, with some flood checking
951     * (at most 5 warnings every 5 seconds)
952     */
953     void
954 michael 3078 sendto_realops_flags_ratelimited(const char *pattern, ...)
955 adx 30 {
956     va_list args;
957 michael 1078 char buffer[IRCD_BUFSIZE];
958 adx 30 static time_t last = 0;
959     static int warnings = 0;
960    
961     /*
962     ** if we're running with TS_WARNINGS enabled and someone does
963     ** something silly like (remotely) connecting a nonTS server,
964     ** we'll get a ton of warnings, so we make sure we don't send
965     ** more than 5 every 5 seconds. -orabidoo
966     */
967    
968     if (CurrentTime - last < 5)
969     {
970     if (++warnings > 5)
971     return;
972     }
973     else
974     {
975     last = CurrentTime;
976     warnings = 0;
977     }
978    
979     va_start(args, pattern);
980 michael 1086 vsnprintf(buffer, sizeof(buffer), pattern, args);
981 adx 30 va_end(args);
982    
983 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE, "%s", buffer);
984 michael 1247 ilog(LOG_TYPE_IRCD, "%s", buffer);
985 adx 30 }
986    
987     /* kill_client()
988     *
989     * inputs - client to send kill towards
990     * - pointer to client to kill
991     * - reason for kill
992     * output - NONE
993     * side effects - NONE
994     */
995     void
996     kill_client(struct Client *client_p, struct Client *diedie,
997     const char *pattern, ...)
998     {
999     va_list args;
1000 michael 3107 struct dbuf_block *buffer;
1001 adx 30
1002 michael 3120 client_p = client_p->from;
1003    
1004 adx 30 if (IsDead(client_p))
1005     return;
1006    
1007 michael 3107 buffer = dbuf_alloc();
1008 adx 30
1009 michael 3113 dbuf_put_fmt(buffer, ":%s KILL %s :",
1010     ID_or_name(&me, client_p),
1011     ID_or_name(diedie, client_p));
1012 michael 3107
1013 adx 30 va_start(args, pattern);
1014 michael 3107 send_format(buffer, pattern, args);
1015 adx 30 va_end(args);
1016    
1017 michael 3107 send_message(client_p, buffer);
1018    
1019     dbuf_ref_free(buffer);
1020 adx 30 }
1021    
1022 michael 3136 /* kill_client_serv_butone()
1023 adx 30 *
1024     * inputs - pointer to client to not send to
1025     * - pointer to client to kill
1026     * output - NONE
1027     * side effects - Send a KILL for the given client
1028     * message to all connected servers
1029     * except the client 'one'. Also deal with
1030     * client being unknown to leaf, as in lazylink...
1031     */
1032     void
1033 michael 2565 kill_client_serv_butone(struct Client *one, struct Client *source_p,
1034     const char *pattern, ...)
1035 adx 30 {
1036     va_list args;
1037 michael 885 dlink_node *ptr = NULL;
1038 michael 3135 struct dbuf_block *uid_buffer = dbuf_alloc();
1039 adx 30
1040 adx 285 va_start(args, pattern);
1041 michael 3135 dbuf_put_fmt(uid_buffer, ":%s KILL %s :", ID(&me), ID(source_p));
1042     send_format(uid_buffer, pattern, args);
1043 adx 30 va_end(args);
1044    
1045     DLINK_FOREACH(ptr, serv_list.head)
1046     {
1047 michael 885 struct Client *client_p = ptr->data;
1048 adx 30
1049     if (one != NULL && (client_p == one->from))
1050     continue;
1051     if (IsDefunct(client_p))
1052     continue;
1053    
1054 michael 3135 send_message(client_p, uid_buffer);
1055 adx 30 }
1056 michael 3107
1057     dbuf_ref_free(uid_buffer);
1058 michael 2638 }

Properties

Name Value
svn:eol-style native
svn:keywords Id Revision