ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/branches/8.2.x/src/channel.c
Revision: 4208
Committed: Sat Jul 12 18:13:06 2014 UTC (9 years, 9 months ago) by michael
Content type: text/x-csrc
File size: 33184 byte(s)
Log Message:
- Renammed global_serv_list to global_server_list

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     /*! \file channel.c
23     * \brief Responsible for managing channels, members, bans and topics
24 knight 31 * \version $Id$
25 adx 30 */
26    
27     #include "stdinc.h"
28 michael 1011 #include "list.h"
29 adx 30 #include "channel.h"
30     #include "channel_mode.h"
31     #include "client.h"
32     #include "hash.h"
33 michael 1632 #include "conf.h"
34 michael 371 #include "hostmask.h"
35 adx 30 #include "irc_string.h"
36     #include "ircd.h"
37     #include "numeric.h"
38 michael 3347 #include "server.h"
39 adx 30 #include "send.h"
40     #include "event.h"
41     #include "memory.h"
42 michael 1654 #include "mempool.h"
43 michael 3347 #include "misc.h"
44 michael 1826 #include "resv.h"
45 adx 30
46 michael 3235
47 michael 3945 dlink_list channel_list;
48 michael 1654 mp_pool_t *ban_pool; /*! \todo ban_pool shouldn't be a global var */
49 adx 30
50 michael 4095 struct event splitmode_event =
51     {
52     .name = "check_splitmode",
53     .handler = check_splitmode,
54 michael 4132 .when = 5
55 michael 4095 };
56    
57 michael 3250 static mp_pool_t *member_pool, *channel_pool;
58 adx 30 static char buf[IRCD_BUFSIZE];
59    
60    
61     /*! \brief Initializes the channel blockheap, adds known channel CAPAB
62     */
63     void
64 michael 1798 channel_init(void)
65 adx 30 {
66     add_capability("EX", CAP_EX, 1);
67     add_capability("IE", CAP_IE, 1);
68    
69 michael 1654 channel_pool = mp_pool_new(sizeof(struct Channel), MP_CHUNK_SIZE_CHANNEL);
70     ban_pool = mp_pool_new(sizeof(struct Ban), MP_CHUNK_SIZE_BAN);
71     member_pool = mp_pool_new(sizeof(struct Membership), MP_CHUNK_SIZE_MEMBER);
72 adx 30 }
73    
74 michael 3308 /*! \brief Adds a user to a channel by adding another link to the
75 adx 30 * channels member chain.
76 michael 3308 * \param chptr Pointer to channel to add client to
77     * \param who Pointer to client (who) to add
78     * \param flags Flags for chanops etc
79     * \param flood_ctrl Whether to count this join in flood calculations
80 adx 30 */
81     void
82     add_user_to_channel(struct Channel *chptr, struct Client *who,
83     unsigned int flags, int flood_ctrl)
84     {
85     struct Membership *ms = NULL;
86    
87     if (GlobalSetOptions.joinfloodtime > 0)
88     {
89     if (flood_ctrl)
90 michael 3250 ++chptr->number_joined;
91 adx 30
92     chptr->number_joined -= (CurrentTime - chptr->last_join_time) *
93     (((float)GlobalSetOptions.joinfloodcount) /
94     (float)GlobalSetOptions.joinfloodtime);
95    
96     if (chptr->number_joined <= 0)
97     {
98     chptr->number_joined = 0;
99     ClearJoinFloodNoticed(chptr);
100     }
101     else if (chptr->number_joined >= GlobalSetOptions.joinfloodcount)
102     {
103     chptr->number_joined = GlobalSetOptions.joinfloodcount;
104    
105     if (!IsSetJoinFloodNoticed(chptr))
106     {
107     SetJoinFloodNoticed(chptr);
108 michael 1618 sendto_realops_flags(UMODE_BOTS, L_ALL, SEND_NOTICE,
109 adx 30 "Possible Join Flooder %s on %s target: %s",
110     get_client_name(who, HIDE_IP),
111     who->servptr->name, chptr->chname);
112     }
113     }
114    
115     chptr->last_join_time = CurrentTime;
116     }
117    
118 michael 1654 ms = mp_pool_get(member_pool);
119 adx 30 ms->client_p = who;
120     ms->chptr = chptr;
121     ms->flags = flags;
122    
123     dlinkAdd(ms, &ms->channode, &chptr->members);
124 michael 4171
125     if (MyConnect(who))
126     dlinkAdd(ms, &ms->locchannode, &chptr->locmembers);
127    
128 adx 30 dlinkAdd(ms, &ms->usernode, &who->channel);
129     }
130    
131 michael 3308 /*! \brief Deletes an user from a channel by removing a link in the
132 adx 30 * channels member chain.
133 michael 3308 * \param member Pointer to Membership struct
134 adx 30 */
135     void
136     remove_user_from_channel(struct Membership *member)
137     {
138     struct Client *client_p = member->client_p;
139     struct Channel *chptr = member->chptr;
140    
141     dlinkDelete(&member->channode, &chptr->members);
142 michael 4171
143     if (MyConnect(client_p))
144     dlinkDelete(&member->locchannode, &chptr->locmembers);
145    
146 adx 30 dlinkDelete(&member->usernode, &client_p->channel);
147    
148 michael 1654 mp_pool_release(member);
149 adx 30
150 michael 1011 if (chptr->members.head == NULL)
151 adx 30 destroy_channel(chptr);
152     }
153    
154     /* send_members()
155     *
156     * inputs -
157     * output - NONE
158     * side effects -
159     */
160     static void
161     send_members(struct Client *client_p, struct Channel *chptr,
162 michael 3145 char *modebuf, char *parabuf)
163 adx 30 {
164 michael 1847 const dlink_node *ptr = NULL;
165 adx 30 int tlen; /* length of text to append */
166     char *t, *start; /* temp char pointer */
167    
168 michael 1847 start = t = buf + snprintf(buf, sizeof(buf), ":%s SJOIN %lu %s %s %s:",
169 michael 3183 me.id, (unsigned long)chptr->channelts,
170 michael 3145 chptr->chname, modebuf, parabuf);
171 adx 30
172     DLINK_FOREACH(ptr, chptr->members.head)
173     {
174 michael 1847 const struct Membership *ms = ptr->data;
175 adx 30
176 michael 4045 tlen = strlen(ms->client_p->id) + 1; /* +1 for space */
177 adx 30
178     if (ms->flags & CHFL_CHANOP)
179 michael 3250 ++tlen;
180 michael 3140 if (ms->flags & CHFL_HALFOP)
181 michael 3250 ++tlen;
182 adx 30 if (ms->flags & CHFL_VOICE)
183 michael 3250 ++tlen;
184 adx 30
185 michael 3250 /*
186     * Space will be converted into CR, but we also need space for LF..
187 michael 3783 * That's why we use '- 1' here -adx
188 michael 3250 */
189 michael 1330 if (t + tlen - buf > IRCD_BUFSIZE - 1)
190 adx 30 {
191 michael 3250 *(t - 1) = '\0'; /* Kill the space and terminate the string */
192 adx 30 sendto_one(client_p, "%s", buf);
193     t = start;
194     }
195    
196 michael 3140 if (ms->flags & CHFL_CHANOP)
197     *t++ = '@';
198     if (ms->flags & CHFL_HALFOP)
199     *t++ = '%';
200     if (ms->flags & CHFL_VOICE)
201 adx 356 *t++ = '+';
202 adx 30
203 michael 3186 strcpy(t, ms->client_p->id);
204 michael 3135
205 adx 30 t += strlen(t);
206     *t++ = ' ';
207     }
208    
209 michael 3250 /* Should always be non-NULL unless we have a kind of persistent channels */
210 michael 3235 if (chptr->members.head)
211 michael 3250 t--; /* Take the space out */
212 adx 30 *t = '\0';
213     sendto_one(client_p, "%s", buf);
214     }
215    
216 michael 3308 /*! \brief Sends +b/+e/+I
217     * \param client_p Client pointer to server
218     * \param chptr Pointer to channel
219 michael 3998 * \param list Pointer to list of modes to send
220 michael 3308 * \param flag Char flag flagging type of mode. Currently this can be 'b', e' or 'I'
221 adx 30 */
222     static void
223     send_mode_list(struct Client *client_p, struct Channel *chptr,
224 michael 3998 const dlink_list *list, char flag)
225 adx 30 {
226 michael 3250 const dlink_node *ptr = NULL;
227 michael 3215 char pbuf[IRCD_BUFSIZE] = "";
228 michael 3524 int tlen, mlen, cur_len;
229 michael 3144 char *pp = pbuf;
230 adx 30
231 michael 3998 if (list->length == 0)
232 adx 30 return;
233    
234 michael 3135 mlen = snprintf(buf, sizeof(buf), ":%s BMASK %lu %s %c :", me.id,
235     (unsigned long)chptr->channelts, chptr->chname, flag);
236     cur_len = mlen;
237 adx 30
238 michael 3998 DLINK_FOREACH(ptr, list->head)
239 adx 30 {
240 michael 3250 const struct Ban *banptr = ptr->data;
241 adx 30
242 michael 3999 tlen = banptr->len + 3; /* +3 for ! + @ + space */
243 adx 30
244     /*
245 michael 3995 * Send buffer and start over if we cannot fit another ban
246 adx 30 */
247 michael 3135 if (cur_len + (tlen - 1) > IRCD_BUFSIZE - 2)
248 adx 30 {
249 michael 3250 *(pp - 1) = '\0'; /* Get rid of trailing space on buffer */
250 michael 3135 sendto_one(client_p, "%s%s", buf, pbuf);
251 adx 30
252 michael 3135 cur_len = mlen;
253 adx 30 pp = pbuf;
254     }
255    
256 michael 2296 pp += sprintf(pp, "%s!%s@%s ", banptr->name, banptr->user,
257 michael 1793 banptr->host);
258 adx 30 cur_len += tlen;
259     }
260    
261 michael 3250 *(pp - 1) = '\0'; /* Get rid of trailing space on buffer */
262 michael 3135 sendto_one(client_p, "%s%s", buf, pbuf);
263 adx 30 }
264    
265 michael 3308 /*! \brief Send "client_p" a full list of the modes for channel chptr
266     * \param client_p Pointer to client client_p
267     * \param chptr Pointer to channel pointer
268 adx 30 */
269     void
270     send_channel_modes(struct Client *client_p, struct Channel *chptr)
271     {
272 michael 3145 char modebuf[MODEBUFLEN] = "";
273     char parabuf[MODEBUFLEN] = "";
274    
275 adx 30 channel_modes(chptr, client_p, modebuf, parabuf);
276     send_members(client_p, chptr, modebuf, parabuf);
277    
278     send_mode_list(client_p, chptr, &chptr->banlist, 'b');
279 michael 1661 send_mode_list(client_p, chptr, &chptr->exceptlist, 'e');
280     send_mode_list(client_p, chptr, &chptr->invexlist, 'I');
281 adx 30 }
282    
283 michael 3308 /*! \brief Check channel name for invalid characters
284     * \param name Pointer to channel name string
285     * \param local Indicates whether it's a local or remote creation
286 michael 632 * \return 0 if invalid, 1 otherwise
287 adx 30 */
288     int
289 michael 1847 check_channel_name(const char *name, const int local)
290 adx 30 {
291 michael 632 const char *p = name;
292 michael 3422
293 adx 30 assert(name != NULL);
294    
295 michael 632 if (!IsChanPrefix(*p))
296     return 0;
297 adx 30
298 db 633 if (!local || !ConfigChannel.disable_fake_channels)
299 michael 632 {
300     while (*++p)
301 db 634 if (!IsChanChar(*p))
302 michael 632 return 0;
303     }
304     else
305     {
306     while (*++p)
307 db 634 if (!IsVisibleChanChar(*p))
308 michael 632 return 0;
309     }
310    
311 michael 3422 return p - name <= CHANNELLEN;
312 adx 30 }
313    
314     void
315     remove_ban(struct Ban *bptr, dlink_list *list)
316     {
317     dlinkDelete(&bptr->node, list);
318    
319     MyFree(bptr->name);
320 michael 2296 MyFree(bptr->user);
321 adx 30 MyFree(bptr->host);
322     MyFree(bptr->who);
323    
324 michael 1654 mp_pool_release(bptr);
325 adx 30 }
326    
327     /* free_channel_list()
328     *
329     * inputs - pointer to dlink_list
330     * output - NONE
331     * side effects -
332     */
333     void
334     free_channel_list(dlink_list *list)
335     {
336 michael 3250 dlink_node *ptr = NULL, *ptr_next = NULL;
337 adx 30
338 michael 3250 DLINK_FOREACH_SAFE(ptr, ptr_next, list->head)
339 adx 30 remove_ban(ptr->data, list);
340    
341     assert(list->tail == NULL && list->head == NULL);
342     }
343    
344     /*! \brief Get Channel block for chname (and allocate a new channel
345     * block, if it didn't exist before)
346 michael 3308 * \param chname Channel name
347     * \return Channel block
348 adx 30 */
349     struct Channel *
350 michael 632 make_channel(const char *chname)
351 adx 30 {
352     struct Channel *chptr = NULL;
353    
354 michael 632 assert(!EmptyString(chname));
355 adx 30
356 michael 1654 chptr = mp_pool_get(channel_pool);
357 adx 30
358 michael 3308 /* Doesn't hurt to set it here */
359 michael 632 chptr->channelts = CurrentTime;
360     chptr->last_join_time = CurrentTime;
361 adx 30
362     strlcpy(chptr->chname, chname, sizeof(chptr->chname));
363 michael 3945 dlinkAdd(chptr, &chptr->node, &channel_list);
364 adx 30
365     hash_add_channel(chptr);
366    
367     return chptr;
368     }
369    
370 michael 3308 /*! \brief Walk through this channel, and destroy it.
371     * \param chptr Channel pointer
372 adx 30 */
373     void
374     destroy_channel(struct Channel *chptr)
375     {
376     dlink_node *ptr = NULL, *ptr_next = NULL;
377    
378     DLINK_FOREACH_SAFE(ptr, ptr_next, chptr->invites.head)
379     del_invite(chptr, ptr->data);
380    
381 michael 3308 /* Free ban/exception/invex lists */
382 adx 30 free_channel_list(&chptr->banlist);
383     free_channel_list(&chptr->exceptlist);
384     free_channel_list(&chptr->invexlist);
385    
386 michael 3945 dlinkDelete(&chptr->node, &channel_list);
387 adx 30 hash_del_channel(chptr);
388    
389 michael 1654 mp_pool_release(chptr);
390 adx 30 }
391    
392     /*!
393 michael 3308 * \param chptr Pointer to channel
394     * \return String pointer "=" if public, "@" if secret else "*"
395 adx 30 */
396     static const char *
397 michael 1013 channel_pub_or_secret(const struct Channel *chptr)
398 adx 30 {
399     if (SecretChannel(chptr))
400     return "@";
401     if (PrivateChannel(chptr))
402     return "*";
403     return "=";
404     }
405    
406     /*! \brief lists all names on given channel
407 michael 3308 * \param source_p Pointer to client struct requesting names
408     * \param chptr Pointer to channel block
409     * \param show_eon Show RPL_ENDOFNAMES numeric or not
410 adx 30 * (don't want it with /names with no params)
411     */
412     void
413     channel_member_names(struct Client *source_p, struct Channel *chptr,
414     int show_eon)
415     {
416 michael 1847 const dlink_node *ptr = NULL;
417 michael 3215 char lbuf[IRCD_BUFSIZE + 1] = "";
418 adx 30 char *t = NULL, *start = NULL;
419     int tlen = 0;
420     int is_member = IsMember(source_p, chptr);
421 michael 1146 int multi_prefix = HasCap(source_p, CAP_MULTI_PREFIX) != 0;
422 michael 2910 int uhnames = HasCap(source_p, CAP_UHNAMES) != 0;
423 adx 30
424     if (PubChannel(chptr) || is_member)
425     {
426 michael 3109 t = lbuf + snprintf(lbuf, sizeof(lbuf), numeric_form(RPL_NAMREPLY),
427 michael 1847 me.name, source_p->name,
428     channel_pub_or_secret(chptr), chptr->chname);
429 adx 30 start = t;
430    
431     DLINK_FOREACH(ptr, chptr->members.head)
432     {
433 michael 1847 const struct Membership *ms = ptr->data;
434 adx 30
435 michael 1847 if (HasUMode(ms->client_p, UMODE_INVISIBLE) && !is_member)
436 adx 30 continue;
437    
438 michael 2910 if (!uhnames)
439 michael 4045 tlen = strlen(ms->client_p->name) + 1; /* +1 for space */
440 michael 2910 else
441     tlen = strlen(ms->client_p->name) + strlen(ms->client_p->username) +
442 michael 4045 strlen(ms->client_p->host) + 3; /* +3 for ! + @ + space */
443 adx 30
444 michael 506 if (!multi_prefix)
445     {
446     if (ms->flags & (CHFL_CHANOP | CHFL_HALFOP | CHFL_VOICE))
447     ++tlen;
448     }
449     else
450     {
451     if (ms->flags & CHFL_CHANOP)
452     ++tlen;
453     if (ms->flags & CHFL_HALFOP)
454     ++tlen;
455     if (ms->flags & CHFL_VOICE)
456     ++tlen;
457     }
458    
459 adx 675 if (t + tlen - lbuf > IRCD_BUFSIZE - 2)
460 adx 30 {
461     *(t - 1) = '\0';
462     sendto_one(source_p, "%s", lbuf);
463     t = start;
464     }
465    
466 michael 2910 if (!uhnames)
467     t += sprintf(t, "%s%s ", get_member_status(ms, multi_prefix),
468     ms->client_p->name);
469     else
470     t += sprintf(t, "%s%s!%s@%s ", get_member_status(ms, multi_prefix),
471     ms->client_p->name, ms->client_p->username,
472     ms->client_p->host);
473 adx 30 }
474    
475 michael 3215 if (tlen)
476 adx 30 {
477     *(t - 1) = '\0';
478     sendto_one(source_p, "%s", lbuf);
479     }
480     }
481    
482     if (show_eon)
483 michael 3109 sendto_one_numeric(source_p, &me, RPL_ENDOFNAMES, chptr->chname);
484 adx 30 }
485    
486 michael 3308 /*! \brief Adds client to invite list
487     * \param chptr Pointer to channel block
488     * \param who Pointer to client to add invite to
489 adx 30 */
490     void
491     add_invite(struct Channel *chptr, struct Client *who)
492     {
493     del_invite(chptr, who);
494    
495     /*
496 michael 3308 * Delete last link in chain if the list is max length
497 adx 30 */
498 michael 317 if (dlink_list_length(&who->localClient->invited) >=
499 michael 3934 ConfigChannel.max_channels)
500 michael 317 del_invite(who->localClient->invited.tail->data, who);
501 adx 30
502 michael 3308 /* Add client to channel invite list */
503 adx 30 dlinkAdd(who, make_dlink_node(), &chptr->invites);
504    
505 michael 3308 /* Add channel to the end of the client invite list */
506 michael 317 dlinkAdd(chptr, make_dlink_node(), &who->localClient->invited);
507 adx 30 }
508    
509     /*! \brief Delete Invite block from channel invite list
510     * and client invite list
511 michael 3308 * \param chptr Pointer to Channel struct
512     * \param who Pointer to client to remove invites from
513 adx 30 */
514     void
515     del_invite(struct Channel *chptr, struct Client *who)
516     {
517     dlink_node *ptr = NULL;
518    
519 michael 317 if ((ptr = dlinkFindDelete(&who->localClient->invited, chptr)))
520 adx 30 free_dlink_node(ptr);
521    
522     if ((ptr = dlinkFindDelete(&chptr->invites, who)))
523     free_dlink_node(ptr);
524     }
525    
526     /* get_member_status()
527     *
528     * inputs - pointer to struct Membership
529     * - YES if we can combine different flags
530     * output - string either @, +, % or "" depending on whether
531     * chanop, voiced or user
532     * side effects -
533     *
534     * NOTE: Returned string is usually a static buffer
535     * (like in get_client_name)
536     */
537     const char *
538 michael 2133 get_member_status(const struct Membership *ms, const int combine)
539 adx 30 {
540 michael 4046 static char buffer[4]; /* 4 for @%+\0 */
541 michael 1902 char *p = buffer;
542 adx 30
543     if (ms->flags & CHFL_CHANOP)
544     {
545     if (!combine)
546     return "@";
547     *p++ = '@';
548     }
549    
550     if (ms->flags & CHFL_HALFOP)
551     {
552     if (!combine)
553     return "%";
554     *p++ = '%';
555     }
556    
557     if (ms->flags & CHFL_VOICE)
558     *p++ = '+';
559     *p = '\0';
560    
561     return buffer;
562     }
563    
564     /*!
565 michael 3308 * \param who Pointer to Client to check
566     * \param list Pointer to ban list to search
567 adx 30 * \return 1 if ban found for given n!u\@h mask, 0 otherwise
568     *
569     */
570     static int
571     find_bmask(const struct Client *who, const dlink_list *const list)
572     {
573     const dlink_node *ptr = NULL;
574    
575     DLINK_FOREACH(ptr, list->head)
576     {
577 michael 1455 const struct Ban *bp = ptr->data;
578 adx 30
579 michael 2296 if (!match(bp->name, who->name) && !match(bp->user, who->username))
580 michael 371 {
581     switch (bp->type)
582     {
583     case HM_HOST:
584 michael 1652 if (!match(bp->host, who->host) || !match(bp->host, who->sockhost))
585 michael 371 return 1;
586     break;
587     case HM_IPV4:
588     if (who->localClient->aftype == AF_INET)
589     if (match_ipv4(&who->localClient->ip, &bp->addr, bp->bits))
590     return 1;
591     break;
592     #ifdef IPV6
593     case HM_IPV6:
594     if (who->localClient->aftype == AF_INET6)
595     if (match_ipv6(&who->localClient->ip, &bp->addr, bp->bits))
596     return 1;
597     break;
598     #endif
599     default:
600     assert(0);
601     }
602     }
603 adx 30 }
604    
605     return 0;
606     }
607    
608     /*!
609 michael 3308 * \param chptr Pointer to channel block
610     * \param who Pointer to client to check access fo
611 adx 30 * \return 0 if not banned, 1 otherwise
612     */
613     int
614 michael 1013 is_banned(const struct Channel *chptr, const struct Client *who)
615 adx 30 {
616 michael 632 if (find_bmask(who, &chptr->banlist))
617 michael 1495 if (!find_bmask(who, &chptr->exceptlist))
618 michael 632 return 1;
619 adx 30
620 michael 632 return 0;
621 adx 30 }
622    
623 michael 3308 /*! Tests if a client can join a certain channel
624     * \param source_p Pointer to client attempting to join
625     * \param chptr Pointer to channel
626     * \param key Key sent by client attempting to join if present
627 adx 30 * \return ERR_BANNEDFROMCHAN, ERR_INVITEONLYCHAN, ERR_CHANNELISFULL
628     * or 0 if allowed to join.
629     */
630 michael 1834 int
631 adx 30 can_join(struct Client *source_p, struct Channel *chptr, const char *key)
632     {
633 michael 2246 if ((chptr->mode.mode & MODE_SSLONLY) && !HasUMode(source_p, UMODE_SSL))
634 michael 1150 return ERR_SSLONLYCHAN;
635    
636 michael 1173 if ((chptr->mode.mode & MODE_REGONLY) && !HasUMode(source_p, UMODE_REGISTERED))
637     return ERR_NEEDREGGEDNICK;
638    
639 michael 1219 if ((chptr->mode.mode & MODE_OPERONLY) && !HasUMode(source_p, UMODE_OPER))
640 michael 1150 return ERR_OPERONLYCHAN;
641    
642 adx 30 if (chptr->mode.mode & MODE_INVITEONLY)
643 michael 317 if (!dlinkFind(&source_p->localClient->invited, chptr))
644 michael 1495 if (!find_bmask(source_p, &chptr->invexlist))
645 adx 30 return ERR_INVITEONLYCHAN;
646    
647 michael 1430 if (chptr->mode.key[0] && (!key || strcmp(chptr->mode.key, key)))
648 adx 30 return ERR_BADCHANNELKEY;
649    
650     if (chptr->mode.limit && dlink_list_length(&chptr->members) >=
651     chptr->mode.limit)
652     return ERR_CHANNELISFULL;
653    
654 michael 2208 if (is_banned(chptr, source_p))
655     return ERR_BANNEDFROMCHAN;
656    
657 michael 1834 return 0;
658 adx 30 }
659    
660     int
661 michael 1847 has_member_flags(const struct Membership *ms, const unsigned int flags)
662 adx 30 {
663 michael 3215 return ms && (ms->flags & flags);
664 adx 30 }
665    
666     struct Membership *
667     find_channel_link(struct Client *client_p, struct Channel *chptr)
668     {
669     dlink_node *ptr = NULL;
670    
671     if (!IsClient(client_p))
672     return NULL;
673    
674 michael 2567 if (dlink_list_length(&chptr->members) < dlink_list_length(&client_p->channel))
675     {
676     DLINK_FOREACH(ptr, chptr->members.head)
677     if (((struct Membership *)ptr->data)->client_p == client_p)
678     return ptr->data;
679     }
680     else
681     {
682     DLINK_FOREACH(ptr, client_p->channel.head)
683     if (((struct Membership *)ptr->data)->chptr == chptr)
684     return ptr->data;
685     }
686 adx 30
687     return NULL;
688     }
689    
690 michael 3889 /*! Tests if a client can send to a channel
691     * \param message The actual message string the client wants to send
692     * \return 1 if the message does contain any control codes, 0 otherwise
693 michael 1937 */
694     static int
695     msg_has_ctrls(const char *message)
696     {
697     const unsigned char *p = (const unsigned char *)message;
698    
699     for (; *p; ++p)
700     {
701     if (*p > 31 || *p == 1)
702 michael 3953 continue; /* No control code or CTCP */
703 michael 1937
704 michael 3889 if (*p == 27) /* Escape */
705 michael 1937 {
706 michael 3889 /* ISO 2022 charset shift sequence */
707 michael 1937 if (*(p + 1) == '$' ||
708     *(p + 1) == '(')
709     {
710     ++p;
711     continue;
712     }
713     }
714    
715 michael 3889 return 1; /* Control code */
716 michael 1937 }
717    
718     return 0;
719     }
720    
721 michael 3308 /*! Tests if a client can send to a channel
722     * \param chptr Pointer to Channel struct
723     * \param source_p Pointer to Client struct
724     * \param ms Pointer to Membership struct (can be NULL)
725     * \param message The actual message string the client wants to send
726 adx 30 * \return CAN_SEND_OPV if op or voiced on channel\n
727     * CAN_SEND_NONOP if can send to channel but is not an op\n
728 michael 1173 * ERR_CANNOTSENDTOCHAN or ERR_NEEDREGGEDNICK if they cannot send to channel\n
729 adx 30 */
730     int
731 michael 1937 can_send(struct Channel *chptr, struct Client *source_p,
732     struct Membership *ms, const char *message)
733 adx 30 {
734 michael 1858 struct MaskItem *conf = NULL;
735    
736 michael 1219 if (IsServer(source_p) || HasFlag(source_p, FLAGS_SERVICE))
737 adx 30 return CAN_SEND_OPV;
738    
739 michael 565 if (MyClient(source_p) && !IsExemptResv(source_p))
740 michael 1219 if (!(HasUMode(source_p, UMODE_OPER) && ConfigFileEntry.oper_pass_resv))
741 michael 1858 if ((conf = match_find_resv(chptr->chname)) && !resv_find_exempt(source_p, conf))
742 michael 1834 return ERR_CANNOTSENDTOCHAN;
743 adx 30
744 michael 1944 if ((chptr->mode.mode & MODE_NOCTRL) && msg_has_ctrls(message))
745     return ERR_NOCTRLSONCHAN;
746     if (ms || (ms = find_channel_link(source_p, chptr)))
747 adx 30 if (ms->flags & (CHFL_CHANOP|CHFL_HALFOP|CHFL_VOICE))
748     return CAN_SEND_OPV;
749 michael 2441 if (!ms && (chptr->mode.mode & MODE_NOPRIVMSGS))
750     return ERR_CANNOTSENDTOCHAN;
751 michael 1944 if (chptr->mode.mode & MODE_MODERATED)
752     return ERR_CANNOTSENDTOCHAN;
753 michael 1954 if ((chptr->mode.mode & MODE_MODREG) && !HasUMode(source_p, UMODE_REGISTERED))
754 michael 1944 return ERR_NEEDREGGEDNICK;
755 adx 30
756 michael 3308 /* Cache can send if banned */
757 michael 1944 if (MyClient(source_p))
758     {
759     if (ms)
760 adx 30 {
761     if (ms->flags & CHFL_BAN_SILENCED)
762 michael 1834 return ERR_CANNOTSENDTOCHAN;
763 adx 30
764     if (!(ms->flags & CHFL_BAN_CHECKED))
765     {
766     if (is_banned(chptr, source_p))
767     {
768     ms->flags |= (CHFL_BAN_CHECKED|CHFL_BAN_SILENCED);
769 michael 1834 return ERR_CANNOTSENDTOCHAN;
770 adx 30 }
771    
772     ms->flags |= CHFL_BAN_CHECKED;
773     }
774     }
775 michael 1944 else if (is_banned(chptr, source_p))
776 michael 1941 return ERR_CANNOTSENDTOCHAN;
777     }
778 adx 30
779     return CAN_SEND_NONOP;
780     }
781    
782     /*! \brief Updates the client's oper_warn_count_down, warns the
783     * IRC operators if necessary, and updates
784     * join_leave_countdown as needed.
785 michael 3308 * \param source_p Pointer to struct Client to check
786     * \param name Channel name or NULL if this is a part.
787 adx 30 */
788     void
789     check_spambot_warning(struct Client *source_p, const char *name)
790     {
791     int t_delta = 0;
792     int decrement_count = 0;
793    
794     if ((GlobalSetOptions.spam_num &&
795     (source_p->localClient->join_leave_count >=
796     GlobalSetOptions.spam_num)))
797     {
798     if (source_p->localClient->oper_warn_count_down > 0)
799     source_p->localClient->oper_warn_count_down--;
800     else
801     source_p->localClient->oper_warn_count_down = 0;
802    
803     if (source_p->localClient->oper_warn_count_down == 0)
804     {
805 michael 3308 /* It's already known as a possible spambot */
806 michael 3235 if (name)
807 michael 1618 sendto_realops_flags(UMODE_BOTS, L_ALL, SEND_NOTICE,
808 adx 30 "User %s (%s@%s) trying to join %s is a possible spambot",
809     source_p->name, source_p->username,
810     source_p->host, name);
811     else
812 michael 1618 sendto_realops_flags(UMODE_BOTS, L_ALL, SEND_NOTICE,
813 adx 30 "User %s (%s@%s) is a possible spambot",
814     source_p->name, source_p->username,
815     source_p->host);
816     source_p->localClient->oper_warn_count_down = OPER_SPAM_COUNTDOWN;
817     }
818     }
819     else
820     {
821     if ((t_delta = (CurrentTime - source_p->localClient->last_leave_time)) >
822     JOIN_LEAVE_COUNT_EXPIRE_TIME)
823     {
824     decrement_count = (t_delta / JOIN_LEAVE_COUNT_EXPIRE_TIME);
825     if (decrement_count > source_p->localClient->join_leave_count)
826     source_p->localClient->join_leave_count = 0;
827     else
828     source_p->localClient->join_leave_count -= decrement_count;
829     }
830     else
831     {
832     if ((CurrentTime - (source_p->localClient->last_join_time)) <
833     GlobalSetOptions.spam_time)
834 michael 3308 source_p->localClient->join_leave_count++; /* It's a possible spambot */
835 adx 30 }
836    
837 michael 3215 if (name)
838 adx 30 source_p->localClient->last_join_time = CurrentTime;
839     else
840     source_p->localClient->last_leave_time = CurrentTime;
841     }
842     }
843    
844 michael 3308 /*! \brief Compares usercount and servercount against their split
845 adx 30 * values and adjusts splitmode accordingly
846     * \param unused Unused address pointer
847     */
848     void
849     check_splitmode(void *unused)
850     {
851     if (splitchecking && (ConfigChannel.no_join_on_split ||
852     ConfigChannel.no_create_on_split))
853     {
854 michael 4208 const unsigned int server = dlink_list_length(&global_server_list);
855 adx 30
856     if (!splitmode && ((server < split_servers) || (Count.total < split_users)))
857     {
858     splitmode = 1;
859    
860 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
861 adx 30 "Network split, activating splitmode");
862 michael 4095 event_add(&splitmode_event, NULL);
863 adx 30 }
864 michael 4065 else if (splitmode && (server >= split_servers) && (Count.total >= split_users))
865 adx 30 {
866     splitmode = 0;
867    
868 michael 1618 sendto_realops_flags(UMODE_ALL, L_ALL, SEND_NOTICE,
869 adx 30 "Network rejoined, deactivating splitmode");
870 michael 4095 event_delete(&splitmode_event);
871 adx 30 }
872     }
873     }
874    
875 michael 3308 /*! \brief Sets the channel topic for a certain channel
876 adx 30 * \param chptr Pointer to struct Channel
877     * \param topic The topic string
878     * \param topic_info n!u\@h formatted string of the topic setter
879 michael 3308 * \param topicts Timestamp on the topic
880     * \param local Whether the topic is set by a local client
881 adx 30 */
882     void
883 michael 3940 channel_set_topic(struct Channel *chptr, const char *topic,
884 michael 1751 const char *topic_info, time_t topicts, int local)
885 adx 30 {
886 michael 1751 if (local)
887     strlcpy(chptr->topic, topic, IRCD_MIN(sizeof(chptr->topic), ServerInfo.max_topic_length + 1));
888     else
889     strlcpy(chptr->topic, topic, sizeof(chptr->topic));
890    
891 michael 1203 strlcpy(chptr->topic_info, topic_info, sizeof(chptr->topic_info));
892 michael 2345 chptr->topic_time = topicts;
893 adx 30 }
894 michael 3913
895     /* do_join_0()
896     *
897     * inputs - pointer to client doing join 0
898     * output - NONE
899     * side effects - Use has decided to join 0. This is legacy
900     * from the days when channels were numbers not names. *sigh*
901     * There is a bunch of evilness necessary here due to
902     * anti spambot code.
903     */
904     void
905     channel_do_join_0(struct Client *source_p)
906     {
907     dlink_node *ptr = NULL, *ptr_next = NULL;
908    
909     if (source_p->channel.head)
910     if (MyConnect(source_p) && !HasUMode(source_p, UMODE_OPER))
911     check_spambot_warning(source_p, NULL);
912    
913     DLINK_FOREACH_SAFE(ptr, ptr_next, source_p->channel.head)
914     {
915     struct Channel *chptr = ((struct Membership *)ptr->data)->chptr;
916    
917     sendto_server(source_p, NOCAPS, NOCAPS, ":%s PART %s",
918     source_p->id, chptr->chname);
919     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s!%s@%s PART %s",
920     source_p->name, source_p->username,
921     source_p->host, chptr->chname);
922    
923     remove_user_from_channel(ptr->data);
924     }
925     }
926    
927     static char *
928     channel_find_last0(struct Client *source_p, char *chanlist)
929     {
930     int join0 = 0;
931    
932     for (char *p = chanlist; *p; ++p) /* Find last "JOIN 0" */
933     {
934     if (*p == '0' && (*(p + 1) == ',' || *(p + 1) == '\0'))
935     {
936     if ((*p + 1) == ',')
937     ++p;
938    
939     chanlist = p + 1;
940     join0 = 1;
941     }
942     else
943     {
944     while (*p != ',' && *p != '\0') /* Skip past channel name */
945     ++p;
946    
947     if (*p == '\0') /* Hit the end */
948     break;
949     }
950     }
951    
952     if (join0)
953     channel_do_join_0(source_p);
954    
955     return chanlist;
956     }
957    
958     void
959 michael 3936 channel_do_join(struct Client *source_p, char *channel, char *key_list)
960 michael 3913 {
961     char *p = NULL;
962 michael 3936 char *chan = NULL;
963 michael 3913 char *chan_list = NULL;
964     struct Channel *chptr = NULL;
965     struct MaskItem *conf = NULL;
966 michael 3934 const struct ClassItem *class = get_class_ptr(&source_p->localClient->confs);
967 michael 3913 int i = 0;
968     unsigned int flags = 0;
969    
970 michael 3936 chan_list = channel_find_last0(source_p, channel);
971 michael 3913
972     for (chan = strtoken(&p, chan_list, ","); chan;
973     chan = strtoken(&p, NULL, ","))
974     {
975     const char *key = NULL;
976    
977     /* If we have any more keys, take the first for this channel. */
978     if (!EmptyString(key_list) && (key_list = strchr(key = key_list, ',')))
979     *key_list++ = '\0';
980    
981     /* Empty keys are the same as no keys. */
982     if (key && *key == '\0')
983     key = NULL;
984    
985     if (!check_channel_name(chan, 1))
986     {
987     sendto_one_numeric(source_p, &me, ERR_BADCHANNAME, chan);
988     continue;
989     }
990    
991     if (!IsExemptResv(source_p) &&
992     !(HasUMode(source_p, UMODE_OPER) && ConfigFileEntry.oper_pass_resv) &&
993     ((conf = match_find_resv(chan)) && !resv_find_exempt(source_p, conf)))
994     {
995     ++conf->count;
996     sendto_one_numeric(source_p, &me, ERR_CHANBANREASON,
997     chan, conf->reason ? conf->reason : "Reserved channel");
998 michael 4152 sendto_realops_flags(UMODE_REJ, L_ALL, SEND_NOTICE,
999 michael 3913 "Forbidding reserved channel %s from user %s",
1000     chan, get_client_name(source_p, HIDE_IP));
1001     continue;
1002     }
1003    
1004     if (dlink_list_length(&source_p->channel) >=
1005 michael 3934 ((class->max_channels) ? class->max_channels : ConfigChannel.max_channels))
1006 michael 3913 {
1007     sendto_one_numeric(source_p, &me, ERR_TOOMANYCHANNELS, chan);
1008     break;
1009     }
1010    
1011     if ((chptr = hash_find_channel(chan)))
1012     {
1013     if (IsMember(source_p, chptr))
1014     continue;
1015    
1016     if (splitmode && !HasUMode(source_p, UMODE_OPER) &&
1017     ConfigChannel.no_join_on_split)
1018     {
1019 michael 3939 sendto_one_numeric(source_p, &me, ERR_UNAVAILRESOURCE, chptr->chname);
1020 michael 3913 continue;
1021     }
1022    
1023     /*
1024     * can_join checks for +i key, bans.
1025     */
1026     if ((i = can_join(source_p, chptr, key)))
1027     {
1028     sendto_one_numeric(source_p, &me, i, chptr->chname);
1029     continue;
1030     }
1031    
1032     /*
1033     * This should never be the case unless there is some sort of
1034     * persistant channels.
1035     */
1036     if (dlink_list_length(&chptr->members) == 0)
1037     flags = CHFL_CHANOP;
1038     else
1039     flags = 0;
1040     }
1041     else
1042     {
1043     if (splitmode && !HasUMode(source_p, UMODE_OPER) &&
1044     (ConfigChannel.no_create_on_split || ConfigChannel.no_join_on_split))
1045     {
1046     sendto_one_numeric(source_p, &me, ERR_UNAVAILRESOURCE, chan);
1047     continue;
1048     }
1049    
1050     flags = CHFL_CHANOP;
1051     chptr = make_channel(chan);
1052     }
1053    
1054     if (!HasUMode(source_p, UMODE_OPER))
1055     check_spambot_warning(source_p, chptr->chname);
1056    
1057     add_user_to_channel(chptr, source_p, flags, 1);
1058    
1059     /*
1060     * Set timestamp if appropriate, and propagate
1061     */
1062     if (flags == CHFL_CHANOP)
1063     {
1064     chptr->channelts = CurrentTime;
1065     chptr->mode.mode |= MODE_TOPICLIMIT;
1066     chptr->mode.mode |= MODE_NOPRIVMSGS;
1067    
1068     sendto_server(source_p, NOCAPS, NOCAPS, ":%s SJOIN %lu %s +nt :@%s",
1069     me.id, (unsigned long)chptr->channelts,
1070     chptr->chname, source_p->id);
1071    
1072     /*
1073     * Notify all other users on the new channel
1074     */
1075     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s!%s@%s JOIN :%s",
1076     source_p->name, source_p->username,
1077     source_p->host, chptr->chname);
1078     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s MODE %s +nt",
1079     me.name, chptr->chname);
1080    
1081     if (source_p->away[0])
1082     sendto_channel_local_butone(source_p, 0, CAP_AWAY_NOTIFY, chptr,
1083     ":%s!%s@%s AWAY :%s",
1084     source_p->name, source_p->username,
1085     source_p->host, source_p->away);
1086     }
1087     else
1088     {
1089     sendto_server(source_p, NOCAPS, NOCAPS, ":%s JOIN %lu %s +",
1090     source_p->id, (unsigned long)chptr->channelts,
1091     chptr->chname);
1092     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s!%s@%s JOIN :%s",
1093     source_p->name, source_p->username,
1094     source_p->host, chptr->chname);
1095    
1096     if (source_p->away[0])
1097     sendto_channel_local_butone(source_p, 0, CAP_AWAY_NOTIFY, chptr,
1098     ":%s!%s@%s AWAY :%s",
1099     source_p->name, source_p->username,
1100     source_p->host, source_p->away);
1101     }
1102    
1103     del_invite(chptr, source_p);
1104    
1105     if (chptr->topic[0])
1106     {
1107     sendto_one_numeric(source_p, &me, RPL_TOPIC, chptr->chname, chptr->topic);
1108     sendto_one_numeric(source_p, &me, RPL_TOPICWHOTIME, chptr->chname,
1109     chptr->topic_info, chptr->topic_time);
1110     }
1111    
1112     channel_member_names(source_p, chptr, 1);
1113    
1114     source_p->localClient->last_join_time = CurrentTime;
1115     }
1116     }
1117    
1118     /*! \brief Removes a client from a specific channel
1119     * \param source_p Pointer to source client to remove
1120     * \param name Name of channel to remove from
1121     * \param reason Part reason to show
1122     */
1123     static void
1124     channel_part_one_client(struct Client *source_p, const char *name, const char *reason)
1125     {
1126     struct Channel *chptr = NULL;
1127     struct Membership *ms = NULL;
1128    
1129     if ((chptr = hash_find_channel(name)) == NULL)
1130     {
1131     sendto_one_numeric(source_p, &me, ERR_NOSUCHCHANNEL, name);
1132     return;
1133     }
1134    
1135     if ((ms = find_channel_link(source_p, chptr)) == NULL)
1136     {
1137     sendto_one_numeric(source_p, &me, ERR_NOTONCHANNEL, chptr->chname);
1138     return;
1139     }
1140    
1141     if (MyConnect(source_p) && !HasUMode(source_p, UMODE_OPER))
1142     check_spambot_warning(source_p, NULL);
1143    
1144     /*
1145     * Remove user from the old channel (if any)
1146     * only allow /part reasons in -m chans
1147     */
1148     if (*reason && (!MyConnect(source_p) ||
1149     ((can_send(chptr, source_p, ms, reason) &&
1150     (source_p->localClient->firsttime + ConfigFileEntry.anti_spam_exit_message_time)
1151     < CurrentTime))))
1152     {
1153     sendto_server(source_p, NOCAPS, NOCAPS, ":%s PART %s :%s",
1154     source_p->id, chptr->chname, reason);
1155     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s!%s@%s PART %s :%s",
1156     source_p->name, source_p->username,
1157     source_p->host, chptr->chname, reason);
1158     }
1159     else
1160     {
1161     sendto_server(source_p, NOCAPS, NOCAPS, ":%s PART %s",
1162     source_p->id, chptr->chname);
1163     sendto_channel_local(ALL_MEMBERS, 0, chptr, ":%s!%s@%s PART %s",
1164     source_p->name, source_p->username,
1165     source_p->host, chptr->chname);
1166     }
1167    
1168     remove_user_from_channel(ms);
1169     }
1170    
1171     void
1172 michael 3936 channel_do_part(struct Client *source_p, char *channel, char *reason)
1173 michael 3913 {
1174     char *p = NULL, *name = NULL;
1175 michael 3936 char reasonbuf[KICKLEN + 1] = "";
1176 michael 3913
1177 michael 3936 if (!EmptyString(reason))
1178     strlcpy(reasonbuf, reason, sizeof(reasonbuf));
1179 michael 3913
1180 michael 3936 for (name = strtoken(&p, channel, ","); name;
1181 michael 3913 name = strtoken(&p, NULL, ","))
1182 michael 3936 channel_part_one_client(source_p, name, reasonbuf);
1183 michael 3913 }

Properties

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