From b90f71ca2ab1b2eea7ca992b1e51a4b39afe0728 Mon Sep 17 00:00:00 2001 From: Scott Perry Date: Mon, 26 May 2008 12:38:15 +0200 Subject: [PATCH] Use strtok_r instead of strchr in IRC_JOIN. This patch does significant cleanup on the join code by using strtok_r instead of mangling strchr to parse channel names and keys in parallel when a JOIN command contains a list of channels and keys. Also adds an strtok_r implementation to libportab. --- configure.in | 2 +- src/ngircd/irc-channel.c | 30 ++++++++---------------------- src/portab/Makefile.am | 4 +--- src/portab/portab.h | 4 ++++ src/portab/strtok_r.c | 27 +++++++++++++++++++++++++++ 5 files changed, 41 insertions(+), 26 deletions(-) create mode 100644 src/portab/strtok_r.c diff --git a/configure.in b/configure.in index 3aa42fce..f02835fb 100644 --- a/configure.in +++ b/configure.in @@ -139,7 +139,7 @@ AC_CHECK_FUNCS([ \ bind gethostbyaddr gethostbyname gethostname inet_ntoa \ setsid setsockopt socket strcasecmp waitpid],,AC_MSG_ERROR([required function missing!])) -AC_CHECK_FUNCS(inet_aton isdigit sigaction snprintf vsnprintf strdup strlcpy strlcat) +AC_CHECK_FUNCS(inet_aton isdigit sigaction snprintf vsnprintf strdup strlcpy strlcat strtok_r) # -- Configuration options -- diff --git a/src/ngircd/irc-channel.c b/src/ngircd/irc-channel.c index 92dd8cd3..203e32f4 100644 --- a/src/ngircd/irc-channel.c +++ b/src/ngircd/irc-channel.c @@ -182,7 +182,7 @@ join_send_topic(CLIENT *Client, CLIENT *target, CHANNEL *chan, GLOBAL bool IRC_JOIN( CLIENT *Client, REQUEST *Req ) { - char *channame, *channame_ptr, *key, *key_ptr, *flags; + char *channame, *key = NULL, *flags, *lastkey, *lastchan; CLIENT *target; CHANNEL *chan; @@ -208,16 +208,11 @@ IRC_JOIN( CLIENT *Client, REQUEST *Req ) return part_from_all_channels(Client, target); /* Are channel keys given? */ - if (Req->argc > 1) { - key = Req->argv[1]; - key_ptr = strchr(key, ','); - if (key_ptr) *key_ptr = '\0'; - } else { - key = key_ptr = NULL; - } + if (Req->argc > 1) + key = strtok_r(Req->argv[1], ",", &lastkey); + channame = Req->argv[0]; - channame_ptr = strchr(channame, ','); - if (channame_ptr) *channame_ptr = '\0'; + channame = strtok_r(channame, ",", &lastchan); while (channame) { flags = NULL; @@ -288,18 +283,9 @@ IRC_JOIN( CLIENT *Client, REQUEST *Req ) break; /* write error */ /* next channel? */ - channame = channame_ptr; - if (channame) { - channame++; - channame_ptr = strchr(channame, ','); - if (channame_ptr) *channame_ptr = '\0'; - - if (key_ptr) { - key = ++key_ptr; - key_ptr = strchr(key, ','); - if (key_ptr) *key_ptr = '\0'; - } - } + channame = strtok_r(NULL, ",", &lastchan); + if (channame && key) + key = strtok_r(NULL, ",", &lastkey); } return CONNECTED; } /* IRC_JOIN */ diff --git a/src/portab/Makefile.am b/src/portab/Makefile.am index 06c6bae4..c48e67ad 100644 --- a/src/portab/Makefile.am +++ b/src/portab/Makefile.am @@ -9,14 +9,12 @@ # Naehere Informationen entnehmen Sie bitter der Datei COPYING. Eine Liste # der an ngIRCd beteiligten Autoren finden Sie in der Datei AUTHORS. # -# $Id: Makefile.am,v 1.8 2005/04/16 09:23:01 fw Exp $ -# AUTOMAKE_OPTIONS = ansi2knr noinst_LIBRARIES = libngportab.a -libngportab_a_SOURCES = strlcpy.c strdup.c vsnprintf.c +libngportab_a_SOURCES = strdup.c strlcpy.c strtok_r.c vsnprintf.c check_PROGRAMS = portabtest diff --git a/src/portab/portab.h b/src/portab/portab.h index 2a133efc..83e11313 100644 --- a/src/portab/portab.h +++ b/src/portab/portab.h @@ -148,6 +148,10 @@ extern size_t strlcpy PARAMS(( char *dst, const char *src, size_t size )); extern char * strdup PARAMS(( const char *s )); #endif +#ifndef HAVE_STRTOK_R +extern char * strtok_r PARAMS((char *str, const char *delim, char **saveptr)); +#endif + #ifndef HAVE_VSNPRINTF #include extern int vsnprintf PARAMS(( char *str, size_t count, const char *fmt, va_list args )); diff --git a/src/portab/strtok_r.c b/src/portab/strtok_r.c new file mode 100644 index 00000000..852c8f70 --- /dev/null +++ b/src/portab/strtok_r.c @@ -0,0 +1,27 @@ +#include "portab.h" +#include + +#ifndef HAVE_STRTOK_R + +char * +strtok_r(char *str, const char *delim, char **saveptr) +{ + char *tmp; + + if (!str) + str = *saveptr; + str += strspn(str, delim); + if (*str == 0) + return NULL; + + tmp = str + strcspn(str, delim); /* get end of token */ + if (*tmp) { /* another delimiter */ + *tmp = 0; + tmp++; + } + *saveptr = tmp; + return str; +} + +#endif +