ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/branches/8.2.x/src/channel_mode.c
Revision: 8446
Committed: Fri Mar 30 11:36:06 2018 UTC (6 years ago) by michael
Content type: text/x-csrc
File size: 33367 byte(s)
Log Message:
- Stylistic 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 8280 * Copyright (c) 1997-2018 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 michael 4564 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
19 adx 30 * USA
20     */
21    
22 michael 2916 /*! \file channel_mode.c
23     * \brief Controls modes on channels.
24     * \version $Id$
25     */
26    
27 adx 30 #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 michael 1632 #include "conf.h"
33 michael 371 #include "hostmask.h"
34 adx 30 #include "irc_string.h"
35     #include "ircd.h"
36     #include "numeric.h"
37 michael 3347 #include "server.h"
38 adx 30 #include "send.h"
39     #include "memory.h"
40 michael 1243 #include "parse.h"
41 adx 30
42    
43 michael 388 static char nuh_mask[MAXPARA][IRCD_BUFSIZE];
44 adx 30 static struct ChModeChange mode_changes[IRCD_BUFSIZE];
45 michael 3057 static unsigned int mode_count;
46     static unsigned int mode_limit; /* number of modes set other than simple */
47     static unsigned int simple_modes_mask; /* bit mask of simple modes already set */
48 adx 30
49    
50     /* check_string()
51     *
52     * inputs - string to check
53     * output - pointer to modified string
54     * side effects - Fixes a string so that the first white space found
55     * becomes an end of string marker (`\0`).
56     * returns the 'fixed' string or "*" if the string
57     * was NULL length or a NULL pointer.
58     */
59     static char *
60     check_string(char *s)
61     {
62 michael 4834 char *const str = s;
63 adx 30 static char star[] = "*";
64    
65 michael 1772 if (EmptyString(str))
66 michael 593 return star;
67 adx 30
68     for (; *s; ++s)
69     {
70     if (IsSpace(*s))
71     {
72     *s = '\0';
73     break;
74     }
75     }
76    
77 michael 1772 return EmptyString(str) ? star : str;
78 adx 30 }
79    
80     /*
81 michael 5582 * Ban functions to work with mode +b/e/I
82 adx 30 */
83 michael 2892 /* add the specified ID to the channel.
84     * -is 8/9/00
85 adx 30 */
86    
87     int
88 michael 3044 add_id(struct Client *client_p, struct Channel *chptr, char *banid, unsigned int type)
89 adx 30 {
90 michael 7796 dlink_list *list;
91     dlink_node *node;
92 michael 3215 char name[NICKLEN + 1] = "";
93     char user[USERLEN + 1] = "";
94     char host[HOSTLEN + 1] = "";
95 michael 593 struct split_nuh_item nuh;
96 adx 30
97     if (MyClient(client_p))
98     {
99 michael 3699 unsigned int num_mask = dlink_list_length(&chptr->banlist) +
100     dlink_list_length(&chptr->exceptlist) +
101     dlink_list_length(&chptr->invexlist);
102 adx 30
103 michael 8045 /* Don't let local clients overflow the b/e/I lists */
104     if (num_mask >= ((HasCMode(chptr, MODE_EXTLIMIT)) ? ConfigChannel.max_bans_large :
105     ConfigChannel.max_bans))
106 adx 30 {
107 michael 4617 sendto_one_numeric(client_p, &me, ERR_BANLISTFULL, chptr->name, banid);
108 adx 30 return 0;
109     }
110    
111     collapse(banid);
112     }
113    
114 michael 593 nuh.nuhmask = check_string(banid);
115     nuh.nickptr = name;
116     nuh.userptr = user;
117     nuh.hostptr = host;
118 adx 30
119 michael 593 nuh.nicksize = sizeof(name);
120     nuh.usersize = sizeof(user);
121     nuh.hostsize = sizeof(host);
122    
123     split_nuh(&nuh);
124    
125 adx 30 /*
126 michael 593 * Re-assemble a new n!u@h and print it back to banid for sending
127 adx 30 * the mode to the channel.
128     */
129 michael 7796 size_t len = snprintf(banid, IRCD_BUFSIZE, "%s!%s@%s", name, user, host);
130 adx 30
131     switch (type)
132     {
133     case CHFL_BAN:
134     list = &chptr->banlist;
135 michael 7763 clear_ban_cache_list(&chptr->locmembers);
136 adx 30 break;
137     case CHFL_EXCEPTION:
138     list = &chptr->exceptlist;
139 michael 7763 clear_ban_cache_list(&chptr->locmembers);
140 adx 30 break;
141     case CHFL_INVEX:
142     list = &chptr->invexlist;
143     break;
144     default:
145 michael 7796 list = NULL; /* Let it crash */
146 adx 30 }
147    
148 michael 4801 DLINK_FOREACH(node, list->head)
149 adx 30 {
150 michael 7796 const struct Ban *ban = node->data;
151 michael 2892
152 michael 4816 if (!irccmp(ban->name, name) &&
153     !irccmp(ban->user, user) &&
154     !irccmp(ban->host, host))
155 adx 30 return 0;
156     }
157    
158 michael 8384 struct Ban *ban = xcalloc(sizeof(*ban));
159 michael 4816 ban->when = CurrentTime;
160     ban->len = len - 2; /* -2 for ! + @ */
161     ban->type = parse_netmask(host, &ban->addr, &ban->bits);
162 michael 5758 strlcpy(ban->name, name, sizeof(ban->name));
163     strlcpy(ban->user, user, sizeof(ban->user));
164     strlcpy(ban->host, host, sizeof(ban->host));
165 michael 3699
166 adx 30 if (IsClient(client_p))
167 michael 5758 snprintf(ban->who, sizeof(ban->who), "%s!%s@%s", client_p->name,
168     client_p->username, client_p->host);
169 michael 5754 else if (IsHidden(client_p) || ConfigServerHide.hide_servers)
170 michael 5758 strlcpy(ban->who, me.name, sizeof(ban->who));
171 adx 30 else
172 michael 5758 strlcpy(ban->who, client_p->name, sizeof(ban->who));
173 adx 30
174 michael 4816 dlinkAdd(ban, &ban->node, list);
175 adx 30
176     return 1;
177     }
178    
179     /*
180     * inputs - pointer to channel
181     * - pointer to ban id
182     * - type of ban, i.e. ban, exception, invex
183     * output - 0 for failure, 1 for success
184     * side effects -
185     */
186     static int
187 michael 3044 del_id(struct Channel *chptr, char *banid, unsigned int type)
188 adx 30 {
189 michael 7796 dlink_list *list;
190     dlink_node *node;
191 michael 3215 char name[NICKLEN + 1] = "";
192     char user[USERLEN + 1] = "";
193     char host[HOSTLEN + 1] = "";
194 michael 593 struct split_nuh_item nuh;
195 adx 30
196 michael 2892 assert(banid);
197 adx 30
198 michael 593 nuh.nuhmask = check_string(banid);
199     nuh.nickptr = name;
200     nuh.userptr = user;
201     nuh.hostptr = host;
202 adx 30
203 michael 593 nuh.nicksize = sizeof(name);
204     nuh.usersize = sizeof(user);
205     nuh.hostsize = sizeof(host);
206    
207     split_nuh(&nuh);
208    
209 adx 30 /*
210 michael 593 * Re-assemble a new n!u@h and print it back to banid for sending
211 adx 30 * the mode to the channel.
212     */
213 michael 4825 snprintf(banid, IRCD_BUFSIZE, "%s!%s@%s", name, user, host);
214 adx 30
215     switch (type)
216     {
217     case CHFL_BAN:
218     list = &chptr->banlist;
219 michael 7763 clear_ban_cache_list(&chptr->locmembers);
220 adx 30 break;
221     case CHFL_EXCEPTION:
222     list = &chptr->exceptlist;
223 michael 7763 clear_ban_cache_list(&chptr->locmembers);
224 adx 30 break;
225     case CHFL_INVEX:
226     list = &chptr->invexlist;
227     break;
228     default:
229 michael 7796 list = NULL; /* Let it crash */
230 adx 30 }
231    
232 michael 4801 DLINK_FOREACH(node, list->head)
233 adx 30 {
234 michael 4816 struct Ban *ban = node->data;
235 adx 30
236 michael 4816 if (!irccmp(name, ban->name) &&
237     !irccmp(user, ban->user) &&
238     !irccmp(host, ban->host))
239 adx 30 {
240 michael 4816 remove_ban(ban, list);
241 adx 30 return 1;
242     }
243     }
244    
245     return 0;
246     }
247    
248     /* channel_modes()
249     *
250     * inputs - pointer to channel
251     * - pointer to client
252     * - pointer to mode buf
253     * - pointer to parameter buf
254     * output - NONE
255     * side effects - write the "simple" list of channel modes for channel
256     * chptr onto buffer mbuf with the parameters in pbuf.
257     */
258     void
259 michael 8438 channel_modes(const struct Channel *chptr, const struct Client *client_p, char *mbuf, char *pbuf)
260 adx 30 {
261     *mbuf++ = '+';
262     *pbuf = '\0';
263    
264 michael 8226 for (const struct chan_mode *tab = cmode_tab; tab->letter; ++tab)
265 michael 8088 if (tab->mode && HasCMode(chptr, tab->mode))
266 michael 1175 *mbuf++ = tab->letter;
267 adx 30
268     if (chptr->mode.limit)
269     {
270     *mbuf++ = 'l';
271    
272 michael 6360 if (IsServer(client_p) || IsMember(client_p, chptr))
273 michael 7356 pbuf += sprintf(pbuf, "%u ", chptr->mode.limit);
274 adx 30 }
275    
276     if (chptr->mode.key[0])
277     {
278     *mbuf++ = 'k';
279    
280 michael 6360 if (IsServer(client_p) || IsMember(client_p, chptr))
281 michael 3693 sprintf(pbuf, "%s ", chptr->mode.key);
282 adx 30 }
283    
284     *mbuf = '\0';
285     }
286    
287     /* fix_key()
288 michael 2892 *
289 adx 30 * inputs - pointer to key to clean up
290     * output - pointer to cleaned up key
291     * side effects - input string is modified
292     *
293     * stolen from Undernet's ircd -orabidoo
294     */
295     static char *
296     fix_key(char *arg)
297     {
298     unsigned char *s, *t, c;
299    
300 michael 8021 for (s = t = (unsigned char *)arg; (c = *s) && s - (unsigned char *)arg < KEYLEN; ++s)
301 adx 30 {
302     c &= 0x7f;
303 michael 2892
304 adx 30 if (c != ':' && c > ' ' && c != ',')
305     *t++ = c;
306     }
307    
308     *t = '\0';
309 michael 2892 return arg;
310 adx 30 }
311    
312 michael 3149 /*
313     * inputs - pointer to channel
314     * output - none
315     * side effects - clear ban cache
316     */
317     void
318 michael 7763 clear_ban_cache_list(dlink_list *list)
319 michael 3149 {
320 michael 7763 dlink_node *node;
321 michael 3149
322 michael 7763 DLINK_FOREACH(node, list->head)
323 michael 3149 {
324 michael 4816 struct Membership *member = node->data;
325 michael 5751 member->flags &= ~(CHFL_BAN_SILENCED | CHFL_BAN_CHECKED);
326 michael 3149 }
327     }
328    
329 michael 3635 /*
330 michael 8430 * Bitmasks for various error returns that channel_mode_set should only return
331 adx 30 * once per call -orabidoo
332     */
333 michael 3635 enum
334     {
335     SM_ERR_NOOPS = 1 << 0, /* No chan ops */
336     SM_ERR_UNKNOWN = 1 << 1,
337     SM_ERR_RPL_B = 1 << 2,
338     SM_ERR_RPL_E = 1 << 3,
339     SM_ERR_RPL_I = 1 << 4,
340     SM_ERR_NOTONCHANNEL = 1 << 5, /* Client is not on channel */
341 michael 8050 SM_ERR_NOTOPER = 1 << 6, /* Only irc-operators can change that mode */
342     SM_ERR_ONLYSERVER = 1 << 7 /* Only servers or services can change that mode */
343 michael 3635 };
344 adx 30
345     /* Mode functions handle mode changes for a particular mode... */
346     static void
347 michael 8090 chm_nosuch(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
348 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
349 adx 30 {
350     if (*errors & SM_ERR_UNKNOWN)
351     return;
352    
353     *errors |= SM_ERR_UNKNOWN;
354 michael 8094 sendto_one_numeric(source_p, &me, ERR_UNKNOWNMODE, c);
355 adx 30 }
356    
357     static void
358 michael 8090 chm_simple(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
359 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
360 adx 30 {
361 michael 8090 if (mode->only_opers)
362 adx 30 {
363 michael 8090 if (MyClient(source_p) && !HasUMode(source_p, UMODE_OPER))
364     {
365     if (!(*errors & SM_ERR_NOTOPER))
366     sendto_one_numeric(source_p, &me, ERR_NOPRIVILEGES);
367 michael 4883
368 michael 8090 *errors |= SM_ERR_NOTOPER;
369     return;
370     }
371 adx 30 }
372    
373 michael 8090 if (mode->only_servers)
374 adx 30 {
375 michael 8090 if (!IsServer(source_p) && !HasFlag(source_p, FLAGS_SERVICE))
376     {
377     if (!(*errors & SM_ERR_ONLYSERVER))
378     sendto_one_numeric(source_p, &me,
379     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
380     ERR_ONLYSERVERSCANCHANGE, chptr->name);
381 adx 30
382 michael 8090 *errors |= SM_ERR_ONLYSERVER;
383     return;
384     }
385 adx 30 }
386    
387 michael 5579 if (alev < CHACCESS_HALFOP)
388 michael 1150 {
389     if (!(*errors & SM_ERR_NOOPS))
390 michael 3109 sendto_one_numeric(source_p, &me,
391     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
392 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
393 michael 3649
394 michael 1150 *errors |= SM_ERR_NOOPS;
395     return;
396     }
397 michael 3649
398 michael 1150 /* If have already dealt with this simple mode, ignore it */
399 michael 8090 if (simple_modes_mask & mode->mode)
400 michael 1150 return;
401    
402 michael 8090 simple_modes_mask |= mode->mode;
403 michael 1150
404 michael 8096 if (dir == MODE_ADD) /* setting + */
405 michael 1150 {
406 michael 8096 if (MyClient(source_p) && HasCMode(chptr, mode->mode))
407     return;
408    
409 michael 8090 AddCMode(chptr, mode->mode);
410 michael 1150 }
411 michael 8096 else if (dir == MODE_DEL) /* setting - */
412 michael 1150 {
413 michael 8096 if (MyClient(source_p) && !HasCMode(chptr, mode->mode))
414     return;
415    
416 michael 8090 DelCMode(chptr, mode->mode);
417 michael 8096 }
418 michael 1150
419 michael 8096 mode_changes[mode_count].letter = mode->letter;
420     mode_changes[mode_count].arg = NULL;
421     mode_changes[mode_count].id = NULL;
422     mode_changes[mode_count].flags = 0;
423     mode_changes[mode_count++].dir = dir;
424 michael 1150 }
425    
426     static void
427 michael 8090 chm_ban(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
428 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
429 adx 30 {
430     if (dir == MODE_QUERY || parc <= *parn)
431     {
432 michael 7796 dlink_node *node;
433 adx 30
434     if (*errors & SM_ERR_RPL_B)
435     return;
436    
437     *errors |= SM_ERR_RPL_B;
438    
439 michael 4801 DLINK_FOREACH(node, chptr->banlist.head)
440 adx 30 {
441 michael 4816 const struct Ban *ban = node->data;
442 michael 8039
443 michael 8043 if (!HasCMode(chptr, MODE_HIDEBMASKS) || alev >= CHACCESS_HALFOP)
444 michael 8039 sendto_one_numeric(source_p, &me, RPL_BANLIST, chptr->name,
445     ban->name, ban->user, ban->host,
446     ban->who, ban->when);
447 adx 30 }
448    
449 michael 4617 sendto_one_numeric(source_p, &me, RPL_ENDOFBANLIST, chptr->name);
450 adx 30 return;
451     }
452    
453     if (alev < CHACCESS_HALFOP)
454     {
455     if (!(*errors & SM_ERR_NOOPS))
456 michael 3109 sendto_one_numeric(source_p, &me,
457     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
458 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
459 michael 4883
460 adx 30 *errors |= SM_ERR_NOOPS;
461     return;
462     }
463    
464     if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
465     return;
466    
467 michael 8050 char *const mask = nuh_mask[*parn];
468 michael 5746 strlcpy(mask, parv[*parn], sizeof(nuh_mask[*parn]));
469 michael 4790 ++(*parn);
470 michael 388
471 michael 5030 if (*mask == ':' || (!MyConnect(source_p) && strchr(mask, ' ')))
472     return;
473 adx 30
474 michael 8159 if (dir == MODE_ADD) /* setting + */
475 adx 30 {
476 michael 8446 if (add_id(source_p, chptr, mask, CHFL_BAN) == 0)
477 michael 8159 return;
478 adx 30 }
479 michael 8159 else if (dir == MODE_DEL) /* setting - */
480     {
481 michael 8446 if (del_id(chptr, mask, CHFL_BAN) == 0)
482 michael 8159 return;
483     }
484 adx 30
485 michael 8090 mode_changes[mode_count].letter = mode->letter;
486 michael 8017 mode_changes[mode_count].arg = mask; /* At this point 'mask' is no longer than NICKLEN + USERLEN + HOSTLEN + 3 */
487 michael 3735 mode_changes[mode_count].id = NULL;
488 michael 8043 if (HasCMode(chptr, MODE_HIDEBMASKS))
489 michael 8039 mode_changes[mode_count].flags = CHFL_CHANOP | CHFL_HALFOP;
490     else
491     mode_changes[mode_count].flags = 0;
492 michael 4798 mode_changes[mode_count++].dir = dir;
493 adx 30 }
494    
495     static void
496 michael 8090 chm_except(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
497 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
498 adx 30 {
499     if (dir == MODE_QUERY || parc <= *parn)
500     {
501 michael 7796 dlink_node *node;
502 adx 30
503     if (*errors & SM_ERR_RPL_E)
504     return;
505    
506     *errors |= SM_ERR_RPL_E;
507    
508 michael 4801 DLINK_FOREACH(node, chptr->exceptlist.head)
509 adx 30 {
510 michael 4816 const struct Ban *ban = node->data;
511 michael 8039
512 michael 8043 if (!HasCMode(chptr, MODE_HIDEBMASKS) || alev >= CHACCESS_HALFOP)
513 michael 8039 sendto_one_numeric(source_p, &me, RPL_EXCEPTLIST, chptr->name,
514     ban->name, ban->user, ban->host,
515     ban->who, ban->when);
516 adx 30 }
517    
518 michael 4617 sendto_one_numeric(source_p, &me, RPL_ENDOFEXCEPTLIST, chptr->name);
519 adx 30 return;
520     }
521    
522 michael 3842 if (alev < CHACCESS_HALFOP)
523     {
524     if (!(*errors & SM_ERR_NOOPS))
525     sendto_one_numeric(source_p, &me,
526     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
527 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
528 michael 4883
529 michael 3842 *errors |= SM_ERR_NOOPS;
530     return;
531     }
532    
533 adx 30 if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
534     return;
535    
536 michael 8050 char *const mask = nuh_mask[*parn];
537 michael 5746 strlcpy(mask, parv[*parn], sizeof(nuh_mask[*parn]));
538 michael 4790 ++(*parn);
539 adx 30
540 michael 5030 if (*mask == ':' || (!MyConnect(source_p) && strchr(mask, ' ')))
541     return;
542 adx 30
543 michael 8159 if (dir == MODE_ADD) /* setting + */
544 adx 30 {
545 michael 8446 if (add_id(source_p, chptr, mask, CHFL_EXCEPTION) == 0)
546 michael 8159 return;
547 adx 30 }
548 michael 8159 else if (dir == MODE_DEL) /* setting - */
549     {
550 michael 8446 if (del_id(chptr, mask, CHFL_EXCEPTION) == 0)
551 michael 8159 return;
552     }
553 adx 30
554 michael 8090 mode_changes[mode_count].letter = mode->letter;
555 michael 8017 mode_changes[mode_count].arg = mask; /* At this point 'mask' is no longer than NICKLEN + USERLEN + HOSTLEN + 3 */
556 michael 3735 mode_changes[mode_count].id = NULL;
557 michael 8043 if (HasCMode(chptr, MODE_HIDEBMASKS))
558 michael 8039 mode_changes[mode_count].flags = CHFL_CHANOP | CHFL_HALFOP;
559     else
560     mode_changes[mode_count].flags = 0;
561 michael 4798 mode_changes[mode_count++].dir = dir;
562 adx 30 }
563    
564     static void
565 michael 8090 chm_invex(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
566 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
567 adx 30 {
568     if (dir == MODE_QUERY || parc <= *parn)
569     {
570 michael 7796 dlink_node *node;
571 adx 30
572     if (*errors & SM_ERR_RPL_I)
573     return;
574    
575     *errors |= SM_ERR_RPL_I;
576    
577 michael 4801 DLINK_FOREACH(node, chptr->invexlist.head)
578 adx 30 {
579 michael 4816 const struct Ban *ban = node->data;
580 michael 8039
581 michael 8043 if (!HasCMode(chptr, MODE_HIDEBMASKS) || alev >= CHACCESS_HALFOP)
582 michael 8039 sendto_one_numeric(source_p, &me, RPL_INVEXLIST, chptr->name,
583     ban->name, ban->user, ban->host,
584     ban->who, ban->when);
585 adx 30 }
586    
587 michael 4617 sendto_one_numeric(source_p, &me, RPL_ENDOFINVEXLIST, chptr->name);
588 adx 30 return;
589     }
590    
591 michael 3842 if (alev < CHACCESS_HALFOP)
592     {
593     if (!(*errors & SM_ERR_NOOPS))
594     sendto_one_numeric(source_p, &me,
595     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
596 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
597 michael 4883
598 michael 3842 *errors |= SM_ERR_NOOPS;
599     return;
600     }
601    
602 adx 30 if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
603     return;
604    
605 michael 8050 char *const mask = nuh_mask[*parn];
606 michael 5746 strlcpy(mask, parv[*parn], sizeof(nuh_mask[*parn]));
607 michael 4790 ++(*parn);
608 adx 30
609 michael 5030 if (*mask == ':' || (!MyConnect(source_p) && strchr(mask, ' ')))
610     return;
611 adx 30
612 michael 8159 if (dir == MODE_ADD) /* setting + */
613 adx 30 {
614 michael 8446 if (add_id(source_p, chptr, mask, CHFL_INVEX) == 0)
615 michael 8159 return;
616 adx 30 }
617 michael 8159 else if (dir == MODE_DEL) /* setting - */
618     {
619 michael 8446 if (del_id(chptr, mask, CHFL_INVEX) == 0)
620 michael 8159 return;
621     }
622 adx 30
623 michael 8090 mode_changes[mode_count].letter = mode->letter;
624 michael 8017 mode_changes[mode_count].arg = mask; /* At this point 'mask' is no longer than NICKLEN + USERLEN + HOSTLEN + 3 */
625 michael 3735 mode_changes[mode_count].id = NULL;
626 michael 8043 if (HasCMode(chptr, MODE_HIDEBMASKS))
627 michael 8039 mode_changes[mode_count].flags = CHFL_CHANOP | CHFL_HALFOP;
628     else
629     mode_changes[mode_count].flags = 0;
630 michael 4798 mode_changes[mode_count++].dir = dir;
631 adx 30 }
632    
633     static void
634 michael 8090 chm_voice(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
635 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
636 adx 30 {
637 michael 3143 struct Client *target_p;
638 adx 30 struct Membership *member;
639    
640 michael 2951 if (alev < CHACCESS_HALFOP)
641 adx 30 {
642     if (!(*errors & SM_ERR_NOOPS))
643 michael 3109 sendto_one_numeric(source_p, &me,
644     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
645 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
646 michael 4883
647 adx 30 *errors |= SM_ERR_NOOPS;
648     return;
649     }
650    
651 michael 3670 if (dir == MODE_QUERY || parc <= *parn)
652 adx 30 return;
653    
654 michael 3673 if ((target_p = find_chasing(source_p, parv[(*parn)++])) == NULL)
655 michael 3736 return; /* find_chasing sends ERR_NOSUCHNICK */
656 adx 30
657 michael 3143 if ((member = find_channel_link(target_p, chptr)) == NULL)
658 adx 30 {
659     if (!(*errors & SM_ERR_NOTONCHANNEL))
660 michael 4617 sendto_one_numeric(source_p, &me, ERR_USERNOTINCHANNEL, target_p->name, chptr->name);
661 michael 4883
662 adx 30 *errors |= SM_ERR_NOTONCHANNEL;
663     return;
664     }
665    
666     if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
667     return;
668    
669 michael 8159 if (dir == MODE_ADD) /* setting + */
670 michael 6929 {
671 michael 8159 if (has_member_flags(member, CHFL_VOICE))
672     return; /* No redundant mode changes */
673 michael 4883
674 michael 8159 AddMemberFlag(member, CHFL_VOICE);
675     }
676     else if (dir == MODE_DEL) /* setting - */
677     {
678 michael 8446 if (has_member_flags(member, CHFL_VOICE) == 0)
679 michael 8159 return; /* No redundant mode changes */
680 adx 30
681 michael 8159 DelMemberFlag(member, CHFL_VOICE);
682 michael 6929 }
683    
684 michael 8090 mode_changes[mode_count].letter = mode->letter;
685 michael 3735 mode_changes[mode_count].arg = target_p->name;
686     mode_changes[mode_count].id = target_p->id;
687 michael 8039 mode_changes[mode_count].flags = 0;
688 michael 4798 mode_changes[mode_count++].dir = dir;
689 adx 30 }
690    
691     static void
692 michael 8090 chm_hop(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
693 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
694 adx 30 {
695 michael 3143 struct Client *target_p;
696 adx 30 struct Membership *member;
697    
698 michael 3714 if (alev < CHACCESS_CHANOP)
699 adx 30 {
700     if (!(*errors & SM_ERR_NOOPS))
701 michael 3109 sendto_one_numeric(source_p, &me,
702     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
703 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
704 michael 4883
705 adx 30 *errors |= SM_ERR_NOOPS;
706     return;
707     }
708    
709 michael 3670 if (dir == MODE_QUERY || parc <= *parn)
710 adx 30 return;
711    
712 michael 3673 if ((target_p = find_chasing(source_p, parv[(*parn)++])) == NULL)
713 michael 3736 return; /* find_chasing sends ERR_NOSUCHNICK */
714 adx 30
715 michael 3143 if ((member = find_channel_link(target_p, chptr)) == NULL)
716 adx 30 {
717     if (!(*errors & SM_ERR_NOTONCHANNEL))
718 michael 4617 sendto_one_numeric(source_p, &me, ERR_USERNOTINCHANNEL, target_p->name, chptr->name);
719 michael 4883
720 adx 30 *errors |= SM_ERR_NOTONCHANNEL;
721     return;
722     }
723    
724     if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
725     return;
726    
727 michael 8159 if (dir == MODE_ADD) /* setting + */
728 michael 6929 {
729 michael 8159 if (has_member_flags(member, CHFL_HALFOP))
730     return; /* No redundant mode changes */
731 michael 4883
732 michael 8159 AddMemberFlag(member, CHFL_HALFOP);
733     }
734     else if (dir == MODE_DEL) /* setting - */
735     {
736 michael 8446 if (has_member_flags(member, CHFL_HALFOP) == 0)
737 michael 8159 return; /* No redundant mode changes */
738 adx 30
739 michael 8159 DelMemberFlag(member, CHFL_HALFOP);
740 michael 6929 }
741    
742 michael 8090 mode_changes[mode_count].letter = mode->letter;
743 michael 3735 mode_changes[mode_count].arg = target_p->name;
744     mode_changes[mode_count].id = target_p->id;
745 michael 8039 mode_changes[mode_count].flags = 0;
746 michael 4798 mode_changes[mode_count++].dir = dir;
747 adx 30 }
748    
749     static void
750 michael 8090 chm_op(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
751 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
752 adx 30 {
753 michael 3143 struct Client *target_p;
754 adx 30 struct Membership *member;
755    
756 michael 2951 if (alev < CHACCESS_CHANOP)
757 adx 30 {
758     if (!(*errors & SM_ERR_NOOPS))
759 michael 3109 sendto_one_numeric(source_p, &me,
760     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
761 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
762 michael 4883
763 adx 30 *errors |= SM_ERR_NOOPS;
764     return;
765     }
766    
767 michael 3670 if (dir == MODE_QUERY || parc <= *parn)
768 adx 30 return;
769    
770 michael 3673 if ((target_p = find_chasing(source_p, parv[(*parn)++])) == NULL)
771 michael 3736 return; /* find_chasing sends ERR_NOSUCHNICK */
772 adx 30
773 michael 3143 if ((member = find_channel_link(target_p, chptr)) == NULL)
774 adx 30 {
775     if (!(*errors & SM_ERR_NOTONCHANNEL))
776 michael 4617 sendto_one_numeric(source_p, &me, ERR_USERNOTINCHANNEL, target_p->name, chptr->name);
777 michael 4883
778 adx 30 *errors |= SM_ERR_NOTONCHANNEL;
779     return;
780     }
781    
782     if (MyClient(source_p) && (++mode_limit > MAXMODEPARAMS))
783     return;
784    
785 michael 8159 if (dir == MODE_ADD) /* setting + */
786 michael 6929 {
787 michael 8159 if (has_member_flags(member, CHFL_CHANOP))
788     return; /* No redundant mode changes */
789 michael 4883
790 michael 8159 AddMemberFlag(member, CHFL_CHANOP);
791     }
792     else if (dir == MODE_DEL) /* setting - */
793     {
794 michael 8446 if (has_member_flags(member, CHFL_CHANOP) == 0)
795 michael 8159 return; /* No redundant mode changes */
796 adx 30
797 michael 8159 DelMemberFlag(member, CHFL_CHANOP);
798 michael 6929 }
799    
800 michael 8090 mode_changes[mode_count].letter = mode->letter;
801 michael 3735 mode_changes[mode_count].arg = target_p->name;
802     mode_changes[mode_count].id = target_p->id;
803 michael 8039 mode_changes[mode_count].flags = 0;
804 michael 4798 mode_changes[mode_count++].dir = dir;
805 adx 30 }
806    
807     static void
808 michael 8090 chm_limit(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
809 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
810 adx 30 {
811     if (alev < CHACCESS_HALFOP)
812     {
813     if (!(*errors & SM_ERR_NOOPS))
814 michael 3109 sendto_one_numeric(source_p, &me,
815     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
816 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
817 adx 30 *errors |= SM_ERR_NOOPS;
818     return;
819     }
820    
821     if (dir == MODE_QUERY)
822     return;
823    
824 michael 3670 if (dir == MODE_ADD && parc > *parn)
825 adx 30 {
826 michael 4834 char *const lstr = parv[(*parn)++];
827 michael 6946 int limit = 0;
828 adx 30
829 michael 3153 if (EmptyString(lstr) || (limit = atoi(lstr)) <= 0)
830 adx 30 return;
831    
832 michael 1793 sprintf(lstr, "%d", limit);
833 adx 30
834 michael 3151 /* If somebody sets MODE #channel +ll 1 2, accept latter --fl */
835 michael 3670 for (unsigned int i = 0; i < mode_count; ++i)
836 michael 8090 if (mode_changes[i].letter == mode->letter && mode_changes[i].dir == MODE_ADD)
837 adx 30 mode_changes[i].letter = 0;
838    
839 michael 8090 mode_changes[mode_count].letter = mode->letter;
840 michael 3735 mode_changes[mode_count].arg = lstr;
841 adx 30 mode_changes[mode_count].id = NULL;
842 michael 8039 mode_changes[mode_count].flags = 0;
843 michael 4798 mode_changes[mode_count++].dir = dir;
844 adx 30
845     chptr->mode.limit = limit;
846     }
847     else if (dir == MODE_DEL)
848     {
849 michael 8446 if (chptr->mode.limit == 0)
850 adx 30 return;
851    
852     chptr->mode.limit = 0;
853    
854 michael 8090 mode_changes[mode_count].letter = mode->letter;
855 michael 3735 mode_changes[mode_count].arg = NULL;
856 adx 30 mode_changes[mode_count].id = NULL;
857 michael 8039 mode_changes[mode_count].flags = 0;
858 michael 4798 mode_changes[mode_count++].dir = dir;
859 adx 30 }
860     }
861    
862     static void
863 michael 8090 chm_key(struct Client *source_p, struct Channel *chptr, int parc, int *parn, char **parv,
864 michael 8094 int *errors, int alev, int dir, const char c, const struct chan_mode *mode)
865 adx 30 {
866     if (alev < CHACCESS_HALFOP)
867     {
868     if (!(*errors & SM_ERR_NOOPS))
869 michael 3109 sendto_one_numeric(source_p, &me,
870     alev == CHACCESS_NOTONCHAN ? ERR_NOTONCHANNEL :
871 michael 4617 ERR_CHANOPRIVSNEEDED, chptr->name);
872 adx 30 *errors |= SM_ERR_NOOPS;
873     return;
874     }
875    
876     if (dir == MODE_QUERY)
877     return;
878    
879 michael 3670 if (dir == MODE_ADD && parc > *parn)
880 adx 30 {
881 michael 4834 char *const key = fix_key(parv[(*parn)++]);
882 adx 30
883 michael 3153 if (EmptyString(key))
884 adx 30 return;
885    
886     assert(key[0] != ' ');
887     strlcpy(chptr->mode.key, key, sizeof(chptr->mode.key));
888    
889 michael 3151 /* If somebody does MODE #channel +kk a b, accept latter --fl */
890 michael 3670 for (unsigned int i = 0; i < mode_count; ++i)
891 michael 8090 if (mode_changes[i].letter == mode->letter && mode_changes[i].dir == MODE_ADD)
892 adx 30 mode_changes[i].letter = 0;
893    
894 michael 8090 mode_changes[mode_count].letter = mode->letter;
895 michael 8021 mode_changes[mode_count].arg = key;
896 adx 30 mode_changes[mode_count].id = NULL;
897 michael 8039 mode_changes[mode_count].flags = 0;
898 michael 4798 mode_changes[mode_count++].dir = dir;
899 adx 30 }
900     else if (dir == MODE_DEL)
901     {
902     if (parc > *parn)
903 michael 4790 ++(*parn);
904 adx 30
905     if (chptr->mode.key[0] == '\0')
906     return;
907    
908     chptr->mode.key[0] = '\0';
909    
910 michael 8090 mode_changes[mode_count].letter = mode->letter;
911 michael 3735 mode_changes[mode_count].arg = "*";
912 adx 30 mode_changes[mode_count].id = NULL;
913 michael 8039 mode_changes[mode_count].flags = 0;
914 michael 4798 mode_changes[mode_count++].dir = dir;
915 adx 30 }
916     }
917    
918     /* get_channel_access()
919     *
920     * inputs - pointer to Client struct
921     * - pointer to Membership struct
922     * output - CHACCESS_CHANOP if we should let them have
923     * chanop level access, 0 for peon level access.
924     * side effects - NONE
925     */
926     static int
927 michael 2941 get_channel_access(const struct Client *source_p,
928     const struct Membership *member)
929 adx 30 {
930     /* Let hacked servers in for now... */
931     if (!MyClient(source_p))
932 michael 4101 return CHACCESS_REMOTE;
933 adx 30
934 michael 3235 if (!member)
935 adx 30 return CHACCESS_NOTONCHAN;
936    
937 michael 3374 /* Just to be sure.. */
938 adx 30 assert(source_p == member->client_p);
939    
940     if (has_member_flags(member, CHFL_CHANOP))
941     return CHACCESS_CHANOP;
942    
943     if (has_member_flags(member, CHFL_HALFOP))
944     return CHACCESS_HALFOP;
945    
946     return CHACCESS_PEON;
947     }
948    
949 michael 3151 /* send_mode_changes_server()
950 michael 3159 * Input: the source client(source_p),
951 adx 30 * the channel to send mode changes for(chptr)
952     * Output: None.
953 michael 3151 * Side-effects: Sends the appropriate mode changes to servers.
954 adx 30 *
955     */
956     static void
957 michael 3150 send_mode_changes_server(struct Client *source_p, struct Channel *chptr)
958 adx 30 {
959 michael 8015 char modebuf[IRCD_BUFSIZE] = "";
960     char parabuf[IRCD_BUFSIZE] = "";
961     char *parptr = parabuf;
962 michael 8041 unsigned int mbl = 0, pbl = 0, arglen = 0, modecount = 0, paracount = 0;
963     unsigned int dir = MODE_QUERY;
964 adx 30
965 michael 8015 mbl = snprintf(modebuf, sizeof(modebuf), ":%s TMODE %ju %s ",
966     source_p->id, chptr->creationtime, chptr->name);
967 adx 30
968 michael 3374 /* Loop the list of modes we have */
969 michael 5546 for (unsigned int i = 0; i < mode_count; ++i)
970 adx 30 {
971 michael 3669 if (mode_changes[i].letter == 0)
972 adx 30 continue;
973    
974 michael 8015 const char *arg;
975 michael 3141 if (mode_changes[i].id)
976     arg = mode_changes[i].id;
977     else
978     arg = mode_changes[i].arg;
979 adx 30
980 michael 3215 if (arg)
981 michael 3141 arglen = strlen(arg);
982     else
983     arglen = 0;
984    
985 michael 3136 /*
986     * If we're creeping past the buf size, we need to send it and make
987 adx 30 * another line for the other modes
988     */
989 michael 8013 if ((paracount == MAXMODEPARAMS) ||
990 michael 8015 ((arglen + mbl + pbl + 2 /* +2 for /r/n */ ) > IRCD_BUFSIZE))
991 adx 30 {
992 michael 8013 if (modecount)
993 michael 4963 sendto_server(source_p, 0, 0, "%s %s", modebuf, parabuf);
994 michael 3215
995 michael 8013 modecount = 0;
996     paracount = 0;
997 adx 30
998 michael 8015 mbl = snprintf(modebuf, sizeof(modebuf), ":%s TMODE %ju %s ",
999     source_p->id, chptr->creationtime, chptr->name);
1000 adx 30
1001     pbl = 0;
1002     parabuf[0] = '\0';
1003     parptr = parabuf;
1004     dir = MODE_QUERY;
1005     }
1006    
1007     if (dir != mode_changes[i].dir)
1008     {
1009     modebuf[mbl++] = (mode_changes[i].dir == MODE_ADD) ? '+' : '-';
1010     dir = mode_changes[i].dir;
1011     }
1012    
1013     modebuf[mbl++] = mode_changes[i].letter;
1014     modebuf[mbl] = '\0';
1015 michael 8013 ++modecount;
1016 adx 30
1017 michael 3215 if (arg)
1018 adx 30 {
1019 michael 8015 int len = sprintf(parptr, (pbl == 0) ? "%s" : " %s", arg);
1020 adx 30 pbl += len;
1021     parptr += len;
1022 michael 8013 ++paracount;
1023 adx 30 }
1024     }
1025    
1026 michael 8013 if (modecount)
1027 michael 4963 sendto_server(source_p, 0, 0, "%s %s", modebuf, parabuf);
1028 adx 30 }
1029    
1030     /* void send_mode_changes(struct Client *client_p,
1031     * struct Client *source_p,
1032     * struct Channel *chptr)
1033     * Input: The client sending(client_p), the source client(source_p),
1034     * the channel to send mode changes for(chptr),
1035     * mode change globals.
1036     * Output: None.
1037     * Side-effects: Sends the appropriate mode changes to other clients
1038     * and propagates to servers.
1039     */
1040     static void
1041 michael 5747 send_mode_changes_client(struct Client *source_p, struct Channel *chptr)
1042 adx 30 {
1043 michael 8039 unsigned int flags = 0;
1044 adx 30
1045 michael 8039 for (unsigned int pass = 2; pass--; flags = CHFL_CHANOP | CHFL_HALFOP)
1046 adx 30 {
1047 michael 8039 char modebuf[IRCD_BUFSIZE] = "";
1048     char parabuf[IRCD_BUFSIZE] = "";
1049     char *parptr = parabuf;
1050 michael 8041 unsigned int mbl = 0, pbl = 0, arglen = 0, modecount = 0, paracount = 0;
1051     unsigned int dir = MODE_QUERY;
1052 adx 30
1053 michael 8430 if (IsClient(source_p))
1054     mbl = snprintf(modebuf, sizeof(modebuf), ":%s!%s@%s MODE %s ", source_p->name,
1055     source_p->username, source_p->host, chptr->name);
1056     else
1057 michael 8039 mbl = snprintf(modebuf, sizeof(modebuf), ":%s MODE %s ", (IsHidden(source_p) ||
1058     ConfigServerHide.hide_servers) ?
1059     me.name : source_p->name, chptr->name);
1060 adx 30
1061 michael 8039 for (unsigned int i = 0; i < mode_count; ++i)
1062 adx 30 {
1063 michael 8039 if (mode_changes[i].letter == 0 || mode_changes[i].flags != flags)
1064     continue;
1065 adx 30
1066 michael 8039 const char *arg = mode_changes[i].arg;
1067     if (arg)
1068     arglen = strlen(arg);
1069 adx 30 else
1070 michael 8039 arglen = 0;
1071 adx 30
1072 michael 8039 if ((paracount == MAXMODEPARAMS) ||
1073     ((arglen + mbl + pbl + 2 /* +2 for /r/n */ ) > IRCD_BUFSIZE))
1074     {
1075     if (modecount)
1076     sendto_channel_local(NULL, chptr, flags, 0, 0, "%s %s", modebuf, parabuf);
1077 adx 30
1078 michael 8039 modecount = 0;
1079     paracount = 0;
1080 adx 30
1081 michael 8430 if (IsClient(source_p))
1082     mbl = snprintf(modebuf, sizeof(modebuf), ":%s!%s@%s MODE %s ", source_p->name,
1083     source_p->username, source_p->host, chptr->name);
1084     else
1085 michael 8039 mbl = snprintf(modebuf, sizeof(modebuf), ":%s MODE %s ", (IsHidden(source_p) ||
1086     ConfigServerHide.hide_servers) ?
1087     me.name : source_p->name, chptr->name);
1088 adx 30
1089 michael 8039 pbl = 0;
1090     parabuf[0] = '\0';
1091     parptr = parabuf;
1092     dir = MODE_QUERY;
1093     }
1094    
1095     if (dir != mode_changes[i].dir)
1096     {
1097     modebuf[mbl++] = (mode_changes[i].dir == MODE_ADD) ? '+' : '-';
1098     dir = mode_changes[i].dir;
1099     }
1100    
1101     modebuf[mbl++] = mode_changes[i].letter;
1102     modebuf[mbl] = '\0';
1103     ++modecount;
1104    
1105     if (arg)
1106     {
1107     int len = sprintf(parptr, (pbl == 0) ? "%s" : " %s", arg);
1108     pbl += len;
1109     parptr += len;
1110     ++paracount;
1111     }
1112 adx 30 }
1113 michael 8039
1114     if (modecount)
1115     sendto_channel_local(NULL, chptr, flags, 0, 0, "%s %s", modebuf, parabuf);
1116 adx 30 }
1117     }
1118    
1119 michael 8090 const struct chan_mode *cmode_map[256];
1120     const struct chan_mode cmode_tab[] =
1121 michael 8088 {
1122 michael 8093 { .letter = 'b', .func = chm_ban },
1123     { .letter = 'c', .mode = MODE_NOCTRL, .func = chm_simple },
1124     { .letter = 'e', .func = chm_except },
1125     { .letter = 'h', .func = chm_hop },
1126     { .letter = 'i', .mode = MODE_INVITEONLY, .func = chm_simple },
1127     { .letter = 'k', .func = chm_key },
1128     { .letter = 'l', .func = chm_limit },
1129     { .letter = 'm', .mode = MODE_MODERATED, .func = chm_simple },
1130     { .letter = 'n', .mode = MODE_NOPRIVMSGS, .func = chm_simple },
1131     { .letter = 'o', .func = chm_op },
1132     { .letter = 'p', .mode = MODE_PRIVATE, .func = chm_simple },
1133     { .letter = 'r', .mode = MODE_REGISTERED, .only_servers = 1, .func = chm_simple },
1134     { .letter = 's', .mode = MODE_SECRET, .func = chm_simple },
1135     { .letter = 't', .mode = MODE_TOPICLIMIT, .func = chm_simple },
1136     { .letter = 'u', .mode = MODE_HIDEBMASKS, .func = chm_simple },
1137     { .letter = 'v', .func = chm_voice },
1138     { .letter = 'C', .mode = MODE_NOCTCP, .func = chm_simple },
1139     { .letter = 'I', .func = chm_invex },
1140     { .letter = 'L', .mode = MODE_EXTLIMIT, .only_opers = 1, .func = chm_simple },
1141 michael 8096 { .letter = 'M', .mode = MODE_MODREG, .func = chm_simple },
1142 michael 8093 { .letter = 'O', .mode = MODE_OPERONLY, .only_opers = 1, .func = chm_simple },
1143 michael 8096 { .letter = 'R', .mode = MODE_REGONLY, .func = chm_simple },
1144     { .letter = 'S', .mode = MODE_SSLONLY, .func = chm_simple },
1145     { .letter = 'T', .mode = MODE_NONOTICE, .func = chm_simple },
1146 michael 8093 { .letter = '\0', .mode = 0 }
1147 michael 8088 };
1148    
1149     void
1150     channel_mode_init(void)
1151     {
1152 michael 8090 for (const struct chan_mode *tab = cmode_tab; tab->letter; ++tab)
1153 michael 8088 cmode_map[tab->letter] = tab;
1154     }
1155    
1156 michael 3670 /*
1157 michael 3159 * Input: The the client this originated
1158 adx 30 * from, the channel, the parameter count starting at the modes,
1159     * the parameters, the channel name.
1160     * Output: None.
1161     * Side-effects: Changes the channel membership and modes appropriately,
1162     * sends the appropriate MODE messages to the appropriate
1163     * clients.
1164     */
1165     void
1166 michael 8430 channel_mode_set(struct Client *source_p, struct Channel *chptr,
1167 michael 2937 struct Membership *member, int parc, char *parv[])
1168 adx 30 {
1169     int dir = MODE_ADD;
1170     int parn = 1;
1171 michael 3670 int alevel = 0, errors = 0;
1172 adx 30
1173     mode_count = 0;
1174     mode_limit = 0;
1175     simple_modes_mask = 0;
1176    
1177     alevel = get_channel_access(source_p, member);
1178    
1179 michael 3283 for (const char *ml = parv[0]; *ml; ++ml)
1180 adx 30 {
1181 michael 3250 switch (*ml)
1182 adx 30 {
1183     case '+':
1184     dir = MODE_ADD;
1185     break;
1186     case '-':
1187     dir = MODE_DEL;
1188     break;
1189     case '=':
1190     dir = MODE_QUERY;
1191     break;
1192     default:
1193 michael 2939 {
1194 michael 8090 const struct chan_mode *mode = cmode_map[(unsigned char)*ml];
1195 michael 2939
1196 michael 8088 if (mode)
1197 michael 8094 mode->func(source_p, chptr, parc, &parn, parv, &errors, alevel, dir, *ml, mode);
1198 michael 8088 else
1199 michael 8250 chm_nosuch(source_p, chptr, parc, &parn, parv, &errors, alevel, dir, *ml, NULL);
1200 adx 30 break;
1201 michael 2939 }
1202 adx 30 }
1203     }
1204    
1205 michael 5750 /* Bail out if we have nothing to do... */
1206 michael 8446 if (mode_count == 0)
1207 michael 5750 return;
1208    
1209 michael 5747 send_mode_changes_client(source_p, chptr);
1210     send_mode_changes_server(source_p, chptr);
1211 adx 30 }

Properties

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