]> jfr.im git - irc/quakenet/snircd-patchqueue.git/blobdiff - split.patch
resolved all patch conflicts and removed all guards, except for hidebanowner.patch
[irc/quakenet/snircd-patchqueue.git] / split.patch
index dc82c169e377c6d0f692326cca7d24f71a4c3f00..2975115950bacdb8a0693644a68bc0e756fdf14a 100644 (file)
@@ -7,9 +7,9 @@ Add split.c split.h m_split.c
 
 NOTE: feature SPLIT is default enabled as that is just easier when working on this patch.
 
-diff -r bcffa56c93d4 doc/api/log.txt
---- a/doc/api/log.txt  Fri Jan 23 18:45:25 2009 +0100
-+++ b/doc/api/log.txt  Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d doc/api/log.txt
+--- a/doc/api/log.txt  Tue Jan 27 18:19:11 2009 +0100
++++ b/doc/api/log.txt  Tue Jan 27 18:25:58 2009 +0100
 @@ -41,7 +41,7 @@
  
  <enum>
@@ -19,9 +19,9 @@ diff -r bcffa56c93d4 doc/api/log.txt
    LS_OPERKILL, LS_SERVKILL, LS_USER, LS_OPER, LS_RESOLVER, LS_SOCKET,
    LS_DEBUG, LS_OLDLOG,
    LS_LAST_SYSTEM
-diff -r bcffa56c93d4 include/handlers.h
---- a/include/handlers.h       Fri Jan 23 18:45:25 2009 +0100
-+++ b/include/handlers.h       Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d include/handlers.h
+--- a/include/handlers.h       Tue Jan 27 18:19:11 2009 +0100
++++ b/include/handlers.h       Tue Jan 27 18:25:58 2009 +0100
 @@ -139,6 +139,7 @@
  extern int m_registered(struct Client*, struct Client*, int, char*[]);
  extern int m_sethost(struct Client*, struct Client*, int, char*[]);
@@ -30,7 +30,7 @@ diff -r bcffa56c93d4 include/handlers.h
  extern int m_stats(struct Client*, struct Client*, int, char*[]);
  extern int m_time(struct Client*, struct Client*, int, char*[]);
  extern int m_topic(struct Client*, struct Client*, int, char*[]);
-@@ -178,6 +179,7 @@
+@@ -179,6 +180,7 @@
  extern int mo_rping(struct Client*, struct Client*, int, char*[]);
  extern int mo_set(struct Client*, struct Client*, int, char*[]);
  extern int mo_settime(struct Client*, struct Client*, int, char*[]);
@@ -38,7 +38,7 @@ diff -r bcffa56c93d4 include/handlers.h
  extern int mo_squit(struct Client*, struct Client*, int, char*[]);
  extern int mo_stats(struct Client*, struct Client*, int, char*[]);
  extern int mo_trace(struct Client*, struct Client*, int, char*[]);
-@@ -232,6 +234,7 @@
+@@ -234,6 +236,7 @@
  extern int ms_sethost(struct Client*, struct Client*, int, char*[]);
  extern int ms_settime(struct Client*, struct Client*, int, char*[]);
  extern int ms_silence(struct Client*, struct Client*, int, char*[]);
@@ -46,10 +46,10 @@ diff -r bcffa56c93d4 include/handlers.h
  extern int ms_squit(struct Client*, struct Client*, int, char*[]);
  extern int ms_stats(struct Client*, struct Client*, int, char*[]);
  extern int ms_topic(struct Client*, struct Client*, int, char*[]);
-diff -r bcffa56c93d4 include/ircd_features.h
---- a/include/ircd_features.h  Fri Jan 23 18:45:25 2009 +0100
-+++ b/include/ircd_features.h  Fri Jan 23 23:04:58 2009 +0100
-@@ -111,6 +111,10 @@
+diff -r 2267c134dc0d include/ircd_features.h
+--- a/include/ircd_features.h  Tue Jan 27 18:19:11 2009 +0100
++++ b/include/ircd_features.h  Tue Jan 27 18:25:58 2009 +0100
+@@ -112,6 +112,10 @@
    FEAT_SETHOST_USER,
    FEAT_SETHOST_AUTO,
  
@@ -60,7 +60,7 @@ diff -r bcffa56c93d4 include/ircd_features.h
    /* HEAD_IN_SAND Features */
    FEAT_HIS_SNOTICES,
    FEAT_HIS_SNOTICES_OPER_ONLY,
-@@ -138,6 +142,7 @@
+@@ -139,6 +143,7 @@
    FEAT_HIS_STATS_q,
    FEAT_HIS_STATS_R,
    FEAT_HIS_STATS_r,
@@ -68,9 +68,9 @@ diff -r bcffa56c93d4 include/ircd_features.h
    FEAT_HIS_STATS_s,
    FEAT_HIS_STATS_t,
    FEAT_HIS_STATS_T,
-diff -r bcffa56c93d4 include/ircd_log.h
---- a/include/ircd_log.h       Fri Jan 23 18:45:25 2009 +0100
-+++ b/include/ircd_log.h       Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d include/ircd_log.h
+--- a/include/ircd_log.h       Tue Jan 27 18:19:11 2009 +0100
++++ b/include/ircd_log.h       Tue Jan 27 18:25:58 2009 +0100
 @@ -55,6 +55,7 @@
    LS_OPERMODE,   /**< Uses of OPMODE, CLEARMODE< etc. */
    LS_GLINE,      /**< Adding, (de-)activating or removing GLINEs. */
