X-Git-Url: https://arthur.barton.de/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fngircd%2Fconn.c;h=e460bb48e6a92aa16aa9dda036995bf8e0c9f16d;hb=b65358b17c8595b1616dc702bf93745dd5c75367;hp=5ac1af460364f9216bd44d87ea559667916a9b3c;hpb=af9123fd82491efeb683722d58758d92dcca4833;p=ngircd-alex.git diff --git a/src/ngircd/conn.c b/src/ngircd/conn.c index 5ac1af46..e460bb48 100644 --- a/src/ngircd/conn.c +++ b/src/ngircd/conn.c @@ -17,7 +17,7 @@ #include "portab.h" #include "io.h" -static char UNUSED id[] = "$Id: conn.c,v 1.177 2005/09/02 13:28:30 alex Exp $"; +static char UNUSED id[] = "$Id: conn.c,v 1.183 2005/09/24 02:48:46 fw Exp $"; #include "imp.h" #include @@ -105,6 +105,7 @@ int deny_severity = LOG_ERR; static void server_login PARAMS((CONN_ID idx)); +static void cb_Read_Resolver_Result PARAMS(( int sock, UNUSED short what)); static void cb_clientserver PARAMS((int sock, short what)); static void @@ -198,22 +199,6 @@ cb_clientserver(int sock, short what) } -static void -FreeRes_stat( CONNECTION *c ) -{ - assert( c != NULL ); - assert( c->res_stat != NULL ); - - if (!c->res_stat) return; - - io_close( c->res_stat->pipe[0] ); - - array_free(&c->res_stat->buffer); - free( c->res_stat ); - c->res_stat = NULL; -} - - GLOBAL void Conn_Init( void ) { @@ -335,23 +320,22 @@ Conn_ExitListeners( void ) { /* Close down all listening sockets */ int *fd; - unsigned int arraylen; + size_t arraylen; #ifdef ZEROCONF Rendezvous_UnregisterListeners( ); #endif arraylen = array_length(&My_Listeners, sizeof (int)); - Log( LOG_INFO, "Shutting down all listening sockets (%d)...", arraylen ); + Log( LOG_INFO, "Shutting down all listening sockets (%d total)...", arraylen ); + fd = array_start(&My_Listeners); while(arraylen--) { - fd = (int*) array_get(&My_Listeners, sizeof (int), arraylen); - if (fd) { - close(*fd); + assert(fd); + assert(*fd >= 0); + io_close(*fd); #ifdef DEBUG - Log( LOG_DEBUG, "Listening socket %d closed.", *fd ); - } else { - Log( LOG_DEBUG, "array_get pos %d returned NULL", arraylen ); + Log( LOG_DEBUG, "Listening socket %d closed.", *fd ); #endif - } + fd++; } array_free(&My_Listeners); } /* Conn_ExitListeners */ @@ -384,40 +368,35 @@ NewListener( const UINT16 Port ) if( inaddr.s_addr == (unsigned)-1 ) #endif { - Log( LOG_CRIT, "Can't listen on %s:%u: can't convert ip address %s!", Conf_ListenAddress, Port, Conf_ListenAddress ); + Log( LOG_CRIT, "Can't listen on %s:%u: can't convert ip address %s!", + Conf_ListenAddress, Port, Conf_ListenAddress ); return -1; } } else inaddr.s_addr = htonl( INADDR_ANY ); addr.sin_addr = inaddr; - /* Socket erzeugen */ sock = socket( PF_INET, SOCK_STREAM, 0); - if( sock < 0 ) - { + if( sock < 0 ) { Log( LOG_CRIT, "Can't create socket: %s!", strerror( errno )); return -1; } if( ! Init_Socket( sock )) return -1; - /* an Port binden */ - if( bind( sock, (struct sockaddr *)&addr, (socklen_t)sizeof( addr )) != 0 ) - { + if( bind( sock, (struct sockaddr *)&addr, (socklen_t)sizeof( addr )) != 0 ) { Log( LOG_CRIT, "Can't bind socket: %s!", strerror( errno )); close( sock ); return -1; } - /* in "listen mode" gehen :-) */ - if( listen( sock, 10 ) != 0 ) - { - Log( LOG_CRIT, "Can't listen on soecket: %s!", strerror( errno )); + if( listen( sock, 10 ) != 0 ) { + Log( LOG_CRIT, "Can't listen on socket: %s!", strerror( errno )); close( sock ); return -1; } - /* Neuen Listener in Strukturen einfuegen */ + /* keep fd in list so we can close it when ngircd restarts/shuts down */ if (!array_catb( &My_Listeners,(char*) &sock, sizeof(int) )) { Log( LOG_CRIT, "Can't add socket to My_Listeners array: %s!", strerror( errno )); close( sock ); @@ -470,12 +449,10 @@ Conn_Handler( void ) int i; unsigned int wdatalen; struct timeval tv; - time_t start, t; + time_t t; bool timeout; - start = time( NULL ); - while(( ! NGIRCd_SignalQuit ) && ( ! NGIRCd_SignalRestart )) - { + while(( ! NGIRCd_SignalQuit ) && ( ! NGIRCd_SignalRestart )) { timeout = true; #ifdef ZEROCONF @@ -492,8 +469,7 @@ Conn_Handler( void ) t = time( NULL ); /* noch volle Lese-Buffer suchen */ - for( i = 0; i < Pool_Size; i++ ) - { + for( i = 0; i < Pool_Size; i++ ) { if(( My_Connections[i].sock > NONE ) && ( array_bytes(&My_Connections[i].rbuf) > 0 ) && ( My_Connections[i].delaytime < t )) { @@ -525,7 +501,7 @@ Conn_Handler( void ) if ( My_Connections[i].sock <= NONE ) continue; - if ( My_Connections[i].res_stat ) { + if (Resolve_INPROGRESS(&My_Connections[i].res_stat)) { /* wait for completion of Resolver Sub-Process */ io_event_del( My_Connections[i].sock, IO_WANTREAD ); continue; @@ -542,12 +518,11 @@ Conn_Handler( void ) io_event_add( My_Connections[i].sock, IO_WANTREAD ); } - /* Timeout initialisieren */ + /* (re-)set timeout - tv_sec/usec are undefined after io_dispatch() returns */ tv.tv_usec = 0; - if( timeout ) tv.tv_sec = 1; - else tv.tv_sec = 0; + tv.tv_sec = timeout ? 1 : 0; - /* Auf Aktivitaet warten */ + /* wait for activity */ i = io_dispatch( &tv ); if (i == -1 && errno != EINTR ) { Log(LOG_EMERG, "Conn_Handler(): io_dispatch(): %s!", strerror(errno)); @@ -789,9 +764,10 @@ Conn_Close( CONN_ID Idx, char *LogMsg, char *FwdMsg, bool InformClient ) Log( LOG_INFO, "Connection %d with %s:%d closed (in: %.1fk, out: %.1fk).", Idx, My_Connections[Idx].host, ntohs( My_Connections[Idx].addr.sin_port ), in_k, out_k ); } - /* Is there a resolver sub-process running? */ - if( My_Connections[Idx].res_stat ) - FreeRes_stat( &My_Connections[Idx] ); + /* cancel running resolver */ + if (Resolve_INPROGRESS(&My_Connections[Idx].res_stat)) { + Resolve_Shutdown(&My_Connections[Idx].res_stat); + } /* Servers: Modify time of next connect attempt? */ Conf_UnsetServer( Idx ); @@ -858,8 +834,10 @@ Handle_Write( CONN_ID Idx ) assert( Idx > NONE ); if ( My_Connections[Idx].sock < 0 ) { - Log(LOG_WARNING, +#ifdef DEBUG + Log(LOG_DEBUG, "Handle_Write() on closed socket, connection %d", Idx); +#endif return false; } assert( My_Connections[Idx].sock > NONE ); @@ -920,7 +898,6 @@ New_Connection( int Sock ) #endif struct sockaddr_in new_addr; int new_sock, new_sock_len; - RES_STAT *s; CONN_ID idx; CLIENT *c; POINTER *ptr; @@ -1043,13 +1020,9 @@ New_Connection( int Sock ) /* Hostnamen ermitteln */ strlcpy( My_Connections[idx].host, inet_ntoa( new_addr.sin_addr ), sizeof( My_Connections[idx].host )); Client_SetHostname( c, My_Connections[idx].host ); -#ifdef IDENTAUTH - s = Resolve_Addr( &new_addr, My_Connections[idx].sock ); -#else - s = Resolve_Addr( &new_addr ); -#endif - /* resolver process has been started */ - if( s ) My_Connections[idx].res_stat = s; + + Resolve_Addr(&My_Connections[idx].res_stat, &new_addr, + My_Connections[idx].sock, cb_Read_Resolver_Result); /* Penalty-Zeit setzen */ Conn_SetPenalty( idx, 4 ); @@ -1282,7 +1255,8 @@ Check_Connections( void ) CONN_ID i; for( i = 0; i < Pool_Size; i++ ) { - if( My_Connections[i].sock == NONE ) continue; + if (My_Connections[i].sock < 0) + continue; c = Client_GetFromConn( i ); if( c && (( Client_Type( c ) == CLIENT_USER ) || ( Client_Type( c ) == CLIENT_SERVER ) || ( Client_Type( c ) == CLIENT_SERVICE ))) @@ -1333,16 +1307,17 @@ Check_Servers( void ) { /* Check if we can establish further server links */ - RES_STAT *s; CONN_ID idx; int i, n; + time_t time_now; /* Search all connections, are there results from the resolver? */ for( idx = 0; idx < Pool_Size; idx++ ) { if( My_Connections[idx].sock != SERVER_WAIT ) continue; /* IP resolved? */ - if( My_Connections[idx].res_stat == NULL ) New_Server( Conf_GetServer( idx ), idx ); + if (Resolve_SUCCESS(&My_Connections[idx].res_stat)) + New_Server(Conf_GetServer( idx ), idx); } /* Check all configured servers */ @@ -1354,33 +1329,34 @@ Check_Servers( void ) /* Is there already a connection in this group? */ if( Conf_Server[i].group > NONE ) { - for( n = 0; n < MAX_SERVERS; n++ ) { - if( n == i ) continue; - if(( Conf_Server[n].conn_id > NONE ) && - ( Conf_Server[n].group == Conf_Server[i].group )) + for (n = 0; n < MAX_SERVERS; n++) { + if (n == i) continue; + if ((Conf_Server[n].conn_id > NONE) && + (Conf_Server[n].group == Conf_Server[i].group)) break; } if (n < MAX_SERVERS) continue; } /* Check last connect attempt? */ - if( Conf_Server[i].lasttry > time( NULL ) - Conf_ConnectRetry ) continue; + time_now = time(NULL); + if( Conf_Server[i].lasttry > (time_now - Conf_ConnectRetry)) + continue; /* Okay, try to connect now */ - Conf_Server[i].lasttry = time( NULL ); + Conf_Server[i].lasttry = time_now; /* Search free connection structure */ for( idx = 0; idx < Pool_Size; idx++ ) if( My_Connections[idx].sock == NONE ) break; - if( idx >= Pool_Size ) - { - Log( LOG_ALERT, "Can't establist server connection: connection limit reached (%d)!", Pool_Size ); + if (idx >= Pool_Size) { + Log( LOG_ALERT, "Can't establist server connection: connection limit reached (%d)!", + Pool_Size ); return; } #ifdef DEBUG Log( LOG_DEBUG, "Preparing connection %d for \"%s\" ...", idx, Conf_Server[i].host ); #endif - /* Verbindungs-Struktur initialisieren */ Init_Conn_Struct( idx ); My_Connections[idx].sock = SERVER_WAIT; Conf_Server[i].conn_id = idx; @@ -1388,10 +1364,10 @@ Check_Servers( void ) /* Resolve Hostname. If this fails, try to use it as an IP address */ strlcpy( Conf_Server[i].ip, Conf_Server[i].host, sizeof( Conf_Server[i].ip )); strlcpy( My_Connections[idx].host, Conf_Server[i].host, sizeof( My_Connections[idx].host )); - s = Resolve_Name( Conf_Server[i].host ); - /* resolver process running? */ - if( s ) My_Connections[idx].res_stat = s; + assert(Resolve_Getfd(&My_Connections[idx].res_stat) < 0); + + Resolve_Name(&My_Connections[idx].res_stat, Conf_Server[i].host, cb_Read_Resolver_Result); } } /* Check_Servers */ @@ -1409,15 +1385,6 @@ New_Server( int Server, CONN_ID Idx ) assert( Server > NONE ); assert( Idx > NONE ); - /* Did we get a valid IP address? */ - if( ! Conf_Server[Server].ip[0] ) { - /* No. Free connection structure and abort: */ - Log( LOG_ERR, "Can't connect to \"%s\": ip address unknown!", Conf_Server[Server].host ); - Init_Conn_Struct( Idx ); - Conf_Server[Server].conn_id = NONE; - return; - } - Log( LOG_INFO, "Establishing connection to \"%s\", %s, port %d ... ", Conf_Server[Server].host, Conf_Server[Server].ip, Conf_Server[Server].port ); @@ -1501,6 +1468,7 @@ Init_Conn_Struct( CONN_ID Idx ) My_Connections[Idx].sock = NONE; My_Connections[Idx].lastdata = now; My_Connections[Idx].lastprivmsg = now; + Resolve_Init(&My_Connections[Idx].res_stat); } /* Init_Conn_Struct */ @@ -1542,128 +1510,86 @@ Init_Socket( int Sock ) } /* Init_Socket */ -GLOBAL -void Read_Resolver_Result( int r_fd ) +static void +cb_Read_Resolver_Result( int r_fd, UNUSED short events ) { /* Read result of resolver sub-process from pipe and update the * apropriate connection/client structure(s): hostname and/or * IDENT user name.*/ CLIENT *c; - int bytes_read, i, n; - unsigned int len; - RES_STAT *s; - char *ptr; - char *bufptr; - char readbuf[HOST_LEN]; + int i, n; + size_t len; + char *identptr; +#ifdef IDENTAUTH + char readbuf[HOST_LEN + 2 + CLIENT_USER_LEN]; +#else + char readbuf[HOST_LEN + 1]; +#endif + +#ifdef DEBUG + Log( LOG_DEBUG, "Resolver: Got callback on fd %d, events %d", r_fd, events ); +#endif - Log( LOG_DEBUG, "Resolver: started, fd %d", r_fd ); /* Search associated connection ... */ for( i = 0; i < Pool_Size; i++ ) { if(( My_Connections[i].sock != NONE ) - && ( My_Connections[i].res_stat != NULL ) - && ( My_Connections[i].res_stat->pipe[0] == r_fd )) + && ( Resolve_Getfd(&My_Connections[i].res_stat) == r_fd )) break; } - if( i >= Pool_Size ) - { + if( i >= Pool_Size ) { /* Ops, none found? Probably the connection has already * been closed!? We'll ignore that ... */ io_close( r_fd ); #ifdef DEBUG - Log( LOG_DEBUG, "Resolver: Got result for unknown connection!?" ); + Log( LOG_DEBUG, "Resolver: Got callback for unknown connection!?" ); #endif return; } - /* Get resolver structure */ - s = My_Connections[i].res_stat; - assert( s != NULL ); - /* Read result from pipe */ - bytes_read = read( r_fd, readbuf, sizeof readbuf -1 ); - if( bytes_read < 0 ) { - /* Error! */ - Log( LOG_CRIT, "Resolver: Can't read result: %s!", strerror( errno )); - FreeRes_stat( &My_Connections[i] ); - return; - } - len = (unsigned int) bytes_read; - readbuf[len] = '\0'; - if (!array_catb(&s->buffer, readbuf, len)) { - Log( LOG_CRIT, "Resolver: Can't append result %s to buffer: %s", readbuf, strerror( errno )); - FreeRes_stat(&My_Connections[i]); + len = Resolve_Read(&My_Connections[i].res_stat, readbuf, sizeof readbuf -1); + if (len == 0) return; - } - if (!array_cat0_temporary(&s->buffer)) { - Log( LOG_CRIT, "Resolver: Can't append result %s to buffer: %s", readbuf, strerror( errno )); - FreeRes_stat(&My_Connections[i]); + readbuf[len] = '\0'; + identptr = strchr(readbuf, '\n'); + assert(identptr != NULL); + if (!identptr) { + Log( LOG_CRIT, "Resolver: Got malformed result!"); return; } - /* If the result string is incomplete, return to main loop and - * wait until we can read in more bytes. */ -#ifdef IDENTAUTH -try_resolve: -#endif - bufptr = (char*) array_start(&s->buffer); - assert(bufptr != NULL); - ptr = strchr( bufptr, '\n' ); - if( ! ptr ) return; - *ptr = '\0'; - + *identptr = '\0'; #ifdef DEBUG - Log( LOG_DEBUG, "Got result from resolver: \"%s\" (%u bytes read), stage %d.", bufptr, len, s->stage); + Log( LOG_DEBUG, "Got result from resolver: \"%s\" (%u bytes read).", readbuf, len); #endif - /* Okay, we got a complete result: this is a host name for outgoing - * connections and a host name or IDENT user name (if enabled) for + * connections and a host name and IDENT user name (if enabled) for * incoming connections.*/ - if( My_Connections[i].sock > NONE ) - { + if( My_Connections[i].sock > NONE ) { /* Incoming connection. Search client ... */ c = Client_GetFromConn( i ); assert( c != NULL ); /* Only update client information of unregistered clients */ - if( Client_Type( c ) == CLIENT_UNKNOWN ) - { - switch(s->stage) { - case 0: /* host name */ - strlcpy( My_Connections[i].host, bufptr, sizeof( My_Connections[i].host)); - - Client_SetHostname( c, bufptr); + if( Client_Type( c ) == CLIENT_UNKNOWN ) { + strlcpy(My_Connections[i].host, readbuf, sizeof( My_Connections[i].host)); + Client_SetHostname( c, readbuf); #ifdef IDENTAUTH - /* clean up buffer for IDENT result */ - len = strlen(bufptr) + 1; - assert(len <= array_bytes(&s->buffer)); - array_moveleft(&s->buffer, 1, len); - - /* Don't close pipe and clean up, but - * instead wait for IDENT result */ - s->stage = 1; - goto try_resolve; - - case 1: /* IDENT user name */ - if (array_bytes(&s->buffer)) { - bufptr = (char*) array_start(&s->buffer); - Log( LOG_INFO, "IDENT lookup for connection %ld: \"%s\".", i, bufptr); - Client_SetUser( c, bufptr, true ); - } - else Log( LOG_INFO, "IDENT lookup for connection %ld: no result.", i ); -#endif - break; - default: - Log( LOG_ERR, "Resolver: got result for unknown stage %d!?", s->stage ); + ++identptr; + if (*identptr) { + Log( LOG_INFO, "IDENT lookup for connection %ld: \"%s\".", i, identptr); + Client_SetUser( c, identptr, true ); + } else { + Log( LOG_INFO, "IDENT lookup for connection %ld: no result.", i ); } +#endif } #ifdef DEBUG else Log( LOG_DEBUG, "Resolver: discarding result for already registered connection %d.", i ); #endif - } - else - { + } else { /* Outgoing connection (server link): set the IP address * so that we can connect to it in the main loop. */ @@ -1671,16 +1597,12 @@ try_resolve: n = Conf_GetServer( i ); assert( n > NONE ); - bufptr = (char*) array_start(&s->buffer); - strlcpy( Conf_Server[n].ip, bufptr, sizeof( Conf_Server[n].ip )); + strlcpy( Conf_Server[n].ip, readbuf, sizeof( Conf_Server[n].ip )); } - /* Clean up ... */ - FreeRes_stat( &My_Connections[i] ); - /* Reset penalty time */ Conn_ResetPenalty( i ); -} /* Read_Resolver_Result */ +} /* cb_Read_Resolver_Result */ static void @@ -1705,8 +1627,7 @@ Count_Connections( struct sockaddr_in addr_in ) int i, cnt; cnt = 0; - for( i = 0; i < Pool_Size; i++ ) - { + for( i = 0; i < Pool_Size; i++ ) { if(( My_Connections[i].sock > NONE ) && ( My_Connections[i].addr.sin_addr.s_addr == addr_in.sin_addr.s_addr )) cnt++; } return cnt;