ViewVC Help
View File | Revision Log | Show Annotations | View Changeset | Root Listing
root/svn/ircd-hybrid/trunk/src/parse.c
Revision: 7006
Committed: Fri Jan 1 00:07:54 2016 UTC (9 years, 7 months ago) by michael
Content type: text/x-csrc
File size: 18803 byte(s)
Log Message:
- Update copyright years

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 7006 * Copyright (c) 1997-2016 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 4565 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
19 adx 30 * USA
20     */
21    
22 michael 2916 /*! \file parse.c
23     * \brief The message parser.
24     * \version $Id$
25     */
26    
27 adx 30 #include "stdinc.h"
28 michael 1243 #include "client.h"
29 adx 30 #include "parse.h"
30     #include "channel.h"
31     #include "hash.h"
32 michael 6161 #include "id.h"
33 adx 30 #include "irc_string.h"
34     #include "ircd.h"
35     #include "numeric.h"
36 michael 1309 #include "log.h"
37 adx 30 #include "send.h"
38 michael 1309 #include "conf.h"
39 adx 30 #include "memory.h"
40 michael 3347 #include "user.h"
41     #include "server.h"
42 adx 30
43 michael 2916
44 adx 30 /*
45     * (based on orabidoo's parser code)
46     *
47     * This has always just been a trie. Look at volume III of Knuth ACP
48     *
49     *
50     * ok, you start out with an array of pointers, each one corresponds
51     * to a letter at the current position in the command being examined.
52     *
53     * so roughly you have this for matching 'trie' or 'tie'
54     *
55     * 't' points -> [MessageTree *] 'r' -> [MessageTree *] -> 'i'
56     * -> [MessageTree *] -> [MessageTree *] -> 'e' and matches
57     *
58 michael 2916 * 'i' -> [MessageTree *] -> 'e' and matches
59 adx 30 *
60     * BUGS (Limitations!)
61 michael 2345 *
62 adx 30 * I designed this trie to parse ircd commands. Hence it currently
63     * casefolds. This is trivial to fix by increasing MAXPTRLEN.
64     * This trie also "folds" '{' etc. down. This means, the input to this
65     * trie must be alpha tokens only. This again, is a limitation that
66     * can be overcome by increasing MAXPTRLEN to include upper/lower case
67     * at the expense of more memory. At the extreme end, you could make
68     * MAXPTRLEN 128.
69     *
70     * This is also not a patricia trie. On short ircd tokens, this is
71 michael 2345 * not likely going to matter.
72 adx 30 *
73     * Diane Bruce (Dianora), June 6 2003
74     */
75    
76 michael 4231 /*
77     * Must be a power of 2, and larger than 26 [a-z]|[A-Z]. It's used to allocate
78     * the set of pointers at each node of the tree.
79     * There are MAXPTRLEN pointers at each node. Obviously, there have to be more
80     * pointers than ASCII letters. 32 is a nice number since there is then no need
81     * to shift 'A'/'a' to base 0 index, at the expense of a few never used
82     * pointers.
83     * For a small parser like this, this is a good compromise and does
84     * make it somewhat faster. - Dianora
85     */
86 michael 6350 enum { MAXPTRLEN = 32 };
87 adx 30
88 michael 4231
89     static struct MessageTree
90 adx 30 {
91 michael 2345 int links; /* Count of all pointers (including msg) at this node
92 michael 2182 * used as reference count for deletion of _this_ node.
93     */
94 adx 30 struct Message *msg;
95     struct MessageTree *pointers[MAXPTRLEN];
96 michael 4231 } msg_tree;
97 adx 30
98    
99 michael 4231 /* remove_unknown()
100     *
101     * inputs -
102     * output -
103     * side effects -
104     */
105     static void
106 michael 4552 parse_remove_unknown(struct Client *client_p, const char *lsender, char *lbuffer)
107 michael 4231 {
108     /*
109     * Do kill if it came from a server because it means there is a ghost
110     * user on the other server which needs to be removed. -avalon
111     * Tell opers about this. -Taner
112     */
113     /*
114     * '[0-9]something' is an ID (KILL/SQUIT depending on its length)
115     * 'nodots' is a nickname (KILL)
116     * 'no.dot.at.start' is a server (SQUIT)
117     */
118     if ((IsDigit(*lsender) && strlen(lsender) <= IRC_MAXSID) || strchr(lsender, '.'))
119     {
120     sendto_realops_flags(UMODE_DEBUG, L_ADMIN, SEND_NOTICE,
121     "Unknown prefix (%s) from %s, Squitting %s",
122     lbuffer, get_client_name(client_p, SHOW_IP), lsender);
123     sendto_realops_flags(UMODE_DEBUG, L_OPER, SEND_NOTICE,
124     "Unknown prefix (%s) from %s, Squitting %s",
125 michael 6437 lbuffer, get_client_name(client_p, MASK_IP), lsender);
126 michael 4231 sendto_one(client_p, ":%s SQUIT %s :(Unknown prefix (%s) from %s)",
127     me.id, lsender, lbuffer, client_p->name);
128     }
129     else
130     sendto_one(client_p, ":%s KILL %s :%s (Unknown Client)",
131     me.id, lsender, me.name);
132     }
133 adx 30
134     /*
135 michael 4231 *
136     * parc number of arguments ('sender' counted as one!)
137     * parv[0] pointer to 'sender' (may point to empty string) (not used)
138     * parv[1]..parv[parc-1]
139     * pointers to additional parameters, this is a NULL
140     * terminated list (parv[parc] == NULL).
141     *
142     * *WARNING*
143     * Numerics are mostly error reports. If there is something
144     * wrong with the message, just *DROP* it! Don't even think of
145     * sending back a neat error message -- big danger of creating
146     * a ping pong error message...
147     *
148     * Rewritten by Nemesi, Jan 1999, to support numeric nicks in parv[1]
149     *
150     * Called when we get a numeric message from a remote _server_ and we are
151     * supposed to forward it somewhere. Note that we always ignore numerics sent
152     * to 'me' and simply drop the message if we can't handle with this properly:
153     * the savvy approach is NEVER generate an error in response to an... error :)
154     */
155     static void
156     parse_handle_numeric(unsigned int numeric, struct Client *source_p, int parc, char *parv[])
157     {
158     struct Client *target_p = NULL;
159     struct Channel *chptr = NULL;
160    
161     /*
162     * Avoid trash, we need it to come from a server and have a target
163     */
164     if (parc < 2 || !IsServer(source_p))
165     return;
166    
167     /*
168     * Who should receive this message ? Will we do something with it ?
169     * Note that we use findUser functions, so the target can't be neither
170     * a server, nor a channel (?) nor a list of targets (?) .. u2.10
171     * should never generate numeric replies to non-users anyway
172     * Ahem... it can be a channel actually, csc bots use it :\ --Nem
173     */
174     if (IsChanPrefix(*parv[1]))
175     chptr = hash_find_channel(parv[1]);
176     else
177     target_p = find_person(source_p, parv[1]);
178    
179 michael 4959 if ((!target_p || target_p->from == source_p->from) && !chptr)
180 michael 4231 return;
181    
182     /*
183     * Remap low number numerics, not that I understand WHY.. --Nemesi
184     */
185     /*
186     * Numerics below 100 talk about the current 'connection', you're not
187     * connected to a remote server so it doesn't make sense to send them
188     * remotely - but the information they contain may be useful, so we
189     * remap them up. Weird, but true. -- Isomer
190     */
191     if (numeric < 100)
192     numeric += 100;
193    
194     if (target_p)
195     {
196     /* Fake it for server hiding, if it's our client */
197     if ((ConfigServerHide.hide_servers || IsHidden(source_p)) && MyConnect(target_p) &&
198     !HasUMode(target_p, UMODE_OPER))
199 michael 5583 sendto_one_numeric(target_p, &me, numeric | SND_EXPLICIT, "%s", parv[2]);
200 michael 4231 else
201 michael 5583 sendto_one_numeric(target_p, source_p, numeric | SND_EXPLICIT, "%s", parv[2]);
202 michael 4231 }
203     else
204     sendto_channel_butone(source_p, source_p, chptr, 0, "%u %s %s",
205 michael 4618 numeric, chptr->name, parv[2]);
206 michael 4231 }
207    
208     /* handle_command()
209     *
210     * inputs - pointer to message block
211     * - pointer to client
212     * - pointer to client message is from
213     * - count of number of args
214     * - pointer to argv[] array
215     * output - -1 if error from server
216     * side effects -
217     */
218     static void
219 michael 4937 parse_handle_command(struct Message *message, struct Client *source_p,
220 michael 4231 unsigned int i, char *para[])
221     {
222     if (IsServer(source_p->from))
223 michael 4937 ++message->rcount;
224 michael 4231
225 michael 4937 ++message->count;
226 michael 4231
227     /* Check right amount of parameters is passed... --is */
228 michael 4937 if (i < message->args_min)
229     sendto_one_numeric(source_p, &me, ERR_NEEDMOREPARAMS, message->cmd);
230 michael 4231 else
231 michael 4937 message->handlers[source_p->from->handler](source_p, i, para);
232 michael 4231 }
233    
234     /*
235 adx 30 * parse a buffer.
236     *
237     * NOTE: parse() should not be called recusively by any other functions!
238     */
239     void
240     parse(struct Client *client_p, char *pbuffer, char *bufend)
241     {
242     struct Client *from = client_p;
243 michael 4937 struct Message *message = NULL;
244 michael 3638 char *para[MAXPARA + 2]; /* <command> + <parameters> + NULL */
245 michael 1246 char *ch = NULL;
246     char *s = NULL;
247 michael 3573 unsigned int numeric = 0;
248 michael 1178 unsigned int parc = 0;
249     unsigned int paramcount;
250 adx 30
251     if (IsDefunct(client_p))
252     return;
253    
254 michael 4588 assert(client_p->connection->fd.flags.open);
255 michael 2786 assert((bufend - pbuffer) < IRCD_BUFSIZE);
256 adx 30
257 michael 3628 for (ch = pbuffer; *ch == ' '; ++ch) /* Skip spaces */
258     ;
259 adx 30
260     if (*ch == ':')
261     {
262 michael 1178 /*
263     * Copy the prefix to 'sender' assuming it terminates
264 adx 30 * with SPACE (or NULL, which is an error, though).
265     */
266 michael 4833 const char *const sender = ++ch;
267 adx 30
268 michael 3235 if ((s = strchr(ch, ' ')))
269 adx 30 {
270     *s = '\0';
271 michael 1178 ch = ++s;
272 adx 30 }
273    
274     if (*sender && IsServer(client_p))
275     {
276 michael 4554 if ((from = hash_find_id(sender)) == NULL)
277     from = hash_find_client(sender);
278 adx 30
279 michael 2345 /*
280     * Hmm! If the client corresponding to the prefix is not found--what is
281     * the correct action??? Now, I will ignore the message (old IRC just
282     * let it through as if the prefix just wasn't there...) --msa
283 adx 30 */
284     if (from == NULL)
285     {
286 michael 896 ++ServerStats.is_unpf;
287 michael 4231 parse_remove_unknown(client_p, sender, pbuffer);
288 adx 30 return;
289     }
290    
291     if (from->from != client_p)
292     {
293 michael 896 ++ServerStats.is_wrdi;
294 michael 4926 sendto_realops_flags(UMODE_DEBUG, L_ADMIN, SEND_NOTICE,
295     "Fake direction: dropped message from %s[%s] via %s",
296     from->name, from->from->name,
297     get_client_name(client_p, SHOW_IP));
298     sendto_realops_flags(UMODE_DEBUG, L_OPER, SEND_NOTICE,
299 michael 4928 "Fake direction: dropped message from %s[%s] via %s",
300 michael 4926 from->name, from->from->name,
301     get_client_name(client_p, MASK_IP));
302 adx 30 return;
303     }
304     }
305    
306     while (*ch == ' ')
307 michael 1178 ++ch;
308 adx 30 }
309    
310     if (*ch == '\0')
311     {
312 michael 896 ++ServerStats.is_empt;
313 adx 30 return;
314     }
315    
316 michael 2345 /*
317     * Extract the command code from the packet. Point s to the end
318 adx 30 * of the command code and calculate the length using pointer
319 michael 2345 * arithmetic. Note: only need length for numerics and *all*
320 adx 30 * numerics must have parameters and thus a space after the command
321     * code. -avalon
322     */
323    
324 michael 3628 /* EOB is 3 characters long but is not a numeric */
325 michael 4298 if (*(ch + 3) == ' ' && /* Ok, let's see if it's a possible numeric */
326 adx 30 IsDigit(*ch) && IsDigit(*(ch + 1)) && IsDigit(*(ch + 2)))
327     {
328 michael 3573 numeric = (*ch - '0') * 100 + (*(ch + 1) - '0') * 10 + (*(ch + 2) - '0');
329 michael 3628 paramcount = 2; /* Destination, and the rest of it */
330 michael 896 ++ServerStats.is_num;
331 michael 3628 s = ch + 3; /* I know this is ' ' from above if */
332     *s++ = '\0'; /* Blow away the ' ', and point s to next part */
333 adx 30 }
334     else
335 michael 2345 {
336 michael 4760 unsigned int length = 0;
337 adx 30
338 michael 3235 if ((s = strchr(ch, ' ')))
339 adx 30 *s++ = '\0';
340    
341 michael 4937 if ((message = find_command(ch)) == NULL)
342 adx 30 {
343 michael 2916 /*
344 michael 2345 * Note: Give error message *only* to recognized
345 adx 30 * persons. It's a nightmare situation to have
346     * two programs sending "Unknown command"'s or
347     * equivalent to each other at full blast....
348     * If it has got to person state, it at least
349     * seems to be well behaving. Perhaps this message
350     * should never be generated, though... --msa
351     * Hm, when is the buffer empty -- if a command
352     * code has been found ?? -Armin
353     */
354 michael 3335 if (*pbuffer)
355 adx 30 if (IsClient(from))
356 michael 3109 sendto_one_numeric(from, &me, ERR_UNKNOWNCOMMAND, ch);
357 adx 30
358 michael 896 ++ServerStats.is_unco;
359 adx 30 return;
360     }
361    
362 michael 4937 assert(message->cmd);
363 adx 30
364 michael 4937 paramcount = message->args_max;
365 michael 4760 length = bufend - ((s) ? s : ch);
366 michael 4937 message->bytes += length;
367 adx 30 }
368    
369 michael 1178 /*
370     * Must the following loop really be so devious? On surface it
371     * splits the message to parameters from blank spaces. But, if
372     * paramcount has been reached, the rest of the message goes into
373     * this last parameter (about same effect as ":" has...) --msa
374     */
375    
376 michael 3250 /* Note initially true: s == NULL || *(s - 1) == '\0' !! */
377 michael 1178
378 michael 3096 para[parc] = ch;
379 michael 1178
380 michael 4937 if (message && (message->flags & MFLG_EXTRA))
381 michael 4545 {
382     /*
383     * XXX: This will have to go away after the command handler rewrite
384     */
385 michael 4937 para[++parc] = message->extra;
386 michael 4545 }
387    
388 michael 1178 if (s)
389 adx 30 {
390 michael 1178 if (paramcount > MAXPARA)
391     paramcount = MAXPARA;
392    
393     while (1)
394     {
395     while (*s == ' ')
396     *s++ = '\0';
397    
398     if (*s == '\0')
399     break;
400    
401     if (*s == ':')
402     {
403 michael 4747 /* The rest is single parameter--can include blanks also. */
404 michael 3250 para[++parc] = s + (!numeric); /* Keep the colon if it's a numeric */
405 michael 1178 break;
406     }
407    
408     para[++parc] = s;
409    
410     if (parc >= paramcount)
411     break;
412    
413     while (*s && *s != ' ')
414     ++s;
415     }
416 adx 30 }
417    
418 michael 1178 para[++parc] = NULL;
419    
420 michael 4937 if (message)
421     parse_handle_command(message, from, parc, para);
422 adx 30 else
423 michael 4231 parse_handle_numeric(numeric, from, parc, para);
424 adx 30 }
425    
426     /* add_msg_element()
427     *
428     * inputs - pointer to MessageTree
429     * - pointer to Message to add for given command
430     * - pointer to current portion of command being added
431     * output - NONE
432     * side effects - recursively build the Message Tree ;-)
433     */
434     /*
435     * How this works.
436     *
437     * The code first checks to see if its reached the end of the command
438     * If so, that struct MessageTree has a msg pointer updated and the links
439     * count incremented, since a msg pointer is a reference.
440     * Then the code descends recursively, building the trie.
441     * If a pointer index inside the struct MessageTree is NULL a new
442     * child struct MessageTree has to be allocated.
443     * The links (reference count) is incremented as they are created
444     * in the parent.
445     */
446     static void
447 michael 1570 add_msg_element(struct MessageTree *mtree_p, struct Message *msg_p,
448     const char *cmd)
449 adx 30 {
450 michael 3624 struct MessageTree *ntree_p = NULL;
451 adx 30
452     if (*cmd == '\0')
453     {
454     mtree_p->msg = msg_p;
455 michael 1570 mtree_p->links++; /* Have msg pointer, so up ref count */
456 adx 30 }
457     else
458     {
459 michael 2345 /*
460     * *cmd & (MAXPTRLEN-1)
461 adx 30 * convert the char pointed to at *cmd from ASCII to an integer
462     * between 0 and MAXPTRLEN.
463     * Thus 'A' -> 0x1 'B' -> 0x2 'c' -> 0x3 etc.
464     */
465 michael 1013 if ((ntree_p = mtree_p->pointers[*cmd & (MAXPTRLEN - 1)]) == NULL)
466 adx 30 {
467 michael 3504 ntree_p = MyCalloc(sizeof(struct MessageTree));
468 michael 1013 mtree_p->pointers[*cmd & (MAXPTRLEN - 1)] = ntree_p;
469 adx 30
470 michael 1570 mtree_p->links++; /* Have new pointer, so up ref count */
471 adx 30 }
472 michael 896
473 michael 1013 add_msg_element(ntree_p, msg_p, cmd + 1);
474 adx 30 }
475     }
476    
477     /* del_msg_element()
478     *
479     * inputs - Pointer to MessageTree to delete from
480     * - pointer to command name to delete
481     * output - NONE
482     * side effects - recursively deletes a token from the Message Tree ;-)
483     */
484     /*
485     * How this works.
486     *
487     * Well, first off, the code recursively descends into the trie
488     * until it finds the terminating letter of the command being removed.
489     * Once it has done that, it marks the msg pointer as NULL then
490     * reduces the reference count on that allocated struct MessageTree
491     * since a command counts as a reference.
492     *
493     * Then it pops up the recurse stack. As it comes back up the recurse
494     * The code checks to see if the child now has no pointers or msg
495 michael 4298 * i.e. the links count has gone to zero. If it's no longer used, the
496 adx 30 * child struct MessageTree can be deleted. The parent reference
497     * to this child is then removed and the parents link count goes down.
498     * Thus, we continue to go back up removing all unused MessageTree(s)
499     */
500     static void
501     del_msg_element(struct MessageTree *mtree_p, const char *cmd)
502     {
503 michael 3624 struct MessageTree *ntree_p = NULL;
504 adx 30
505 michael 1246 /*
506     * In case this is called for a nonexistent command
507 adx 30 * check that there is a msg pointer here, else links-- goes -ve
508     * -db
509     */
510 michael 3235 if (*cmd == '\0' && mtree_p->msg)
511 adx 30 {
512     mtree_p->msg = NULL;
513     mtree_p->links--;
514     }
515     else
516     {
517 michael 3235 if ((ntree_p = mtree_p->pointers[*cmd & (MAXPTRLEN - 1)]))
518 adx 30 {
519 michael 1013 del_msg_element(ntree_p, cmd + 1);
520 michael 896
521 adx 30 if (ntree_p->links == 0)
522     {
523 michael 1570 mtree_p->pointers[*cmd & (MAXPTRLEN - 1)] = NULL;
524     mtree_p->links--;
525     MyFree(ntree_p);
526 adx 30 }
527     }
528     }
529     }
530    
531     /* msg_tree_parse()
532     *
533     * inputs - Pointer to command to find
534     * - Pointer to MessageTree root
535     * output - Find given command returning Message * if found NULL if not
536     * side effects - none
537     */
538     static struct Message *
539 michael 1427 msg_tree_parse(const char *cmd)
540 adx 30 {
541 michael 1427 struct MessageTree *mtree = &msg_tree;
542 michael 3250
543 michael 5777 assert(!EmptyString(cmd));
544 adx 30
545 michael 522 while (IsAlpha(*cmd) && (mtree = mtree->pointers[*cmd & (MAXPTRLEN - 1)]))
546     if (*++cmd == '\0')
547     return mtree->msg;
548 adx 30
549 michael 522 return NULL;
550 adx 30 }
551    
552     /* mod_add_cmd()
553     *
554     * inputs - pointer to struct Message
555     * output - none
556     * side effects - load this one command name
557     */
558     void
559     mod_add_cmd(struct Message *msg)
560     {
561 michael 6879 assert(msg);
562     assert(msg->cmd);
563 adx 30
564 michael 3250 /* Command already added? */
565 michael 1427 if (msg_tree_parse(msg->cmd))
566 adx 30 return;
567    
568     add_msg_element(&msg_tree, msg, msg->cmd);
569     }
570    
571     /* mod_del_cmd()
572     *
573     * inputs - pointer to struct Message
574     * output - none
575     * side effects - unload this one command name
576     */
577     void
578     mod_del_cmd(struct Message *msg)
579     {
580 michael 6879 assert(msg);
581     assert(msg->cmd);
582 adx 30
583 michael 3614 if (!msg_tree_parse(msg->cmd))
584     return;
585    
586 adx 30 del_msg_element(&msg_tree, msg->cmd);
587     }
588    
589     /* find_command()
590     *
591     * inputs - command name
592     * output - pointer to struct Message
593     * side effects - none
594     */
595     struct Message *
596     find_command(const char *cmd)
597     {
598 michael 1427 return msg_tree_parse(cmd);
599 adx 30 }
600    
601 michael 1246 static void
602     recurse_report_messages(struct Client *source_p, const struct MessageTree *mtree)
603     {
604 michael 3235 if (mtree->msg)
605 michael 3109 sendto_one_numeric(source_p, &me, RPL_STATSCOMMANDS,
606     mtree->msg->cmd,
607     mtree->msg->count, mtree->msg->bytes,
608     mtree->msg->rcount);
609 michael 1246
610 michael 3235 for (unsigned int i = 0; i < MAXPTRLEN; ++i)
611     if (mtree->pointers[i])
612 michael 1246 recurse_report_messages(source_p, mtree->pointers[i]);
613     }
614    
615 adx 30 /* report_messages()
616     *
617     * inputs - pointer to client to report to
618     * output - NONE
619     * side effects - client is shown list of commands
620     */
621     void
622     report_messages(struct Client *source_p)
623     {
624 michael 4833 const struct MessageTree *const mtree = &msg_tree;
625 adx 30
626 michael 3235 for (unsigned int i = 0; i < MAXPTRLEN; ++i)
627     if (mtree->pointers[i])
628 adx 30 recurse_report_messages(source_p, mtree->pointers[i]);
629     }
630    
631     /* m_not_oper()
632 michael 2345 * inputs -
633 adx 30 * output -
634     * side effects - just returns a nastyogram to given user
635     */
636 michael 2820 int
637 michael 3156 m_not_oper(struct Client *source_p, int parc, char *parv[])
638 adx 30 {
639 michael 3109 sendto_one_numeric(source_p, &me, ERR_NOPRIVILEGES);
640 michael 2820 return 0;
641 adx 30 }
642    
643 michael 2820 int
644 michael 3156 m_unregistered(struct Client *source_p, int parc, char *parv[])
645 adx 30 {
646 michael 3109 sendto_one_numeric(source_p, &me, ERR_NOTREGISTERED);
647 michael 2820 return 0;
648 adx 30 }
649    
650 michael 2820 int
651 michael 3156 m_registered(struct Client *source_p, int parc, char *parv[])
652 adx 30 {
653 michael 3109 sendto_one_numeric(source_p, &me, ERR_ALREADYREGISTRED);
654 michael 2820 return 0;
655 adx 30 }
656    
657 michael 2820 int
658 michael 3156 m_ignore(struct Client *source_p, int parc, char *parv[])
659 adx 30 {
660 michael 2820 return 0;
661 adx 30 }

Properties

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