@@ -79,10 +79,10 @@ diff -r bcffa56c93d4 include/ircd_log.h
    LS_WHO,        /**< Use of extended WHO privileges. */
    LS_NETWORK,    /**< New server connections. */
    LS_OPERKILL,   /**< Kills by IRC operators. */
-diff -r bcffa56c93d4 include/msg.h
---- a/include/msg.h    Fri Jan 23 18:45:25 2009 +0100
-+++ b/include/msg.h    Fri Jan 23 23:04:58 2009 +0100
-@@ -332,6 +332,10 @@
+diff -r 2267c134dc0d include/msg.h
+--- a/include/msg.h    Tue Jan 27 18:19:11 2009 +0100
++++ b/include/msg.h    Tue Jan 27 18:25:58 2009 +0100
+@@ -336,6 +336,10 @@
  #define TOK_JUPE                "JU"
  #define CMD_JUPE              MSG_JUPE, TOK_JUPE
  
@@ -93,18 +93,18 @@ diff -r bcffa56c93d4 include/msg.h
  #define MSG_OPMODE              "OPMODE"        /* OPMO */
  #define TOK_OPMODE              "OM"
  #define CMD_OPMODE            MSG_OPMODE, TOK_OPMODE
-diff -r bcffa56c93d4 include/numeric.h
---- a/include/numeric.h        Fri Jan 23 18:45:25 2009 +0100
-+++ b/include/numeric.h        Fri Jan 23 23:04:58 2009 +0100
-@@ -117,6 +117,7 @@
-       RPL_STATSVLINE       227           unreal */
+diff -r 2267c134dc0d include/numeric.h
+--- a/include/numeric.h        Tue Jan 27 18:19:11 2009 +0100
++++ b/include/numeric.h        Tue Jan 27 18:25:58 2009 +0100
+@@ -118,6 +118,7 @@
  #define RPL_STATSALINE       226        /* Hybrid, Undernet */
+ #define RPL_STATSWELCOME     227        /* QuakeNet extension */
  #define RPL_STATSQLINE       228        /* Undernet extension */
 +#define RPL_STATSSPLIT       229        /* QuakeNet extension */
  #define RPL_STATSHEADER      230        /* QuakeNet extension */
  
  /*      RPL_SERVICEINFO      231      unused */
-@@ -178,6 +179,8 @@
+@@ -179,6 +180,8 @@
  #define RPL_STATSDLINE       275        /* Undernet extension */
  #define RPL_STATSRLINE       276        /* Undernet extension */
  
@@ -113,17 +113,17 @@ diff -r bcffa56c93d4 include/numeric.h
  #define RPL_GLIST            280        /* Undernet extension */
  #define RPL_ENDOFGLIST       281        /* Undernet extension */
  #define RPL_JUPELIST         282        /* Undernet extension - jupe -Kev */
-@@ -442,6 +445,7 @@
- /*      ERR_GHOSTEDCLIENT    503           efnet */
/*    ERR_VWORLDWARN       503           austnet */
+@@ -445,6 +448,7 @@
#define ERR_NOSUCHWELCOME    509        /* QuakeNet extension */
  
 +#define ERR_NOSUCHSPLIT      510        /* QuakeNet extension */
  #define ERR_SILELISTFULL     511        /* Undernet extension */
  /*      ERR_NOTIFYFULL       512           aircd */
  /*    ERR_TOOMANYWATCH     512           Numeric List: Dalnet */
-diff -r bcffa56c93d4 include/split.h
+diff -r 2267c134dc0d include/split.h
 --- /dev/null  Thu Jan 01 00:00:00 1970 +0000
-+++ b/include/split.h  Fri Jan 23 23:04:58 2009 +0100
++++ b/include/split.h  Tue Jan 27 18:25:58 2009 +0100
 @@ -0,0 +1,106 @@
 +#ifndef INCLUDED_split_h
 +#define INCLUDED_split_h
@@ -190,14 +190,14 @@ diff -r bcffa56c93d4 include/split.h
 +};
 +
 +/* Split update flags. */
-+#define SPLIT_EXPIRE     0x02  /**< Expiration time update. */
-+#define SPLIT_LIFETIME   0x04  /**< Record lifetime update. */
-+#define SPLIT_REASON     0x08  /**< Reason update. */
-+#define SPLIT_CREATION   0x10  /**< TODO: Creation time update - unused. */
-+#define SPLIT_MODIFY     0x20  /**< No state change. */
++#define SPLIT_EXPIRE     0x01  /**< Expiration time update. */
++#define SPLIT_LIFETIME   0x02  /**< Record lifetime update. */
++#define SPLIT_REASON     0x04  /**< Reason update. */
++#define SPLIT_CREATION   0x08  /**< TODO: Creation time update - unused. */
++#define SPLIT_MODIFY     0x10  /**< No state change. */
 +
 +/* mask for Split update flags. */
-+#define SPLIT_UPDATE (SPLIT_EXPIRE | SPLIT_LIFETIME | SPLIT_REASON)
++#define SPLIT_UPDATE (SPLIT_EXPIRE | SPLIT_CREATION | SPLIT_LIFETIME | SPLIT_REASON)
 +
 +/* test whether a split entry is active. */
 +#define SplitIsActive(s)      ((s)->sp_flags & SPLIT_ACTIVE)
@@ -231,9 +231,9 @@ diff -r bcffa56c93d4 include/split.h
 +extern int split_memory_count(size_t *sp_size);
 +
 +#endif /* INCLUDED_split_h */
-diff -r bcffa56c93d4 ircd/Makefile.in
---- a/ircd/Makefile.in Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/Makefile.in Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/Makefile.in
+--- a/ircd/Makefile.in Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/Makefile.in Tue Jan 27 18:25:58 2009 +0100
 @@ -173,6 +173,7 @@
        m_sethost.c \
        m_settime.c \
