2 * charybdis: A slightly useful ircd.
3 * chmode.c: channel mode management
5 * Copyright (C) 1990 Jarkko Oikarinen and University of Oulu, Co Center
6 * Copyright (C) 1996-2002 Hybrid Development Team
7 * Copyright (C) 2002-2005 ircd-ratbox development team
8 * Copyright (C) 2005-2006 charybdis development team
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
25 * $Id: chmode.c 3580 2007-11-07 23:45:14Z jilles $
37 #include "s_serv.h" /* captab */
41 #include "s_conf.h" /* ConfigFileEntry, ConfigChannel */
42 #include "s_newconf.h"
46 /* bitmasks for error returns, so we send once per call */
47 #define SM_ERR_NOTS 0x00000001 /* No TS on channel */
48 #define SM_ERR_NOOPS 0x00000002 /* No chan ops */
49 #define SM_ERR_UNKNOWN 0x00000004
50 #define SM_ERR_RPL_C 0x00000008
51 #define SM_ERR_RPL_B 0x00000010
52 #define SM_ERR_RPL_E 0x00000020
53 #define SM_ERR_NOTONCHANNEL 0x00000040 /* Not on channel */
54 #define SM_ERR_RPL_I 0x00000100
55 #define SM_ERR_RPL_D 0x00000200
56 #define SM_ERR_NOPRIVS 0x00000400
57 #define SM_ERR_RPL_Q 0x00000800
58 #define SM_ERR_RPL_F 0x00001000
60 #define MAXMODES_SIMPLE 46 /* a-zA-Z except bqeIov */
62 static struct ChModeChange mode_changes
[BUFSIZE
];
63 static int mode_count
;
64 static int mode_limit
;
65 static int mode_limit_simple
;
69 char cflagsmyinfo
[256];
71 int chmode_flags
[256];
73 /* OPTIMIZE ME! -- dwr */
75 construct_noparam_modes(void)
78 char *ptr
= cflagsbuf
;
79 char *ptr2
= cflagsmyinfo
;
80 static int prev_chmode_flags
[256];
85 for(i
= 0; i
< 256; i
++)
87 if( !(chmode_table
[i
].set_func
== chm_ban
) &&
88 !(chmode_table
[i
].set_func
== chm_forward
) &&
89 !(chmode_table
[i
].set_func
== chm_throttle
) &&
90 !(chmode_table
[i
].set_func
== chm_key
) &&
91 !(chmode_table
[i
].set_func
== chm_limit
) &&
92 !(chmode_table
[i
].set_func
== chm_op
) &&
93 !(chmode_table
[i
].set_func
== chm_voice
))
95 chmode_flags
[i
] = chmode_table
[i
].mode_type
;
102 if (prev_chmode_flags
[i
] != 0 && prev_chmode_flags
[i
] != chmode_flags
[i
])
104 if (chmode_flags
[i
] == 0)
106 chmode_table
[i
].set_func
= chm_orphaned
;
107 sendto_realops_snomask(SNO_DEBUG
, L_ALL
, "Cmode +%c is now orphaned", i
);
111 sendto_realops_snomask(SNO_DEBUG
, L_ALL
, "Orphaned cmode +%c is picked up by module", i
);
113 chmode_flags
[i
] = prev_chmode_flags
[i
];
116 prev_chmode_flags
[i
] = chmode_flags
[i
];
118 switch (chmode_flags
[i
])
121 case MODE_DISFORWARD
:
122 if(ConfigChannel
.use_forward
)
129 if(rb_dlink_list_length(&service_list
))
136 if(chmode_flags
[i
] != 0)
142 /* Should we leave orphaned check here? -- dwr */
143 if(!(chmode_table
[i
].set_func
== chm_nosuch
) && !(chmode_table
[i
].set_func
== chm_orphaned
))
157 * outputs - an available cflag bitmask or
158 * 0 if no cflags are available
159 * side effects - NONE
162 find_cflag_slot(void)
164 unsigned int all_cflags
= 0, my_cflag
= 0, i
;
166 for (i
= 0; i
< 256; i
++)
167 all_cflags
|= chmode_flags
[i
];
169 for (my_cflag
= 1; my_cflag
&& (all_cflags
& my_cflag
);
176 get_channel_access(struct Client
*source_p
, struct membership
*msptr
)
178 if(!MyClient(source_p
) || is_chanop(msptr
))
186 * inputs - client, channel, id to add, type
187 * outputs - 0 on failure, 1 on success
188 * side effects - given id is added to the appropriate list
191 add_id(struct Client
*source_p
, struct Channel
*chptr
, const char *banid
,
192 rb_dlink_list
* list
, long mode_type
)
194 struct Ban
*actualBan
;
195 static char who
[USERHOST_REPLYLEN
];
196 char *realban
= LOCAL_COPY(banid
);
199 /* dont let local clients overflow the banlist, or set redundant
202 if(MyClient(source_p
))
204 if((rb_dlink_list_length(&chptr
->banlist
) + rb_dlink_list_length(&chptr
->exceptlist
) + rb_dlink_list_length(&chptr
->invexlist
) + rb_dlink_list_length(&chptr
->quietlist
)) >= (chptr
->mode
.mode
& MODE_EXLIMIT
? ConfigChannel
.max_bans_large
: ConfigChannel
.max_bans
))
206 sendto_one(source_p
, form_str(ERR_BANLISTFULL
),
207 me
.name
, source_p
->name
, chptr
->chname
, realban
);
211 RB_DLINK_FOREACH(ptr
, list
->head
)
213 actualBan
= ptr
->data
;
214 if(mask_match(actualBan
->banstr
, realban
))
218 /* dont let remotes set duplicates */
221 RB_DLINK_FOREACH(ptr
, list
->head
)
223 actualBan
= ptr
->data
;
224 if(!irccmp(actualBan
->banstr
, realban
))
230 if(IsPerson(source_p
))
231 rb_sprintf(who
, "%s!%s@%s", source_p
->name
, source_p
->username
, source_p
->host
);
233 rb_strlcpy(who
, source_p
->name
, sizeof(who
));
235 actualBan
= allocate_ban(realban
, who
);
236 actualBan
->when
= rb_current_time();
238 rb_dlinkAdd(actualBan
, &actualBan
->node
, list
);
240 /* invalidate the can_send() cache */
241 if(mode_type
== CHFL_BAN
|| mode_type
== CHFL_QUIET
|| mode_type
== CHFL_EXCEPTION
)
249 * inputs - channel, id to remove, type
250 * outputs - 0 on failure, 1 on success
251 * side effects - given id is removed from the appropriate list
254 del_id(struct Channel
*chptr
, const char *banid
, rb_dlink_list
* list
, long mode_type
)
259 if(EmptyString(banid
))
262 RB_DLINK_FOREACH(ptr
, list
->head
)
266 if(irccmp(banid
, banptr
->banstr
) == 0)
268 rb_dlinkDelete(&banptr
->node
, list
);
271 /* invalidate the can_send() cache */
272 if(mode_type
== CHFL_BAN
|| mode_type
== CHFL_QUIET
|| mode_type
== CHFL_EXCEPTION
)
284 * input - string to check
285 * output - pointer to 'fixed' string, or "*" if empty
286 * side effects - any white space found becomes \0
289 check_string(char *s
)
292 static char splat
[] = "*";
309 * inputs - mask to pretty
310 * outputs - better version of the mask
311 * side effects - mask is chopped to limits, and transformed:
319 pretty_mask(const char *idmask
)
321 static char mask_buf
[BUFSIZE
];
323 char *nick
, *user
, *host
;
326 char ne
= 0, ue
= 0, he
= 0; /* save values at nick[NICKLEN], et all */
329 mask
= LOCAL_COPY(idmask
);
330 mask
= check_string(mask
);
333 nick
= user
= host
= splat
;
335 if((size_t) BUFSIZE
- mask_pos
< strlen(mask
) + 5)
338 old_mask_pos
= mask_pos
;
342 mask_pos
+= rb_sprintf(mask_buf
+ mask_pos
, "%s", mask
) + 1;
343 t
= mask_buf
+ old_mask_pos
+ 1;
349 return mask_buf
+ old_mask_pos
;
353 if((t
= strchr(mask
, '@')) != NULL
)
360 if((t
= strchr(mask
, '!')) != NULL
)
375 else if((t
= strchr(mask
, '!')) != NULL
)
384 else if(strchr(mask
, '.') != NULL
|| strchr(mask
, ':') != NULL
)
395 /* truncate values to max lengths */
396 if(strlen(nick
) > NICKLEN
- 1)
398 ne
= nick
[NICKLEN
- 1];
399 nick
[NICKLEN
- 1] = '\0';
401 if(strlen(user
) > USERLEN
)
404 user
[USERLEN
] = '\0';
406 if(strlen(host
) > HOSTLEN
)
409 host
[HOSTLEN
] = '\0';
412 mask_pos
+= rb_sprintf(mask_buf
+ mask_pos
, "%s!%s@%s", nick
, user
, host
) + 1;
414 /* restore mask, since we may need to use it again later */
420 nick
[NICKLEN
- 1] = ne
;
426 return mask_buf
+ old_mask_pos
;
432 * output - the same key, fixed
433 * side effects - anything below ascii 13 is discarded, ':' discarded,
434 * high ascii is dropped to lower half of ascii table
441 for(s
= t
= (u_char
*) arg
; (c
= *s
); s
++)
444 if(c
!= ':' && c
!= ',' && c
> ' ')
455 * ouput - the same key, fixed
456 * side effects - high ascii dropped to lower half of table,
457 * CR/LF/':' are dropped
460 fix_key_remote(char *arg
)
464 for(s
= t
= (u_char
*) arg
; (c
= *s
); s
++)
467 if((c
!= 0x0a) && (c
!= ':') && (c
!= ',') && (c
!= 0x0d) && (c
!= ' '))
477 * The handlers for each specific mode.
480 chm_nosuch(struct Client
*source_p
, struct Channel
*chptr
,
481 int alevel
, int parc
, int *parn
,
482 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
484 if(*errors
& SM_ERR_UNKNOWN
)
486 *errors
|= SM_ERR_UNKNOWN
;
487 sendto_one(source_p
, form_str(ERR_UNKNOWNMODE
), me
.name
, source_p
->name
, c
);
491 chm_simple(struct Client
*source_p
, struct Channel
*chptr
,
492 int alevel
, int parc
, int *parn
,
493 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
495 if(alevel
!= CHFL_CHANOP
)
497 if(!(*errors
& SM_ERR_NOOPS
))
498 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
499 me
.name
, source_p
->name
, chptr
->chname
);
500 *errors
|= SM_ERR_NOOPS
;
504 /* flags (possibly upto 32) + 4 with param */
505 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
509 if((dir
== MODE_ADD
) && !(chptr
->mode
.mode
& mode_type
))
511 /* if +f is disabled, ignore an attempt to set +QF locally */
512 if(!ConfigChannel
.use_forward
&& MyClient(source_p
) &&
513 (c
== 'Q' || c
== 'F'))
516 chptr
->mode
.mode
|= mode_type
;
518 mode_changes
[mode_count
].letter
= c
;
519 mode_changes
[mode_count
].dir
= MODE_ADD
;
520 mode_changes
[mode_count
].caps
= 0;
521 mode_changes
[mode_count
].nocaps
= 0;
522 mode_changes
[mode_count
].id
= NULL
;
523 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
524 mode_changes
[mode_count
++].arg
= NULL
;
526 else if((dir
== MODE_DEL
) && (chptr
->mode
.mode
& mode_type
))
528 chptr
->mode
.mode
&= ~mode_type
;
530 mode_changes
[mode_count
].letter
= c
;
531 mode_changes
[mode_count
].dir
= MODE_DEL
;
532 mode_changes
[mode_count
].caps
= 0;
533 mode_changes
[mode_count
].nocaps
= 0;
534 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
535 mode_changes
[mode_count
].id
= NULL
;
536 mode_changes
[mode_count
++].arg
= NULL
;
541 chm_orphaned(struct Client
*source_p
, struct Channel
*chptr
,
542 int alevel
, int parc
, int *parn
,
543 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
545 if(MyClient(source_p
))
548 if((dir
== MODE_ADD
) && !(chptr
->mode
.mode
& mode_type
))
550 chptr
->mode
.mode
|= mode_type
;
552 mode_changes
[mode_count
].letter
= c
;
553 mode_changes
[mode_count
].dir
= MODE_ADD
;
554 mode_changes
[mode_count
].caps
= 0;
555 mode_changes
[mode_count
].nocaps
= 0;
556 mode_changes
[mode_count
].id
= NULL
;
557 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
558 mode_changes
[mode_count
++].arg
= NULL
;
560 else if((dir
== MODE_DEL
) && (chptr
->mode
.mode
& mode_type
))
562 chptr
->mode
.mode
&= ~mode_type
;
564 mode_changes
[mode_count
].letter
= c
;
565 mode_changes
[mode_count
].dir
= MODE_DEL
;
566 mode_changes
[mode_count
].caps
= 0;
567 mode_changes
[mode_count
].nocaps
= 0;
568 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
569 mode_changes
[mode_count
].id
= NULL
;
570 mode_changes
[mode_count
++].arg
= NULL
;
575 chm_staff(struct Client
*source_p
, struct Channel
*chptr
,
576 int alevel
, int parc
, int *parn
,
577 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
579 if(!IsOper(source_p
) && !IsServer(source_p
))
581 if(!(*errors
& SM_ERR_NOPRIVS
))
582 sendto_one_numeric(source_p
, ERR_NOPRIVILEGES
, form_str(ERR_NOPRIVILEGES
));
583 *errors
|= SM_ERR_NOPRIVS
;
586 if(MyClient(source_p
) && !IsOperResv(source_p
))
588 if(!(*errors
& SM_ERR_NOPRIVS
))
589 sendto_one(source_p
, form_str(ERR_NOPRIVS
), me
.name
,
590 source_p
->name
, "resv");
591 *errors
|= SM_ERR_NOPRIVS
;
595 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
599 if((dir
== MODE_ADD
) && !(chptr
->mode
.mode
& mode_type
))
601 chptr
->mode
.mode
|= mode_type
;
603 mode_changes
[mode_count
].letter
= c
;
604 mode_changes
[mode_count
].dir
= MODE_ADD
;
605 mode_changes
[mode_count
].caps
= 0;
606 mode_changes
[mode_count
].nocaps
= 0;
607 mode_changes
[mode_count
].id
= NULL
;
608 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
609 mode_changes
[mode_count
++].arg
= NULL
;
611 else if((dir
== MODE_DEL
) && (chptr
->mode
.mode
& mode_type
))
613 chptr
->mode
.mode
&= ~mode_type
;
615 mode_changes
[mode_count
].letter
= c
;
616 mode_changes
[mode_count
].dir
= MODE_DEL
;
617 mode_changes
[mode_count
].caps
= 0;
618 mode_changes
[mode_count
].nocaps
= 0;
619 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
620 mode_changes
[mode_count
].id
= NULL
;
621 mode_changes
[mode_count
++].arg
= NULL
;
626 chm_ban(struct Client
*source_p
, struct Channel
*chptr
,
627 int alevel
, int parc
, int *parn
,
628 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
631 const char *raw_mask
;
644 list
= &chptr
->banlist
;
645 errorval
= SM_ERR_RPL_B
;
646 rpl_list
= RPL_BANLIST
;
647 rpl_endlist
= RPL_ENDOFBANLIST
;
653 /* if +e is disabled, allow all but +e locally */
654 if(!ConfigChannel
.use_except
&& MyClient(source_p
) &&
655 ((dir
== MODE_ADD
) && (parc
> *parn
)))
658 list
= &chptr
->exceptlist
;
659 errorval
= SM_ERR_RPL_E
;
660 rpl_list
= RPL_EXCEPTLIST
;
661 rpl_endlist
= RPL_ENDOFEXCEPTLIST
;
664 if(ConfigChannel
.use_except
|| (dir
== MODE_DEL
))
671 /* if +I is disabled, allow all but +I locally */
672 if(!ConfigChannel
.use_invex
&& MyClient(source_p
) &&
673 (dir
== MODE_ADD
) && (parc
> *parn
))
676 list
= &chptr
->invexlist
;
677 errorval
= SM_ERR_RPL_I
;
678 rpl_list
= RPL_INVITELIST
;
679 rpl_endlist
= RPL_ENDOFINVITELIST
;
682 if(ConfigChannel
.use_invex
|| (dir
== MODE_DEL
))
689 list
= &chptr
->quietlist
;
690 errorval
= SM_ERR_RPL_Q
;
691 rpl_list
= RPL_BANLIST
;
692 rpl_endlist
= RPL_ENDOFBANLIST
;
698 sendto_realops_snomask(SNO_GENERAL
, L_ALL
, "chm_ban() called with unknown type!");
703 if(dir
== 0 || parc
<= *parn
)
705 if((*errors
& errorval
) != 0)
709 /* non-ops cant see +eI lists.. */
710 if(alevel
!= CHFL_CHANOP
&& mode_type
!= CHFL_BAN
&&
711 mode_type
!= CHFL_QUIET
)
713 if(!(*errors
& SM_ERR_NOOPS
))
714 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
715 me
.name
, source_p
->name
, chptr
->chname
);
716 *errors
|= SM_ERR_NOOPS
;
720 RB_DLINK_FOREACH(ptr
, list
->head
)
723 sendto_one(source_p
, form_str(rpl_list
),
724 me
.name
, source_p
->name
, chptr
->chname
,
725 banptr
->banstr
, banptr
->who
, banptr
->when
);
727 if (mode_type
== CHFL_QUIET
)
728 sendto_one(source_p
, ":%s %d %s %s :End of Channel Quiet List", me
.name
, rpl_endlist
, source_p
->name
, chptr
->chname
);
730 sendto_one(source_p
, form_str(rpl_endlist
), me
.name
, source_p
->name
, chptr
->chname
);
734 if(alevel
!= CHFL_CHANOP
)
736 if(!(*errors
& SM_ERR_NOOPS
))
737 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
738 me
.name
, source_p
->name
, chptr
->chname
);
739 *errors
|= SM_ERR_NOOPS
;
743 if(MyClient(source_p
) && (++mode_limit
> MAXMODEPARAMS
))
746 raw_mask
= parv
[(*parn
)];
749 /* empty ban, or starts with ':' which messes up s2s, ignore it */
750 if(EmptyString(raw_mask
) || *raw_mask
== ':')
753 if(!MyClient(source_p
))
755 if(strchr(raw_mask
, ' '))
761 mask
= pretty_mask(raw_mask
);
763 /* we'd have problems parsing this, hyb6 does it too
764 * also make sure it will always fit on a line with channel
767 if(strlen(mask
) > IRCD_MIN(BANLEN
, MODEBUFLEN
- 5))
770 /* if we're adding a NEW id */
773 if (*mask
== '$' && MyClient(source_p
))
775 if (!valid_extban(mask
, source_p
, chptr
, mode_type
))
776 /* XXX perhaps return an error message here */
780 /* dont allow local clients to overflow the banlist, dont
781 * let remote servers set duplicate bans
783 if(!add_id(source_p
, chptr
, mask
, list
, mode_type
))
786 mode_changes
[mode_count
].letter
= c
;
787 mode_changes
[mode_count
].dir
= MODE_ADD
;
788 mode_changes
[mode_count
].caps
= caps
;
789 mode_changes
[mode_count
].nocaps
= 0;
790 mode_changes
[mode_count
].mems
= mems
;
791 mode_changes
[mode_count
].id
= NULL
;
792 mode_changes
[mode_count
++].arg
= mask
;
794 else if(dir
== MODE_DEL
)
796 if(del_id(chptr
, mask
, list
, mode_type
) == 0)
798 /* mask isn't a valid ban, check raw_mask */
799 if(del_id(chptr
, raw_mask
, list
, mode_type
))
803 mode_changes
[mode_count
].letter
= c
;
804 mode_changes
[mode_count
].dir
= MODE_DEL
;
805 mode_changes
[mode_count
].caps
= caps
;
806 mode_changes
[mode_count
].nocaps
= 0;
807 mode_changes
[mode_count
].mems
= mems
;
808 mode_changes
[mode_count
].id
= NULL
;
809 mode_changes
[mode_count
++].arg
= mask
;
814 chm_op(struct Client
*source_p
, struct Channel
*chptr
,
815 int alevel
, int parc
, int *parn
,
816 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
818 struct membership
*mstptr
;
820 struct Client
*targ_p
;
822 if(alevel
!= CHFL_CHANOP
)
824 if(!(*errors
& SM_ERR_NOOPS
))
825 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
826 me
.name
, source_p
->name
, chptr
->chname
);
827 *errors
|= SM_ERR_NOOPS
;
831 if((dir
== MODE_QUERY
) || (parc
<= *parn
))
834 opnick
= parv
[(*parn
)];
838 if(EmptyString(opnick
))
840 sendto_one_numeric(source_p
, ERR_NOSUCHNICK
, form_str(ERR_NOSUCHNICK
), "*");
844 if((targ_p
= find_chasing(source_p
, opnick
, NULL
)) == NULL
)
849 mstptr
= find_channel_membership(chptr
, targ_p
);
853 if(!(*errors
& SM_ERR_NOTONCHANNEL
) && MyClient(source_p
))
854 sendto_one_numeric(source_p
, ERR_USERNOTINCHANNEL
,
855 form_str(ERR_USERNOTINCHANNEL
), opnick
, chptr
->chname
);
856 *errors
|= SM_ERR_NOTONCHANNEL
;
860 if(MyClient(source_p
) && (++mode_limit
> MAXMODEPARAMS
))
865 if(targ_p
== source_p
)
868 mode_changes
[mode_count
].letter
= c
;
869 mode_changes
[mode_count
].dir
= MODE_ADD
;
870 mode_changes
[mode_count
].caps
= 0;
871 mode_changes
[mode_count
].nocaps
= 0;
872 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
873 mode_changes
[mode_count
].id
= targ_p
->id
;
874 mode_changes
[mode_count
].arg
= targ_p
->name
;
875 mode_changes
[mode_count
++].client
= targ_p
;
877 mstptr
->flags
|= CHFL_CHANOP
;
881 if(MyClient(source_p
) && IsService(targ_p
))
883 sendto_one(source_p
, form_str(ERR_ISCHANSERVICE
),
884 me
.name
, source_p
->name
, targ_p
->name
, chptr
->chname
);
888 mode_changes
[mode_count
].letter
= c
;
889 mode_changes
[mode_count
].dir
= MODE_DEL
;
890 mode_changes
[mode_count
].caps
= 0;
891 mode_changes
[mode_count
].nocaps
= 0;
892 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
893 mode_changes
[mode_count
].id
= targ_p
->id
;
894 mode_changes
[mode_count
].arg
= targ_p
->name
;
895 mode_changes
[mode_count
++].client
= targ_p
;
897 mstptr
->flags
&= ~CHFL_CHANOP
;
902 chm_voice(struct Client
*source_p
, struct Channel
*chptr
,
903 int alevel
, int parc
, int *parn
,
904 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
906 struct membership
*mstptr
;
908 struct Client
*targ_p
;
910 if(alevel
!= CHFL_CHANOP
)
912 if(!(*errors
& SM_ERR_NOOPS
))
913 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
914 me
.name
, source_p
->name
, chptr
->chname
);
915 *errors
|= SM_ERR_NOOPS
;
919 if((dir
== MODE_QUERY
) || parc
<= *parn
)
922 opnick
= parv
[(*parn
)];
926 if(EmptyString(opnick
))
928 sendto_one_numeric(source_p
, ERR_NOSUCHNICK
, form_str(ERR_NOSUCHNICK
), "*");
932 if((targ_p
= find_chasing(source_p
, opnick
, NULL
)) == NULL
)
937 mstptr
= find_channel_membership(chptr
, targ_p
);
941 if(!(*errors
& SM_ERR_NOTONCHANNEL
) && MyClient(source_p
))
942 sendto_one_numeric(source_p
, ERR_USERNOTINCHANNEL
,
943 form_str(ERR_USERNOTINCHANNEL
), opnick
, chptr
->chname
);
944 *errors
|= SM_ERR_NOTONCHANNEL
;
948 if(MyClient(source_p
) && (++mode_limit
> MAXMODEPARAMS
))
953 mode_changes
[mode_count
].letter
= c
;
954 mode_changes
[mode_count
].dir
= MODE_ADD
;
955 mode_changes
[mode_count
].caps
= 0;
956 mode_changes
[mode_count
].nocaps
= 0;
957 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
958 mode_changes
[mode_count
].id
= targ_p
->id
;
959 mode_changes
[mode_count
].arg
= targ_p
->name
;
960 mode_changes
[mode_count
++].client
= targ_p
;
962 mstptr
->flags
|= CHFL_VOICE
;
966 mode_changes
[mode_count
].letter
= 'v';
967 mode_changes
[mode_count
].dir
= MODE_DEL
;
968 mode_changes
[mode_count
].caps
= 0;
969 mode_changes
[mode_count
].nocaps
= 0;
970 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
971 mode_changes
[mode_count
].id
= targ_p
->id
;
972 mode_changes
[mode_count
].arg
= targ_p
->name
;
973 mode_changes
[mode_count
++].client
= targ_p
;
975 mstptr
->flags
&= ~CHFL_VOICE
;
980 chm_limit(struct Client
*source_p
, struct Channel
*chptr
,
981 int alevel
, int parc
, int *parn
,
982 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
985 static char limitstr
[30];
988 if(alevel
!= CHFL_CHANOP
)
990 if(!(*errors
& SM_ERR_NOOPS
))
991 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
992 me
.name
, source_p
->name
, chptr
->chname
);
993 *errors
|= SM_ERR_NOOPS
;
997 if(dir
== MODE_QUERY
)
1000 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
1003 if((dir
== MODE_ADD
) && parc
> *parn
)
1005 lstr
= parv
[(*parn
)];
1008 if(EmptyString(lstr
) || (limit
= atoi(lstr
)) <= 0)
1011 rb_sprintf(limitstr
, "%d", limit
);
1013 mode_changes
[mode_count
].letter
= c
;
1014 mode_changes
[mode_count
].dir
= MODE_ADD
;
1015 mode_changes
[mode_count
].caps
= 0;
1016 mode_changes
[mode_count
].nocaps
= 0;
1017 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1018 mode_changes
[mode_count
].id
= NULL
;
1019 mode_changes
[mode_count
++].arg
= limitstr
;
1021 chptr
->mode
.limit
= limit
;
1023 else if(dir
== MODE_DEL
)
1025 if(!chptr
->mode
.limit
)
1028 chptr
->mode
.limit
= 0;
1030 mode_changes
[mode_count
].letter
= c
;
1031 mode_changes
[mode_count
].dir
= MODE_DEL
;
1032 mode_changes
[mode_count
].caps
= 0;
1033 mode_changes
[mode_count
].nocaps
= 0;
1034 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1035 mode_changes
[mode_count
].id
= NULL
;
1036 mode_changes
[mode_count
++].arg
= NULL
;
1041 chm_throttle(struct Client
*source_p
, struct Channel
*chptr
,
1042 int alevel
, int parc
, int *parn
,
1043 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
1045 int joins
= 0, timeslice
= 0;
1047 if(alevel
!= CHFL_CHANOP
)
1049 if(!(*errors
& SM_ERR_NOOPS
))
1050 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
1051 me
.name
, source_p
->name
, chptr
->chname
);
1052 *errors
|= SM_ERR_NOOPS
;
1056 if(dir
== MODE_QUERY
)
1059 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
1062 if((dir
== MODE_ADD
) && parc
> *parn
)
1064 sscanf(parv
[(*parn
)], "%d:%d", &joins
, ×lice
);
1066 if(!joins
|| !timeslice
)
1069 mode_changes
[mode_count
].letter
= c
;
1070 mode_changes
[mode_count
].dir
= MODE_ADD
;
1071 mode_changes
[mode_count
].caps
= 0;
1072 mode_changes
[mode_count
].nocaps
= 0;
1073 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1074 mode_changes
[mode_count
].id
= NULL
;
1075 mode_changes
[mode_count
++].arg
= parv
[(*parn
)];
1079 chptr
->mode
.join_num
= joins
;
1080 chptr
->mode
.join_time
= timeslice
;
1082 else if(dir
== MODE_DEL
)
1084 if(!chptr
->mode
.join_num
)
1087 chptr
->mode
.join_num
= 0;
1088 chptr
->mode
.join_time
= 0;
1089 chptr
->join_count
= 0;
1090 chptr
->join_delta
= 0;
1092 mode_changes
[mode_count
].letter
= c
;
1093 mode_changes
[mode_count
].dir
= MODE_DEL
;
1094 mode_changes
[mode_count
].caps
= 0;
1095 mode_changes
[mode_count
].nocaps
= 0;
1096 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1097 mode_changes
[mode_count
].id
= NULL
;
1098 mode_changes
[mode_count
++].arg
= NULL
;
1103 chm_forward(struct Client
*source_p
, struct Channel
*chptr
,
1104 int alevel
, int parc
, int *parn
,
1105 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
1107 struct Channel
*targptr
= NULL
;
1108 struct membership
*msptr
;
1109 const char *forward
;
1111 /* if +f is disabled, ignore local attempts to set it */
1112 if(!ConfigChannel
.use_forward
&& MyClient(source_p
) &&
1113 (dir
== MODE_ADD
) && (parc
> *parn
))
1116 if(dir
== MODE_QUERY
|| (dir
== MODE_ADD
&& parc
<= *parn
))
1118 if (!(*errors
& SM_ERR_RPL_F
))
1120 if (*chptr
->mode
.forward
== '\0')
1121 sendto_one_notice(source_p
, ":%s has no forward channel", chptr
->chname
);
1123 sendto_one_notice(source_p
, ":%s forward channel is %s", chptr
->chname
, chptr
->mode
.forward
);
1124 *errors
|= SM_ERR_RPL_F
;
1129 #ifndef FORWARD_OPERONLY
1130 if(alevel
!= CHFL_CHANOP
)
1132 if(!(*errors
& SM_ERR_NOOPS
))
1133 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
1134 me
.name
, source_p
->name
, chptr
->chname
);
1135 *errors
|= SM_ERR_NOOPS
;
1139 if(!IsOper(source_p
) && !IsServer(source_p
))
1141 if(!(*errors
& SM_ERR_NOPRIVS
))
1142 sendto_one_numeric(source_p
, ERR_NOPRIVILEGES
, form_str(ERR_NOPRIVILEGES
));
1143 *errors
|= SM_ERR_NOPRIVS
;
1148 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
1151 if(dir
== MODE_ADD
&& parc
> *parn
)
1153 forward
= parv
[(*parn
)];
1156 if(EmptyString(forward
))
1158 if(!check_channel_name(forward
) ||
1159 (MyClient(source_p
) && (strlen(forward
) > LOC_CHANNELLEN
|| hash_find_resv(forward
))))
1161 sendto_one_numeric(source_p
, ERR_BADCHANNAME
, form_str(ERR_BADCHANNAME
), forward
);
1164 /* don't forward to inconsistent target -- jilles */
1165 if(chptr
->chname
[0] == '#' && forward
[0] == '&')
1167 sendto_one_numeric(source_p
, ERR_BADCHANNAME
,
1168 form_str(ERR_BADCHANNAME
), forward
);
1171 if(MyClient(source_p
) && (targptr
= find_channel(forward
)) == NULL
)
1173 sendto_one_numeric(source_p
, ERR_NOSUCHCHANNEL
,
1174 form_str(ERR_NOSUCHCHANNEL
), forward
);
1177 if(MyClient(source_p
) && !(targptr
->mode
.mode
& MODE_FREETARGET
))
1179 if((msptr
= find_channel_membership(targptr
, source_p
)) == NULL
||
1180 get_channel_access(source_p
, msptr
) != CHFL_CHANOP
)
1182 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
1183 me
.name
, source_p
->name
, targptr
->chname
);
1188 rb_strlcpy(chptr
->mode
.forward
, forward
, sizeof(chptr
->mode
.forward
));
1190 mode_changes
[mode_count
].letter
= c
;
1191 mode_changes
[mode_count
].dir
= MODE_ADD
;
1192 mode_changes
[mode_count
].caps
= 0;
1193 mode_changes
[mode_count
].nocaps
= 0;
1194 mode_changes
[mode_count
].mems
= ConfigChannel
.use_forward
? ALL_MEMBERS
: ONLY_SERVERS
;
1195 mode_changes
[mode_count
].id
= NULL
;
1196 mode_changes
[mode_count
++].arg
= forward
;
1198 else if(dir
== MODE_DEL
)
1200 if(!(*chptr
->mode
.forward
))
1203 *chptr
->mode
.forward
= '\0';
1205 mode_changes
[mode_count
].letter
= c
;
1206 mode_changes
[mode_count
].dir
= MODE_DEL
;
1207 mode_changes
[mode_count
].caps
= 0;
1208 mode_changes
[mode_count
].nocaps
= 0;
1209 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1210 mode_changes
[mode_count
].id
= NULL
;
1211 mode_changes
[mode_count
++].arg
= NULL
;
1216 chm_key(struct Client
*source_p
, struct Channel
*chptr
,
1217 int alevel
, int parc
, int *parn
,
1218 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
1222 if(alevel
!= CHFL_CHANOP
)
1224 if(!(*errors
& SM_ERR_NOOPS
))
1225 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
1226 me
.name
, source_p
->name
, chptr
->chname
);
1227 *errors
|= SM_ERR_NOOPS
;
1231 if(dir
== MODE_QUERY
)
1234 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
1237 if((dir
== MODE_ADD
) && parc
> *parn
)
1239 key
= LOCAL_COPY(parv
[(*parn
)]);
1242 if(MyClient(source_p
))
1245 fix_key_remote(key
);
1247 if(EmptyString(key
))
1250 s_assert(key
[0] != ' ');
1251 rb_strlcpy(chptr
->mode
.key
, key
, sizeof(chptr
->mode
.key
));
1253 mode_changes
[mode_count
].letter
= c
;
1254 mode_changes
[mode_count
].dir
= MODE_ADD
;
1255 mode_changes
[mode_count
].caps
= 0;
1256 mode_changes
[mode_count
].nocaps
= 0;
1257 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1258 mode_changes
[mode_count
].id
= NULL
;
1259 mode_changes
[mode_count
++].arg
= chptr
->mode
.key
;
1261 else if(dir
== MODE_DEL
)
1263 static char splat
[] = "*";
1269 if(!(*chptr
->mode
.key
))
1272 /* hack time. when we get a +k-k mode, the +k arg is
1273 * chptr->mode.key, which the -k sets to \0, so hunt for a
1274 * +k when we get a -k, and set the arg to splat. --anfl
1276 for(i
= 0; i
< mode_count
; i
++)
1278 if(mode_changes
[i
].letter
== 'k' && mode_changes
[i
].dir
== MODE_ADD
)
1279 mode_changes
[i
].arg
= splat
;
1282 *chptr
->mode
.key
= 0;
1284 mode_changes
[mode_count
].letter
= c
;
1285 mode_changes
[mode_count
].dir
= MODE_DEL
;
1286 mode_changes
[mode_count
].caps
= 0;
1287 mode_changes
[mode_count
].nocaps
= 0;
1288 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1289 mode_changes
[mode_count
].id
= NULL
;
1290 mode_changes
[mode_count
++].arg
= "*";
1295 chm_regonly(struct Client
*source_p
, struct Channel
*chptr
,
1296 int alevel
, int parc
, int *parn
,
1297 const char **parv
, int *errors
, int dir
, char c
, long mode_type
)
1299 if(alevel
!= CHFL_CHANOP
)
1301 if(!(*errors
& SM_ERR_NOOPS
))
1302 sendto_one(source_p
, form_str(ERR_CHANOPRIVSNEEDED
),
1303 me
.name
, source_p
->name
, chptr
->chname
);
1304 *errors
|= SM_ERR_NOOPS
;
1308 if(dir
== MODE_QUERY
)
1311 if(((dir
== MODE_ADD
) && (chptr
->mode
.mode
& mode_type
)) ||
1312 ((dir
== MODE_DEL
) && !(chptr
->mode
.mode
& mode_type
)))
1315 if(MyClient(source_p
) && (++mode_limit_simple
> MAXMODES_SIMPLE
))
1319 chptr
->mode
.mode
|= mode_type
;
1321 chptr
->mode
.mode
&= ~mode_type
;
1323 mode_changes
[mode_count
].letter
= c
;
1324 mode_changes
[mode_count
].dir
= dir
;
1325 mode_changes
[mode_count
].caps
= CAP_SERVICE
;
1326 mode_changes
[mode_count
].nocaps
= 0;
1327 mode_changes
[mode_count
].mems
= ALL_MEMBERS
;
1328 mode_changes
[mode_count
].id
= NULL
;
1329 mode_changes
[mode_count
++].arg
= NULL
;
1333 struct ChannelMode chmode_table
[256] =
1335 {chm_nosuch
, 0 }, /* 0x00 */
1336 {chm_nosuch
, 0 }, /* 0x01 */
1337 {chm_nosuch
, 0 }, /* 0x02 */
1338 {chm_nosuch
, 0 }, /* 0x03 */
1339 {chm_nosuch
, 0 }, /* 0x04 */
1340 {chm_nosuch
, 0 }, /* 0x05 */
1341 {chm_nosuch
, 0 }, /* 0x06 */
1342 {chm_nosuch
, 0 }, /* 0x07 */
1343 {chm_nosuch
, 0 }, /* 0x08 */
1344 {chm_nosuch
, 0 }, /* 0x09 */
1345 {chm_nosuch
, 0 }, /* 0x0a */
1346 {chm_nosuch
, 0 }, /* 0x0b */
1347 {chm_nosuch
, 0 }, /* 0x0c */
1348 {chm_nosuch
, 0 }, /* 0x0d */
1349 {chm_nosuch
, 0 }, /* 0x0e */
1350 {chm_nosuch
, 0 }, /* 0x0f */
1351 {chm_nosuch
, 0 }, /* 0x10 */
1352 {chm_nosuch
, 0 }, /* 0x11 */
1353 {chm_nosuch
, 0 }, /* 0x12 */
1354 {chm_nosuch
, 0 }, /* 0x13 */
1355 {chm_nosuch
, 0 }, /* 0x14 */
1356 {chm_nosuch
, 0 }, /* 0x15 */
1357 {chm_nosuch
, 0 }, /* 0x16 */
1358 {chm_nosuch
, 0 }, /* 0x17 */
1359 {chm_nosuch
, 0 }, /* 0x18 */
1360 {chm_nosuch
, 0 }, /* 0x19 */
1361 {chm_nosuch
, 0 }, /* 0x1a */
1362 {chm_nosuch
, 0 }, /* 0x1b */
1363 {chm_nosuch
, 0 }, /* 0x1c */
1364 {chm_nosuch
, 0 }, /* 0x1d */
1365 {chm_nosuch
, 0 }, /* 0x1e */
1366 {chm_nosuch
, 0 }, /* 0x1f */
1367 {chm_nosuch
, 0 }, /* 0x20 */
1368 {chm_nosuch
, 0 }, /* 0x21 */
1369 {chm_nosuch
, 0 }, /* 0x22 */
1370 {chm_nosuch
, 0 }, /* 0x23 */
1371 {chm_nosuch
, 0 }, /* 0x24 */
1372 {chm_nosuch
, 0 }, /* 0x25 */
1373 {chm_nosuch
, 0 }, /* 0x26 */
1374 {chm_nosuch
, 0 }, /* 0x27 */
1375 {chm_nosuch
, 0 }, /* 0x28 */
1376 {chm_nosuch
, 0 }, /* 0x29 */
1377 {chm_nosuch
, 0 }, /* 0x2a */
1378 {chm_nosuch
, 0 }, /* 0x2b */
1379 {chm_nosuch
, 0 }, /* 0x2c */
1380 {chm_nosuch
, 0 }, /* 0x2d */
1381 {chm_nosuch
, 0 }, /* 0x2e */
1382 {chm_nosuch
, 0 }, /* 0x2f */
1383 {chm_nosuch
, 0 }, /* 0x30 */
1384 {chm_nosuch
, 0 }, /* 0x31 */
1385 {chm_nosuch
, 0 }, /* 0x32 */
1386 {chm_nosuch
, 0 }, /* 0x33 */
1387 {chm_nosuch
, 0 }, /* 0x34 */
1388 {chm_nosuch
, 0 }, /* 0x35 */
1389 {chm_nosuch
, 0 }, /* 0x36 */
1390 {chm_nosuch
, 0 }, /* 0x37 */
1391 {chm_nosuch
, 0 }, /* 0x38 */
1392 {chm_nosuch
, 0 }, /* 0x39 */
1393 {chm_nosuch
, 0 }, /* 0x3a */
1394 {chm_nosuch
, 0 }, /* 0x3b */
1395 {chm_nosuch
, 0 }, /* 0x3c */
1396 {chm_nosuch
, 0 }, /* 0x3d */
1397 {chm_nosuch
, 0 }, /* 0x3e */
1398 {chm_nosuch
, 0 }, /* 0x3f */
1400 {chm_nosuch
, 0 }, /* @ */
1401 {chm_nosuch
, 0 }, /* A */
1402 {chm_nosuch
, 0 }, /* B */
1403 {chm_nosuch
, 0 }, /* C */
1404 {chm_nosuch
, 0 }, /* D */
1405 {chm_nosuch
, 0 }, /* E */
1406 {chm_simple
, MODE_FREETARGET
}, /* F */
1407 {chm_nosuch
, 0 }, /* G */
1408 {chm_nosuch
, 0 }, /* H */
1409 {chm_ban
, CHFL_INVEX
}, /* I */
1410 {chm_nosuch
, 0 }, /* J */
1411 {chm_nosuch
, 0 }, /* K */
1412 {chm_staff
, MODE_EXLIMIT
}, /* L */
1413 {chm_nosuch
, 0 }, /* M */
1414 {chm_nosuch
, 0 }, /* N */
1415 {chm_nosuch
, 0 }, /* O */
1416 {chm_staff
, MODE_PERMANENT
}, /* P */
1417 {chm_simple
, MODE_DISFORWARD
}, /* Q */
1418 {chm_nosuch
, 0 }, /* R */
1419 {chm_nosuch
, 0 }, /* S */
1420 {chm_nosuch
, 0 }, /* T */
1421 {chm_nosuch
, 0 }, /* U */
1422 {chm_nosuch
, 0 }, /* V */
1423 {chm_nosuch
, 0 }, /* W */
1424 {chm_nosuch
, 0 }, /* X */
1425 {chm_nosuch
, 0 }, /* Y */
1426 {chm_nosuch
, 0 }, /* Z */
1433 {chm_nosuch
, 0 }, /* a */
1434 {chm_ban
, CHFL_BAN
}, /* b */
1435 {chm_simple
, MODE_NOCOLOR
}, /* c */
1436 {chm_nosuch
, 0 }, /* d */
1437 {chm_ban
, CHFL_EXCEPTION
}, /* e */
1438 {chm_forward
, 0 }, /* f */
1439 {chm_simple
, MODE_FREEINVITE
}, /* g */
1440 {chm_nosuch
, 0 }, /* h */
1441 {chm_simple
, MODE_INVITEONLY
}, /* i */
1442 {chm_throttle
, 0 }, /* j */
1443 {chm_key
, 0 }, /* k */
1444 {chm_limit
, 0 }, /* l */
1445 {chm_simple
, MODE_MODERATED
}, /* m */
1446 {chm_simple
, MODE_NOPRIVMSGS
}, /* n */
1447 {chm_op
, 0 }, /* o */
1448 {chm_simple
, MODE_PRIVATE
}, /* p */
1449 {chm_ban
, CHFL_QUIET
}, /* q */
1450 {chm_regonly
, MODE_REGONLY
}, /* r */
1451 {chm_simple
, MODE_SECRET
}, /* s */
1452 {chm_simple
, MODE_TOPICLIMIT
}, /* t */
1453 {chm_nosuch
, 0 }, /* u */
1454 {chm_voice
, 0 }, /* v */
1455 {chm_nosuch
, 0 }, /* w */
1456 {chm_nosuch
, 0 }, /* x */
1457 {chm_nosuch
, 0 }, /* y */
1458 {chm_simple
, MODE_OPMODERATE
}, /* z */
1460 {chm_nosuch
, 0 }, /* 0x7b */
1461 {chm_nosuch
, 0 }, /* 0x7c */
1462 {chm_nosuch
, 0 }, /* 0x7d */
1463 {chm_nosuch
, 0 }, /* 0x7e */
1464 {chm_nosuch
, 0 }, /* 0x7f */
1466 {chm_nosuch
, 0 }, /* 0x80 */
1467 {chm_nosuch
, 0 }, /* 0x81 */
1468 {chm_nosuch
, 0 }, /* 0x82 */
1469 {chm_nosuch
, 0 }, /* 0x83 */
1470 {chm_nosuch
, 0 }, /* 0x84 */
1471 {chm_nosuch
, 0 }, /* 0x85 */
1472 {chm_nosuch
, 0 }, /* 0x86 */
1473 {chm_nosuch
, 0 }, /* 0x87 */
1474 {chm_nosuch
, 0 }, /* 0x88 */
1475 {chm_nosuch
, 0 }, /* 0x89 */
1476 {chm_nosuch
, 0 }, /* 0x8a */
1477 {chm_nosuch
, 0 }, /* 0x8b */
1478 {chm_nosuch
, 0 }, /* 0x8c */
1479 {chm_nosuch
, 0 }, /* 0x8d */
1480 {chm_nosuch
, 0 }, /* 0x8e */
1481 {chm_nosuch
, 0 }, /* 0x8f */
1483 {chm_nosuch
, 0 }, /* 0x90 */
1484 {chm_nosuch
, 0 }, /* 0x91 */
1485 {chm_nosuch
, 0 }, /* 0x92 */
1486 {chm_nosuch
, 0 }, /* 0x93 */
1487 {chm_nosuch
, 0 }, /* 0x94 */
1488 {chm_nosuch
, 0 }, /* 0x95 */
1489 {chm_nosuch
, 0 }, /* 0x96 */
1490 {chm_nosuch
, 0 }, /* 0x97 */
1491 {chm_nosuch
, 0 }, /* 0x98 */
1492 {chm_nosuch
, 0 }, /* 0x99 */
1493 {chm_nosuch
, 0 }, /* 0x9a */
1494 {chm_nosuch
, 0 }, /* 0x9b */
1495 {chm_nosuch
, 0 }, /* 0x9c */
1496 {chm_nosuch
, 0 }, /* 0x9d */
1497 {chm_nosuch
, 0 }, /* 0x9e */
1498 {chm_nosuch
, 0 }, /* 0x9f */
1500 {chm_nosuch
, 0 }, /* 0xa0 */
1501 {chm_nosuch
, 0 }, /* 0xa1 */
1502 {chm_nosuch
, 0 }, /* 0xa2 */
1503 {chm_nosuch
, 0 }, /* 0xa3 */
1504 {chm_nosuch
, 0 }, /* 0xa4 */
1505 {chm_nosuch
, 0 }, /* 0xa5 */
1506 {chm_nosuch
, 0 }, /* 0xa6 */
1507 {chm_nosuch
, 0 }, /* 0xa7 */
1508 {chm_nosuch
, 0 }, /* 0xa8 */
1509 {chm_nosuch
, 0 }, /* 0xa9 */
1510 {chm_nosuch
, 0 }, /* 0xaa */
1511 {chm_nosuch
, 0 }, /* 0xab */
1512 {chm_nosuch
, 0 }, /* 0xac */
1513 {chm_nosuch
, 0 }, /* 0xad */
1514 {chm_nosuch
, 0 }, /* 0xae */
1515 {chm_nosuch
, 0 }, /* 0xaf */
1517 {chm_nosuch
, 0 }, /* 0xb0 */
1518 {chm_nosuch
, 0 }, /* 0xb1 */
1519 {chm_nosuch
, 0 }, /* 0xb2 */
1520 {chm_nosuch
, 0 }, /* 0xb3 */
1521 {chm_nosuch
, 0 }, /* 0xb4 */
1522 {chm_nosuch
, 0 }, /* 0xb5 */
1523 {chm_nosuch
, 0 }, /* 0xb6 */
1524 {chm_nosuch
, 0 }, /* 0xb7 */
1525 {chm_nosuch
, 0 }, /* 0xb8 */
1526 {chm_nosuch
, 0 }, /* 0xb9 */
1527 {chm_nosuch
, 0 }, /* 0xba */
1528 {chm_nosuch
, 0 }, /* 0xbb */
1529 {chm_nosuch
, 0 }, /* 0xbc */
1530 {chm_nosuch
, 0 }, /* 0xbd */
1531 {chm_nosuch
, 0 }, /* 0xbe */
1532 {chm_nosuch
, 0 }, /* 0xbf */
1534 {chm_nosuch
, 0 }, /* 0xc0 */
1535 {chm_nosuch
, 0 }, /* 0xc1 */
1536 {chm_nosuch
, 0 }, /* 0xc2 */
1537 {chm_nosuch
, 0 }, /* 0xc3 */
1538 {chm_nosuch
, 0 }, /* 0xc4 */
1539 {chm_nosuch
, 0 }, /* 0xc5 */
1540 {chm_nosuch
, 0 }, /* 0xc6 */
1541 {chm_nosuch
, 0 }, /* 0xc7 */
1542 {chm_nosuch
, 0 }, /* 0xc8 */
1543 {chm_nosuch
, 0 }, /* 0xc9 */
1544 {chm_nosuch
, 0 }, /* 0xca */
1545 {chm_nosuch
, 0 }, /* 0xcb */
1546 {chm_nosuch
, 0 }, /* 0xcc */
1547 {chm_nosuch
, 0 }, /* 0xcd */
1548 {chm_nosuch
, 0 }, /* 0xce */
1549 {chm_nosuch
, 0 }, /* 0xcf */
1551 {chm_nosuch
, 0 }, /* 0xd0 */
1552 {chm_nosuch
, 0 }, /* 0xd1 */
1553 {chm_nosuch
, 0 }, /* 0xd2 */
1554 {chm_nosuch
, 0 }, /* 0xd3 */
1555 {chm_nosuch
, 0 }, /* 0xd4 */
1556 {chm_nosuch
, 0 }, /* 0xd5 */
1557 {chm_nosuch
, 0 }, /* 0xd6 */
1558 {chm_nosuch
, 0 }, /* 0xd7 */
1559 {chm_nosuch
, 0 }, /* 0xd8 */
1560 {chm_nosuch
, 0 }, /* 0xd9 */
1561 {chm_nosuch
, 0 }, /* 0xda */
1562 {chm_nosuch
, 0 }, /* 0xdb */
1563 {chm_nosuch
, 0 }, /* 0xdc */
1564 {chm_nosuch
, 0 }, /* 0xdd */
1565 {chm_nosuch
, 0 }, /* 0xde */
1566 {chm_nosuch
, 0 }, /* 0xdf */
1568 {chm_nosuch
, 0 }, /* 0xe0 */
1569 {chm_nosuch
, 0 }, /* 0xe1 */
1570 {chm_nosuch
, 0 }, /* 0xe2 */
1571 {chm_nosuch
, 0 }, /* 0xe3 */
1572 {chm_nosuch
, 0 }, /* 0xe4 */
1573 {chm_nosuch
, 0 }, /* 0xe5 */
1574 {chm_nosuch
, 0 }, /* 0xe6 */
1575 {chm_nosuch
, 0 }, /* 0xe7 */
1576 {chm_nosuch
, 0 }, /* 0xe8 */
1577 {chm_nosuch
, 0 }, /* 0xe9 */
1578 {chm_nosuch
, 0 }, /* 0xea */
1579 {chm_nosuch
, 0 }, /* 0xeb */
1580 {chm_nosuch
, 0 }, /* 0xec */
1581 {chm_nosuch
, 0 }, /* 0xed */
1582 {chm_nosuch
, 0 }, /* 0xee */
1583 {chm_nosuch
, 0 }, /* 0xef */
1585 {chm_nosuch
, 0 }, /* 0xf0 */
1586 {chm_nosuch
, 0 }, /* 0xf1 */
1587 {chm_nosuch
, 0 }, /* 0xf2 */
1588 {chm_nosuch
, 0 }, /* 0xf3 */
1589 {chm_nosuch
, 0 }, /* 0xf4 */
1590 {chm_nosuch
, 0 }, /* 0xf5 */
1591 {chm_nosuch
, 0 }, /* 0xf6 */
1592 {chm_nosuch
, 0 }, /* 0xf7 */
1593 {chm_nosuch
, 0 }, /* 0xf8 */
1594 {chm_nosuch
, 0 }, /* 0xf9 */
1595 {chm_nosuch
, 0 }, /* 0xfa */
1596 {chm_nosuch
, 0 }, /* 0xfb */
1597 {chm_nosuch
, 0 }, /* 0xfc */
1598 {chm_nosuch
, 0 }, /* 0xfd */
1599 {chm_nosuch
, 0 }, /* 0xfe */
1600 {chm_nosuch
, 0 }, /* 0xff */
1605 /* set_channel_mode()
1607 * inputs - client, source, channel, membership pointer, params
1609 * side effects - channel modes/memberships are changed, MODE is issued
1611 * Extensively modified to be hotpluggable, 03/09/06 -- nenolod
1614 set_channel_mode(struct Client
*client_p
, struct Client
*source_p
,
1615 struct Channel
*chptr
, struct membership
*msptr
, int parc
, const char *parv
[])
1617 static char modebuf
[BUFSIZE
];
1618 static char parabuf
[BUFSIZE
];
1621 int cur_len
, mlen
, paralen
, paracount
, arglen
, len
;
1627 const char *ml
= parv
[0];
1629 struct Client
*fakesource_p
;
1634 mode_limit_simple
= 0;
1636 alevel
= get_channel_access(source_p
, msptr
);
1638 /* Hide connecting server on netburst -- jilles */
1639 if (ConfigServerHide
.flatten_links
&& IsServer(source_p
) && !has_id(source_p
) && !HasSentEob(source_p
))
1642 fakesource_p
= source_p
;
1644 for(; (c
= *ml
) != 0; ml
++)
1658 chmode_table
[(unsigned char) c
].set_func(fakesource_p
, chptr
, alevel
,
1661 chmode_table
[(unsigned char) c
].mode_type
);
1666 /* bail out if we have nothing to do... */
1670 if(IsServer(source_p
))
1671 mlen
= rb_sprintf(modebuf
, ":%s MODE %s ", fakesource_p
->name
, chptr
->chname
);
1673 mlen
= rb_sprintf(modebuf
, ":%s!%s@%s MODE %s ",
1674 source_p
->name
, source_p
->username
,
1675 source_p
->host
, chptr
->chname
);
1677 for(j
= 0, flags
= ALL_MEMBERS
; j
< 2; j
++, flags
= ONLY_CHANOPS
)
1680 mbuf
= modebuf
+ mlen
;
1683 paracount
= paralen
= 0;
1686 for(i
= 0; i
< mode_count
; i
++)
1688 if(mode_changes
[i
].letter
== 0 || mode_changes
[i
].mems
!= flags
)
1691 if(mode_changes
[i
].arg
!= NULL
)
1693 arglen
= strlen(mode_changes
[i
].arg
);
1695 if(arglen
> MODEBUFLEN
- 5)
1701 /* if we're creeping over MAXMODEPARAMSSERV, or over
1702 * bufsize (4 == +/-,modechar,two spaces) send now.
1704 if(mode_changes
[i
].arg
!= NULL
&&
1705 ((paracount
== MAXMODEPARAMSSERV
) ||
1706 ((cur_len
+ paralen
+ arglen
+ 4) > (BUFSIZE
- 3))))
1711 sendto_channel_local(flags
, chptr
, "%s %s", modebuf
,
1716 paracount
= paralen
= 0;
1718 mbuf
= modebuf
+ mlen
;
1724 if(dir
!= mode_changes
[i
].dir
)
1726 *mbuf
++ = (mode_changes
[i
].dir
== MODE_ADD
) ? '+' : '-';
1728 dir
= mode_changes
[i
].dir
;
1731 *mbuf
++ = mode_changes
[i
].letter
;
1734 if(mode_changes
[i
].arg
!= NULL
)
1737 len
= rb_sprintf(pbuf
, "%s ", mode_changes
[i
].arg
);
1743 if(paralen
&& parabuf
[paralen
- 1] == ' ')
1744 parabuf
[paralen
- 1] = '\0';
1748 sendto_channel_local(flags
, chptr
, "%s %s", modebuf
, parabuf
);
1751 /* only propagate modes originating locally, or if we're hubbing */
1752 if(MyClient(source_p
) || rb_dlink_list_length(&serv_list
) > 1)
1753 send_cap_mode_changes(client_p
, source_p
, chptr
, mode_changes
, mode_count
);