@@ -242,15 +242,15 @@ diff -r bcffa56c93d4 ircd/Makefile.in
        m_squit.c \
        m_stats.c \
        m_time.c \
-@@ -212,6 +213,7 @@
+@@ -213,6 +214,7 @@
        s_stats.c \
        s_user.c \
        send.c \
 +      split.c \
        uping.c \
        userload.c \
-       whocmds.c \
-@@ -1052,6 +1054,15 @@
+       welcome.c \
+@@ -1054,6 +1056,15 @@
    ../include/ircd_chattr.h ../include/list.h ../include/msg.h \
    ../include/numeric.h ../include/numnicks.h ../include/s_user.h \
    ../include/send.h ../include/struct.h
@@ -266,7 +266,7 @@ diff -r bcffa56c93d4 ircd/Makefile.in
  m_squit.o: m_squit.c ../config.h ../include/client.h \
    ../include/ircd_defs.h ../include/dbuf.h ../include/msgq.h \
    ../include/ircd_events.h ../config.h ../include/ircd_handler.h \
-@@ -1325,6 +1336,7 @@
+@@ -1332,6 +1343,7 @@
    ../include/msgq.h ../include/numeric.h ../include/numnicks.h \
    ../include/res.h ../include/s_bsd.h ../include/s_conf.h \
    ../include/s_user.h ../include/s_stats.h ../include/send.h \
@@ -274,7 +274,7 @@ diff -r bcffa56c93d4 ircd/Makefile.in
    ../include/struct.h ../include/sys.h ../include/whowas.h
  s_err.o: s_err.c ../config.h ../include/numeric.h ../include/ircd_log.h \
    ../include/s_debug.h ../config.h ../include/ircd_defs.h
-@@ -1410,6 +1422,16 @@
+@@ -1417,6 +1429,16 @@
    ../include/msg.h ../include/numnicks.h ../include/parse.h \
    ../include/s_bsd.h ../include/s_debug.h ../include/s_misc.h \
    ../include/s_user.h ../include/struct.h ../include/sys.h
@@ -291,9 +291,9 @@ diff -r bcffa56c93d4 ircd/Makefile.in
  uping.o: uping.c ../config.h ../include/uping.h ../include/ircd_defs.h \
    ../include/ircd_events.h ../config.h ../include/res.h \
    ../include/client.h ../include/dbuf.h ../include/msgq.h \
-diff -r bcffa56c93d4 ircd/ircd.c
---- a/ircd/ircd.c      Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/ircd.c      Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/ircd.c
+--- a/ircd/ircd.c      Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/ircd.c      Tue Jan 27 18:25:58 2009 +0100
 @@ -55,6 +55,7 @@
  #include "s_misc.h"
  #include "s_stats.h"
@@ -312,10 +312,10 @@ diff -r bcffa56c93d4 ircd/ircd.c
    event_loop();
  
    return 0;
-diff -r bcffa56c93d4 ircd/ircd_features.c
---- a/ircd/ircd_features.c     Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/ircd_features.c     Fri Jan 23 23:04:58 2009 +0100
-@@ -365,6 +365,10 @@
+diff -r 2267c134dc0d ircd/ircd_features.c
+--- a/ircd/ircd_features.c     Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/ircd_features.c     Tue Jan 27 18:25:58 2009 +0100
+@@ -366,6 +366,10 @@
    F_B(SETHOST_USER, 0, 0, 0),
    F_B(SETHOST_AUTO, 0, 0, 0),
  
@@ -326,7 +326,7 @@ diff -r bcffa56c93d4 ircd/ircd_features.c
    /* HEAD_IN_SAND Features */
    F_B(HIS_SNOTICES, 0, 1, 0),
    F_B(HIS_SNOTICES_OPER_ONLY, 0, 1, 0),
-@@ -392,6 +396,7 @@
+@@ -393,6 +397,7 @@
    F_B(HIS_STATS_q, 0, 1, 0),
    F_B(HIS_STATS_R, 0, 1, 0),
    F_B(HIS_STATS_r, 0, 1, 0),
@@ -334,9 +334,9 @@ diff -r bcffa56c93d4 ircd/ircd_features.c
    F_B(HIS_STATS_s, 0, 1, 0),
    F_B(HIS_STATS_t, 0, 1, 0),
    F_B(HIS_STATS_T, 0, 1, 0),
-diff -r bcffa56c93d4 ircd/ircd_log.c
---- a/ircd/ircd_log.c  Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/ircd_log.c  Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/ircd_log.c
+--- a/ircd/ircd_log.c  Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/ircd_log.c  Tue Jan 27 18:25:58 2009 +0100
 @@ -154,6 +154,7 @@
    S(OPERMODE, -1, SNO_HACK4),
    S(GLINE, -1, SNO_GLINE),
@@ -345,9 +345,9 @@ diff -r bcffa56c93d4 ircd/ircd_log.c
    S(WHO, -1, 0),
    S(NETWORK, -1, SNO_NETWORK),
    S(OPERKILL, -1, 0),
-diff -r bcffa56c93d4 ircd/m_endburst.c
---- a/ircd/m_endburst.c        Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/m_endburst.c        Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/m_endburst.c
+--- a/ircd/m_endburst.c        Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/m_endburst.c        Tue Jan 27 18:25:58 2009 +0100
 @@ -85,6 +85,7 @@
  #include "client.h"
  #include "hash.h"
@@ -383,9 +383,9 @@ diff -r bcffa56c93d4 ircd/m_endburst.c
    if (MyConnect(sptr))
      sendcmdto_one(&me, CMD_END_OF_BURST_ACK, sptr, "");
  
-diff -r bcffa56c93d4 ircd/m_reburst.c
---- a/ircd/m_reburst.c Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/m_reburst.c Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/m_reburst.c
+--- a/ircd/m_reburst.c Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/m_reburst.c Tue Jan 27 18:25:58 2009 +0100
 @@ -102,6 +102,7 @@
  #include "ircd_snprintf.h"
  #include "gline.h"
@@ -405,9 +405,9 @@ diff -r bcffa56c93d4 ircd/m_reburst.c
      default:
        break;
    }
-diff -r bcffa56c93d4 ircd/m_server.c
---- a/ircd/m_server.c  Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/m_server.c  Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/m_server.c
+--- a/ircd/m_server.c  Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/m_server.c  Tue Jan 27 18:25:58 2009 +0100
 @@ -47,6 +47,7 @@
  #include "s_misc.h"
  #include "s_serv.h"
@@ -426,10 +426,10 @@ diff -r bcffa56c93d4 ircd/m_server.c
    /*
     * Old sendto_serv_but_one() call removed because we now need to send
     * different names to different servers (domain name matching).
-diff -r bcffa56c93d4 ircd/m_split.c
+diff -r 2267c134dc0d ircd/m_split.c
 --- /dev/null  Thu Jan 01 00:00:00 1970 +0000
-+++ b/ircd/m_split.c   Fri Jan 23 23:04:58 2009 +0100
-@@ -0,0 +1,374 @@
++++ b/ircd/m_split.c   Tue Jan 27 18:25:58 2009 +0100
+@@ -0,0 +1,376 @@
 +/*
 + * IRC - Internet Relay Chat, ircd/m_split.c
 + * Copyright (C) 1990 Jarkko Oikarinen and
@@ -554,11 +554,11 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +  unsigned int flags = 0;
 +  enum SplitAction action = SPLIT_MODIFY;
 +  time_t creation = 0, expire = 0, lastmod = 0, lifetime = 0;
-+  char *server = parv[1], *tmp = 0;
++  char *server = 0, *tmp = 0;
 +  const char *reason = "No reason";
 +
-+  /* TODO: perhaps make some fields optional? reason? */
-+  if (parc < 6)
++  /* reason is optional - for lazy services, we always send it */
++  if (parc < 5)
 +    return need_more_params(sptr, "SPLIT");
 +
 +  if (*server == '!') {
@@ -581,6 +581,7 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +  }
 +
 +  /* Next, try to find the split entry... */
++  server = parv[1];
 +  asplit = split_find(server);
 +
 +  /* We now have all the pieces to tell us what we've got; let's put
@@ -591,12 +592,18 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +  if (!asplit && action == SPLIT_MODIFY)
 +    action = SPLIT_ACTIVATE;
 +
-+  /* OK, let's figure out what other parameters we may have... */
++  /* set all flags for modification */
++  if (asplit && action != SPLIT_REMOVE)
++    flags = SPLIT_UPDATE;
++
++  /* set the things we need */
 +  creation = atoi(parv[2]);
 +  expire = atoi(parv[3]);
 +  lastmod = atoi(parv[4]);
 +  lifetime = atoi(parv[5]);
-+  reason = parv[parc - 1];
++  /* got a reason too */
++  if (parc < 6)
++    reason = parv[parc - 1];
 +
 +  Debug((DEBUG_DEBUG, "I have a SPLIT I am acting upon now; "
 +    "server %s, action %s, creation %Tu, expire %Tu, "
@@ -617,10 +624,21 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +                      lastmod, lifetime, flags);
 +  }
 +
++  /* desynch or cross - we do not have it so we cannot remove it */
++  /* TODO: manually propagate the message ?
++   *   ..but if we do not have it, can our downlinks have it?
++   * they could if my oper force removed it earlier, and then my downlinks linked
++   * no, then they would have bursted it again to us.. argh
++   */
++  if (!asplit && action == SPLIT_REMOVE)
++    return 0;
++
 +  assert(action != SPLIT_MODIFY);
++  assert(action != SPLIT_REMOVE);
 +
++  /* only give one flag when needed - SPLIT_ACTIVE */
 +  return split_add(cptr, sptr, server, reason, creation, expire, lastmod, lifetime,
-+                 flags | ((action == SPLIT_ACTIVATE) ? SPLIT_ACTIVE : 0));
++    (flags & SPLIT_ACTIVATE) ? SPLIT_ACTIVE : 0);
 +}
 +
 +
@@ -648,6 +666,7 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +  char *server = parv[1], *end;
 +  const char *reason = NULL;
 +
++  /* listing */
 +  if (parc < 2) {
 +    if (!HasPriv(sptr, PRIV_ROUTEINFO)) 
 +      return send_reply(sptr, ERR_NOPRIVILEGES);
@@ -655,9 +674,10 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +      return split_list(sptr, 0);
 +  }
 +
++  /* force removal */
 +  if (*server == '!') {
 +    server++;
-+    action = SPLIT_REMOVE; /* force removal */
++    action = SPLIT_REMOVE;
 +  }
 +
 +  switch (*server) { /* handle + and - */
@@ -673,82 +693,67 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +    server++;
 +    break;
 +  }
-+  
-+  /* OK, let's figure out the parameters... */
-+  switch (action) {
-+    /* no specific action */
-+    case SPLIT_MODIFY:
-+    /* user wants a listing of a specific SPLIT */
-+    if (parc == 2)
-+      return split_list(sptr, server);
-+    expire = strtol(parv[2], &end, 10) + TStime(); /* and the expiration */
-+    if (*end != '\0')
-+      return send_reply(sptr, SND_EXPLICIT | ERR_BADEXPIRE,
-+        "%s :Bad expire time", parv[2]);
 +
-+    flags |= SPLIT_EXPIRE; /* remember that we got an expire time */
-+    
-+    if (parc > 3) { /* also got a reason... */
-+      reason = parv[parc - 1];
++  /* user wants listing */
++  if (action == SPLIT_MODIFY && parc == 2)
++    return split_list(sptr, server);
++
++  /* if we have an expire, this must be it */
++  if (parc > 2)
++    expire = strtol(parv[2], &end, 10); /* and the expiration */
++
++  /* got reason and expire */
++  if (parc > 3) {
++    flags |= SPLIT_EXPIRE;
++    flags |= SPLIT_REASON;
++  }
++  /* got reason or expire */
++  else if (parc > 2) {
++    if (*end != '\0')
 +      flags |= SPLIT_REASON;
-+    }
-+    break;
-+    
-+    case SPLIT_REMOVE: /* TODO: require reason for this, but not expire? */
-+    if (parc < 4)
-+      return need_more_params(sptr, "SPLIT");
++    else
++      flags |= SPLIT_EXPIRE;
++  }
++
++  /* set reason */
++  if (flags & SPLIT_REASON) {
 +    reason = parv[parc - 1];
-+    flags |= SPLIT_REASON;      
-+    break;
-+    
-+    case SPLIT_ACTIVATE: /* TODO: require expire and reason when new */
-+    if (parc > 2) {
-+      expire = strtol(parv[2], &end, 10) + TStime(); /* and the expiration */
-+      if (*end != '\0')
-+        return send_reply(sptr, SND_EXPLICIT | ERR_BADEXPIRE,
-+          "%s :Bad expire time", parv[2]);
-+      flags |= SPLIT_EXPIRE; /* remember that we got an expire time */
-+    }
-+    
-+    if (parc > 3) { /* also got a reason... */
-+      reason = parv[parc - 1];
-+      flags |= SPLIT_REASON;
-+    }
-+    break;
-+    
-+    case SPLIT_DEACTIVATE: /* TODO: duplicate code? must be a cleaner way */
-+    if (parc > 2) {
-+      expire = strtol(parv[2], &end, 10) + TStime(); /* and the expiration */
-+      if (*end != '\0')
-+        return send_reply(sptr, SND_EXPLICIT | ERR_BADEXPIRE,
-+          "%s :Bad expire time", parv[2]);
-+      flags |= SPLIT_EXPIRE; /* remember that we got an expire time */
-+    }
-+    
-+    if (parc > 3) { /* also got a reason... */
-+      reason = parv[parc - 1];
-+      flags |= SPLIT_REASON;
-+    }
-+    break;
++    if (reason[0] == '\0') /* reason is empty */
++      flags &= ~SPLIT_REASON;
 +  }
-+  
 +
-+  /* check for permissions... */
++  /* bad expire */  
++  if (flags & SPLIT_EXPIRE && *end != '\0')
++    return send_reply(sptr, SND_EXPLICIT | ERR_BADEXPIRE, "%s :Bad expire time", parv[2]);
++
++  /* require a reason for forced removal */
++  if (!(flags & SPLIT_REASON) && action == SPLIT_REMOVE)
++    return need_more_params(sptr, "SPLIT");
++
++  /* we got expire */
++  if (flags & SPLIT_EXPIRE) {
++    if (expire == 0) /* dummy value 0 given, no change to expire */
++      flags &= ~SPLIT_EXPIRE;
++    else /* turn expire into timestamp */
++      expire += TStime();
++   }
++
++  /* check feature and privs */
 +  if (!feature_bool(FEAT_SPLIT)) /* TODO: but allow force removal? */
 +    return send_reply(sptr, ERR_DISABLED, "SPLIT");
 +  else if (!HasPriv(sptr, PRIV_SERVERINFO)) /* TODO: create PRIV_SPLIT - need help there */
 +    return send_reply(sptr, ERR_NOPRIVILEGES);
 +
-+  /* Next, try to find the SPLIT... */
++  /* find the split */
 +  asplit = split_find(server);
 +
 +  /* We now have all the pieces to tell us what we've got; let's put
 +   * it all together and convert the rest of the arguments.
 +   */
 +
-+  /* SPLIT not found and thus we:
-+   *   cannot remove SPLIT we do not have
-+   *   cannot add new SPLIT without expire and reason
++  /* SPLIT not found and thus we cannot:
++   *   remove SPLIT we do not have
++   *   add new SPLIT without expire and reason
 +   */
 +  if (!asplit &&
 +      ((action == SPLIT_REMOVE) ||
@@ -791,9 +796,6 @@ diff -r bcffa56c93d4 ircd/m_split.c
 + * m_split - user message handler
 + *
 + * parv[0] = Send prefix
-+ *
-+ * From user:
-+ *
 + * parv[1] = [<server name>]
 + *
 + */
@@ -804,26 +806,26 @@ diff -r bcffa56c93d4 ircd/m_split.c
 +
 +  return split_list(sptr, parv[1]);
 +}
-diff -r bcffa56c93d4 ircd/parse.c
---- a/ircd/parse.c     Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/parse.c     Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/parse.c
+--- a/ircd/parse.c     Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/parse.c     Tue Jan 27 18:25:58 2009 +0100
 @@ -505,6 +505,13 @@
-     0, MAXPARA, MFLG_SLOW, 0, NULL,
-     /* UNREG, CLIENT, SERVER, OPER, SERVICE */
-     { m_unregistered, m_not_oper, ms_jupe, mo_jupe, m_ignore }
-+  },
+     /* UNREG, CLIENT, SERVER, OPER, SERVICE, HELP */
+     { m_unregistered, m_not_oper, ms_jupe, mo_jupe, m_ignore, mh_jupe }
+   },
 +  {
 +    MSG_SPLIT,
 +    TOK_SPLIT,
 +    0, MAXPARA, MFLG_SLOW, 0, NULL,
-+    /* UNREG, CLIENT, SERVER, OPER, SERVICE */
-+    { m_unregistered, m_not_oper, ms_split, mo_split, m_ignore }
-   },
++    /* UNREG, CLIENT, SERVER, OPER, SERVICE, HELP */
++    { m_unregistered, m_not_oper, ms_split, mo_split, m_ignore, mh_nohelp }
++   },
    {
      MSG_OPMODE,
-diff -r bcffa56c93d4 ircd/s_conf.c
---- a/ircd/s_conf.c    Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_conf.c    Fri Jan 23 23:04:58 2009 +0100
+     TOK_OPMODE,
+diff -r 2267c134dc0d ircd/s_conf.c
+--- a/ircd/s_conf.c    Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_conf.c    Tue Jan 27 18:25:58 2009 +0100
 @@ -53,6 +53,7 @@
  #include "s_debug.h"
  #include "s_misc.h"
@@ -840,9 +842,9 @@ diff -r bcffa56c93d4 ircd/s_conf.c
  
    return ret;
  }
-diff -r bcffa56c93d4 ircd/s_debug.c
---- a/ircd/s_debug.c   Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_debug.c   Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/s_debug.c
+--- a/ircd/s_debug.c   Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_debug.c   Tue Jan 27 18:25:58 2009 +0100
 @@ -48,6 +48,7 @@
  #include "s_user.h"
  #include "s_stats.h"
@@ -850,39 +852,40 @@ diff -r bcffa56c93d4 ircd/s_debug.c
 +#include "split.h"
  #include "struct.h"
  #include "sys.h"
- #include "whowas.h"
-@@ -231,7 +232,8 @@
-       aw = 0,                   /* aways set */
+ #include "welcome.h"
+@@ -233,6 +234,7 @@
        wwa = 0,                  /* whowas aways */
        gl = 0,                   /* glines */
--      ju = 0;                   /* jupes */
-+      ju = 0,                   /* jupes */
-+      sp = 0;                   /* split entries */
+       ju = 0,                   /* jupes */
++      sp = 0,                   /* split entries */
+       we = 0;                   /* welcome messages */
  
    size_t chm = 0,               /* memory used by channels */
-       chbm = 0,                 /* memory used by channel bans */
-@@ -244,6 +246,7 @@
+@@ -246,6 +248,7 @@
        wwm = 0,                  /* whowas array memory used */
        glm = 0,                  /* memory used by glines */
        jum = 0,                  /* memory used by jupes */
 +      spm = 0,                  /* memory used by split entries */
+       wem = 0,                  /* memory used by welcome messages */
        com = 0,                  /* memory used by conf lines */
        dbufs_allocated = 0,      /* memory used by dbufs */
-       dbufs_used = 0,           /* memory used by dbufs */
-@@ -348,8 +351,9 @@
+@@ -351,10 +354,11 @@
  
    gl = gline_memory_count(&glm);
    ju = jupe_memory_count(&jum);
 +  sp = split_memory_count(&spm);
+   we = welcome_memory_count(&wem);
    send_reply(cptr, SND_EXPLICIT | RPL_STATSDEBUG,
--           ":Glines %d(%zu) Jupes %d(%zu)", gl, glm, ju, jum);
-+           ":Glines %d(%zu) Jupes %d(%zu) Splits %d(%zu)", gl, glm, ju, jum, sp, spm);
+-           ":Glines %d(%zu) Jupes %d(%zu) WelcomeMessages %d(%zu)",
+-           gl, glm, ju, jum, we, wem);
++           ":Glines %d(%zu) Jupes %d(%zu) Splits %d(%zu) WelcomeMessages %d(%zu)",
++           gl, glm, ju, jum, sp, spm, we, wem);
  
    send_reply(cptr, SND_EXPLICIT | RPL_STATSDEBUG,
             ":Hash: client %d(%zu), chan is the same", HASHSIZE,
-diff -r bcffa56c93d4 ircd/s_err.c
---- a/ircd/s_err.c     Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_err.c     Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/s_err.c
+--- a/ircd/s_err.c     Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_err.c     Tue Jan 27 18:25:58 2009 +0100
 @@ -490,7 +490,7 @@
  /* 228 */
    { RPL_STATSQLINE, "Q %s :%s", "228" },
@@ -906,16 +909,16 @@ diff -r bcffa56c93d4 ircd/s_err.c
  /* 281 */
 @@ -1052,7 +1052,7 @@
  /* 509 */
-   { 0 },
+   { ERR_NOSUCHWELCOME, "%s :No such welcome", "509" },
  /* 510 */
 -  { 0 },
 +  { ERR_NOSUCHSPLIT, "%s :No such split", "510" },
  /* 511 */
    { ERR_SILELISTFULL, "%s :Your silence list is full", "511" },
  /* 512 */
-diff -r bcffa56c93d4 ircd/s_misc.c
---- a/ircd/s_misc.c    Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_misc.c    Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/s_misc.c
+--- a/ircd/s_misc.c    Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_misc.c    Tue Jan 27 18:25:58 2009 +0100
 @@ -53,6 +53,7 @@
  #include "s_stats.h"
  #include "s_user.h"
@@ -948,9 +951,9 @@ diff -r bcffa56c93d4 ircd/s_misc.c
    }
  
    /*
-diff -r bcffa56c93d4 ircd/s_serv.c
---- a/ircd/s_serv.c    Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_serv.c    Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/s_serv.c
+--- a/ircd/s_serv.c    Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_serv.c    Tue Jan 27 18:25:58 2009 +0100
 @@ -54,6 +54,7 @@
  #include "s_misc.h"
  #include "s_user.h"
@@ -959,17 +962,17 @@ diff -r bcffa56c93d4 ircd/s_serv.c
  #include "struct.h"
  #include "sys.h"
  #include "userload.h"
-@@ -196,6 +197,7 @@
+@@ -197,6 +198,7 @@
     */
    gline_burst(cptr);
    jupe_burst(cptr);
 +  split_burst(cptr);
+   welcome_burst(cptr);
  
    /*
-    * Pass on my client information to the new server
-diff -r bcffa56c93d4 ircd/s_stats.c
---- a/ircd/s_stats.c   Fri Jan 23 18:45:25 2009 +0100
-+++ b/ircd/s_stats.c   Fri Jan 23 23:04:58 2009 +0100
+diff -r 2267c134dc0d ircd/s_stats.c
+--- a/ircd/s_stats.c   Tue Jan 27 18:19:11 2009 +0100
++++ b/ircd/s_stats.c   Tue Jan 27 18:25:58 2009 +0100
 @@ -52,6 +52,7 @@
  #include "s_stats.h"
  #include "s_user.h"
@@ -977,8 +980,8 @@ diff -r bcffa56c93d4 ircd/s_stats.c
 +#include "split.h"
  #include "struct.h"
  #include "userload.h"
-@@ -668,7 +669,10 @@
+ #include "welcome.h"
+@@ -669,7 +670,10 @@
      send_usage, 0,
      "System resource usage (Debug only)." },
  #endif
@@ -990,10 +993,10 @@ diff -r bcffa56c93d4 ircd/s_stats.c
      stats_sline, 0,
      "Spoofed hosts information." },
    { 'T', "motds", (STAT_FLAG_OPERFEAT | STAT_FLAG_CASESENS), FEAT_HIS_STATS_T,
-diff -r bcffa56c93d4 ircd/split.c
+diff -r 2267c134dc0d ircd/split.c
 --- /dev/null  Thu Jan 01 00:00:00 1970 +0000
-+++ b/ircd/split.c     Fri Jan 23 23:04:58 2009 +0100
-@@ -0,0 +1,773 @@
++++ b/ircd/split.c     Tue Jan 27 18:25:58 2009 +0100
+@@ -0,0 +1,786 @@
 +/*
 + * IRC - Internet Relay Chat, ircd/split.c
 + * Copyright (C) 1990 Jarkko Oikarinen and
@@ -1128,11 +1131,12 @@ diff -r bcffa56c93d4 ircd/split.c
 +
 +  struct Split *asplit;
 +  struct Client *acptr;
++  char noreason[BUFSIZE];
 +
++  assert(0 != cptr);
++  assert(0 != sptr);
 +  assert(0 != server);
 +  assert(0 != reason);
-+  assert(NULL != cptr);
-+  assert(NULL != sptr);
 +
 +  Debug((DEBUG_DEBUG, "split_add(\"%s\", \"%s\", \"%s\", \"%s\", %Tu, %Tu, %Tu, "
 +       "%Tu, 0x%04x)", cli_name(cptr), cli_name(sptr), server, reason,
@@ -1155,7 +1159,8 @@ diff -r bcffa56c93d4 ircd/split.c
 +   * duration greater than SPLIT_MAX_EXPIRE.
 +   */
 +  if (expire - TStime() <= 0 || expire - TStime() > SPLIT_MAX_EXPIRE) {
-+    if (!IsServer(cptr) && MyConnect(cptr))
++    /* TODO: cptr or sptr here? */
++    if (MyUser(sptr))
 +      return send_reply(cptr, ERR_BADEXPIRE, expire - TStime());
 +    if (lifetime <= TStime()) /* no point going further */
 +      /* CHECK: sptr may have the wrong idea about the nettime?
@@ -1165,21 +1170,26 @@ diff -r bcffa56c93d4 ircd/split.c
 +      return 0;
 +  }
 +
++  /* empty reason */
++  if (reason[0] == '\0')
++    ircd_snprintf(0, noreason, sizeof(noreason), "%C gave no reason.", sptr);
++
 +  /* TODO: add opername */
 +  /* inform ops and log it */
 +  sendto_opmask_butone(0, SNO_NETWORK, "%s adding%sSPLIT for %s, expiring at %Tu: %s",
 +    (feature_bool(FEAT_HIS_SNOTICES) || IsServer(sptr)) ?
 +      cli_name(sptr) : cli_name((cli_user(sptr))->server),
 +    !(flags & SPLIT_ACTIVE) ? " deactivated " : " ",
-+    server, expire, reason);
++    server, expire, reason[0] == '\0' ? noreason : reason);
 +
 +  log_write(LS_SPLIT, L_INFO, LOG_NOSNOTICE,
 +    "%#C adding%sSPLIT for %s, expiring at %Tu: %s",
 +    sptr, !(flags & SPLIT_ACTIVE) ? " deactivated " : " ",
-+    server, expire, reason);
++    server, expire, reason[0] == '\0' ? noreason : reason);
 +
 +  /* make the split entry */
-+  asplit = split_make(server, reason, creation, expire, lastmod, lifetime, flags);
++  asplit = split_make(server, reason[0] == '\0' ? noreason : reason,
++             creation, expire, lastmod, lifetime, flags);
 +  
 +  assert(asplit);
 +  
@@ -1212,13 +1222,13 @@ diff -r bcffa56c93d4 ircd/split.c
 +  char buf[BUFSIZE];
 +  int pos = 0;
 +
-+  /* TODO: assert() on other inputs? */
-+  assert(split);
-+  assert(NULL != cptr);
-+  assert(NULL != sptr);
++  assert(0 != cptr);
++  assert(0 != sptr);
++  assert(0 != split);
 +
-+  Debug((DEBUG_DEBUG, "split_modify(\"%s\", \"%s\", \"%s\", \"%s\", %d, %Tu, %Tu, %Tu, "
-+       "%Tu, 0x%04x)", cli_name(cptr), cli_name(sptr), split->sp_server, reason, action,
++  Debug((DEBUG_DEBUG, "split_modify(cptr \"%s\", sptr \"%s\", split \"%s\", reason \"%s\","
++    " action %d, creation %Tu, expire %Tu, lastmod %Tu, lifetime %Tu, flags 0x%04x)",
++    cli_name(cptr), cli_name(sptr), split->sp_server, reason, action,
 +       creation, expire, lastmod, lifetime, flags));
 +
 +  /* not modifying SPLIT for server that is linked
@@ -1240,26 +1250,28 @@ diff -r bcffa56c93d4 ircd/split.c
 +      split_free(split); /* and free it */
 +    return 0;
 +  }
-+  
-+  /* First, let's check lastmod... */
++
++  /* check lastmod */
 +  if (split->sp_lastmod > lastmod) { /* we have a more recent version */
-+    if (IsBurstOrBurstAck(cptr))
-+      return 0; /* middle of a burst, it'll resync on its own */
++    if (IsBurstOrBurstAck(cptr)) /* middle of a burst, it will resync on its own */
++      return 0;
 +    return split_resend(cptr, split); /* resync the server */
-+  } else if (split->sp_lastmod == lastmod)
-+     return 0; /* we have that version of the split entry... */
++  } else if (split->sp_lastmod == lastmod) /* same version */
++     return 0;
 +  
-+  /* All right, we know that there's a change of some sort.  What is it? */
-+  /* first, check out the expiration time... */
-+  /* TODO: expire != 0 or NULL - check  that we have something? */
-+  if (expire != 0 && expire != split->sp_expire) {
-+    if (expire - TStime() <= 0 || expire - TStime() > SPLIT_MAX_EXPIRE) {
-+      if (!IsServer(sptr) && MyConnect(sptr)) /* bad expiration time */
++  /* there is a change of some sort */
++  /* check out the expiration time */
++  if (flags & SPLIT_EXPIRE) {
++    if (expire != split->sp_expire) {
++      /* bad expiration time */
++      if (MyUser(sptr) && (expire < TStime() || expire - TStime() > SPLIT_MAX_EXPIRE))
 +        return send_reply(sptr, ERR_BADEXPIRE, expire - TStime());
-+    }
-+    flags |= SPLIT_EXPIRE;
-+  } else
-+    flags &= ~SPLIT_EXPIRE;
++      /* expire timestamp is in the past, so remap to deactivate */
++      if (expire < TStime() && action == SPLIT_ACTIVATE)
++        action = SPLIT_DEACTIVATE;
++    } else
++      flags &= ~SPLIT_EXPIRE;
++  }
 +
 +  /* Next, check out lifetime */
 +  if (!(flags & SPLIT_LIFETIME) || !lifetime)
@@ -1278,11 +1290,9 @@ diff -r bcffa56c93d4 ircd/split.c
 +    flags &= ~SPLIT_LIFETIME; /* no change to lifetime */
 +    lifetime = 0;
 +  }
-+  
++
 +  /* Finally, let's see if the reason needs to be updated */
-+  /* TODO: (reason) or use != NULL / 0 ? */
-+  if ((flags & SPLIT_REASON) && reason &&
-+      ircd_strcmp(split->sp_reason, reason) != 0)
++  if ((flags & SPLIT_REASON) && (reason[0] == '\0' || !ircd_strcmp(split->sp_reason, reason)))
 +    flags &= ~SPLIT_REASON; /* no changes to the reason */
 +
 +  /* OK, now let's take a look at the action... */
@@ -1298,8 +1308,12 @@ diff -r bcffa56c93d4 ircd/split.c
 +     (action == SPLIT_MODIFY ? "SPLIT_MODIFY" : "<UNKNOWN>"))));
 +
 +  /* If there are no changes to perform, do no changes */
-+  if (!(flags & SPLIT_UPDATE) && action == SPLIT_MODIFY)
++  if (!(flags & SPLIT_UPDATE) && action == SPLIT_MODIFY) {
++    if (MyUser(sptr))
++      sendcmdto_one(&me, CMD_NOTICE, sptr,
++        "%C :Cannot modify SPLIT %s - nothing to change.", sptr, split->sp_server);
 +    return 0;
++  }
 +
 +  /* Start by updating lastmod, if indicated... */
 +  split->sp_lastmod = lastmod;
@@ -1320,6 +1334,8 @@ diff -r bcffa56c93d4 ircd/split.c
 +    break;
 +  }
 +
++  /* TODO: Handle creation changes... */
++
 +  /* Handle expiration changes... */
 +  if (flags & SPLIT_EXPIRE) {
 +    split->sp_expire = expire; /* save new expiration time */