+} /* Check_Connections */
+
+
+LOCAL VOID
+Check_Servers( VOID )
+{
+ /* Pruefen, ob Server-Verbindungen aufgebaut werden
+ * muessen bzw. koennen */
+
+ RES_STAT *s;
+ LONG idx, n;
+ INT i;
+
+ /* Wenn "Passive-Mode" aktiv: nicht verbinden */
+ if( NGIRCd_Passive ) return;
+
+ for( i = 0; i < Conf_Server_Count; i++ )
+ {
+ /* Ist ein Hostname und Port definiert? */
+ if(( ! Conf_Server[i].host[0] ) || ( ! Conf_Server[i].port > 0 )) continue;
+
+ /* Haben wir schon eine Verbindung? */
+ for( n = 0; n < Pool_Size; n++ )
+ {
+ if( My_Connections[n].sock == NONE ) continue;
+
+ /* Verbindung zu diesem Server? */
+ if( My_Connections[n].our_server == i )
+ {
+ /* Komplett aufgebaute Verbindung? */
+ if( My_Connections[n].sock > NONE ) break;
+
+ /* IP schon aufgeloest? */
+ if( My_Connections[n].res_stat == NULL ) New_Server( i, n );
+ }
+
+ /* Verbindung in dieser Server-Gruppe? */
+ if(( My_Connections[n].our_server != NONE ) && ( Conf_Server[i].group != NONE ))
+ {
+ if( Conf_Server[My_Connections[n].our_server].group == Conf_Server[i].group ) break;
+ }
+ }
+ if( n < Pool_Size ) continue;
+
+ /* Wann war der letzte Connect-Versuch? */
+ if( Conf_Server[i].lasttry > time( NULL ) - Conf_ConnectRetry ) continue;
+
+ /* Okay, Verbindungsaufbau versuchen */
+ Conf_Server[i].lasttry = time( NULL );
+
+ /* Freie Connection-Struktur suschen */
+ 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 );
+ return;
+ }
+ Log( LOG_DEBUG, "Preparing connection %d for \"%s\" ...", idx, Conf_Server[i].host );
+
+ /* Verbindungs-Struktur initialisieren */
+ Init_Conn_Struct( idx );
+ My_Connections[idx].sock = SERVER_WAIT;
+ My_Connections[idx].our_server = i;
+
+ /* Hostnamen in IP aufloesen (Default bzw. im Fehlerfall: versuchen, den
+ * konfigurierten Text direkt als IP-Adresse zu verwenden ... */
+ strcpy( Conf_Server[My_Connections[idx].our_server].ip, Conf_Server[i].host );
+ strcpy( My_Connections[idx].host, Conf_Server[i].host );
+ s = Resolve_Name( Conf_Server[i].host );
+ if( s )
+ {
+ /* Sub-Prozess wurde asyncron gestartet */
+ My_Connections[idx].res_stat = s;
+ }
+ }
+} /* Check_Servers */
+
+
+LOCAL VOID
+New_Server( INT Server, CONN_ID Idx )
+{
+ /* Neue Server-Verbindung aufbauen */
+
+ struct sockaddr_in new_addr;
+ struct in_addr inaddr;
+ INT res, new_sock;
+ CLIENT *c;
+
+ assert( Server > NONE );
+ assert( Idx > NONE );
+
+ /* Wurde eine gueltige IP-Adresse gefunden? */
+ if( ! Conf_Server[Server].ip[0] )
+ {
+ /* Nein. Verbindung wieder freigeben: */
+ Init_Conn_Struct( Idx );
+ Log( LOG_ERR, "Can't connect to \"%s\" (connection %d): ip address unknown!", Conf_Server[Server].host, Idx );
+ return;
+ }
+
+ Log( LOG_INFO, "Establishing connection to \"%s\", %s, port %d (connection %d) ... ", Conf_Server[Server].host, Conf_Server[Server].ip, Conf_Server[Server].port, Idx );
+
+#ifdef HAVE_INET_ATON
+ if( inet_aton( Conf_Server[Server].ip, &inaddr ) == 0 )
+#else
+ memset( &inaddr, 0, sizeof( inaddr ));
+ inaddr.s_addr = inet_addr( Conf_Server[Server].ip );
+ if( inaddr.s_addr == (unsigned)-1 )
+#endif
+ {
+ /* Konnte Adresse nicht konvertieren */
+ Init_Conn_Struct( Idx );
+ Log( LOG_ERR, "Can't connect to \"%s\" (connection %d): can't convert ip address %s!", Conf_Server[Server].host, Idx, Conf_Server[Server].ip );
+ return;
+ }
+
+ memset( &new_addr, 0, sizeof( new_addr ));
+ new_addr.sin_family = AF_INET;
+ new_addr.sin_addr = inaddr;
+ new_addr.sin_port = htons( Conf_Server[Server].port );
+
+ new_sock = socket( PF_INET, SOCK_STREAM, 0 );
+ if ( new_sock < 0 )
+ {
+ Init_Conn_Struct( Idx );
+ Log( LOG_CRIT, "Can't create socket: %s!", strerror( errno ));
+ return;
+ }
+
+ if( ! Init_Socket( new_sock )) return;
+
+ res = connect( new_sock, (struct sockaddr *)&new_addr, sizeof( new_addr ));
+ if(( res != 0 ) && ( errno != EINPROGRESS ))
+ {
+ Log( LOG_CRIT, "Can't connect socket: %s!", strerror( errno ));
+ close( new_sock );
+ Init_Conn_Struct( Idx );
+ return;
+ }
+
+ /* Client-Struktur initialisieren */
+ c = Client_NewLocal( Idx, inet_ntoa( new_addr.sin_addr ), CLIENT_UNKNOWNSERVER, FALSE );
+ if( ! c )
+ {
+ close( new_sock );
+ Init_Conn_Struct( Idx );
+ Log( LOG_ALERT, "Can't establish connection: can't create client structure!" );
+ return;
+ }
+ Client_SetIntroducer( c, c );
+ Client_SetToken( c, TOKEN_OUTBOUND );
+
+ /* Verbindung registrieren */
+ My_Connections[Idx].sock = new_sock;
+ My_Connections[Idx].addr = new_addr;
+ strcpy( My_Connections[Idx].host, Conf_Server[Server].host );
+
+ /* Neuen Socket registrieren */
+ FD_SET( new_sock, &My_Sockets );
+ FD_SET( new_sock, &My_Connects );
+ if( new_sock > Conn_MaxFD ) Conn_MaxFD = new_sock;
+
+ Log( LOG_DEBUG, "Registered new connection %d on socket %d.", Idx, My_Connections[Idx].sock );
+} /* New_Server */
+
+
+LOCAL VOID
+Init_Conn_Struct( LONG Idx )
+{
+ /* Connection-Struktur initialisieren */
+
+ My_Connections[Idx].sock = NONE;
+ My_Connections[Idx].res_stat = NULL;
+ My_Connections[Idx].host[0] = '\0';
+ My_Connections[Idx].rbuf[0] = '\0';
+ My_Connections[Idx].rdatalen = 0;
+ My_Connections[Idx].wbuf[0] = '\0';
+ My_Connections[Idx].wdatalen = 0;
+ My_Connections[Idx].our_server = NONE;
+ My_Connections[Idx].starttime = time( NULL );
+ My_Connections[Idx].lastdata = time( NULL );
+ My_Connections[Idx].lastping = 0;
+ My_Connections[Idx].lastprivmsg = time( NULL );
+ My_Connections[Idx].delaytime = 0;
+ My_Connections[Idx].bytes_in = 0;
+ My_Connections[Idx].bytes_out = 0;
+ My_Connections[Idx].msg_in = 0;
+ My_Connections[Idx].msg_out = 0;
+ My_Connections[Idx].flag = 0;
+ My_Connections[Idx].options = 0;
+
+#ifdef USE_ZLIB
+ My_Connections[Idx].zip.rbuf[0] = '\0';
+ My_Connections[Idx].zip.rdatalen = 0;
+ My_Connections[Idx].zip.wbuf[0] = '\0';
+ My_Connections[Idx].zip.wdatalen = 0;
+ My_Connections[Idx].zip.bytes_in = 0;
+ My_Connections[Idx].zip.bytes_out = 0;
+#endif
+} /* Init_Conn_Struct */
+
+
+LOCAL BOOLEAN
+Init_Socket( INT Sock )
+{
+ /* Socket-Optionen setzen */
+
+ INT on = 1;
+
+#ifdef O_NONBLOCK /* A/UX kennt das nicht? */
+ if( fcntl( Sock, F_SETFL, O_NONBLOCK ) != 0 )
+ {
+ Log( LOG_CRIT, "Can't enable non-blocking mode: %s!", strerror( errno ));
+ close( Sock );
+ return FALSE;
+ }
+#endif
+ if( setsockopt( Sock, SOL_SOCKET, SO_REUSEADDR, &on, (socklen_t)sizeof( on )) != 0)
+ {
+ Log( LOG_ERR, "Can't set socket options: %s!", strerror( errno ));
+ /* dieser Fehler kann ignoriert werden. */
+ }
+
+ return TRUE;
+} /* Init_Socket */
+
+
+LOCAL VOID
+Read_Resolver_Result( INT r_fd )
+{
+ /* Ergebnis von Resolver Sub-Prozess aus Pipe lesen
+ * und entsprechende Connection aktualisieren */
+
+ CHAR result[HOST_LEN];
+ CLIENT *c;
+ INT len, i;
+
+ FD_CLR( r_fd, &Resolver_FDs );
+
+ /* Anfrage vom Parent lesen */
+ len = read( r_fd, result, HOST_LEN - 1 );
+ if( len < 0 )
+ {
+ /* Fehler beim Lesen aus der Pipe */
+ close( r_fd );
+ Log( LOG_CRIT, "Resolver: Can't read result: %s!", strerror( errno ));
+ return;
+ }
+ result[len] = '\0';
+
+ /* zugehoerige Connection suchen */
+ for( i = 0; i < Pool_Size; i++ )
+ {
+ if(( My_Connections[i].sock != NONE ) && ( My_Connections[i].res_stat ) && ( My_Connections[i].res_stat->pipe[0] == r_fd )) break;
+ }
+ if( i >= Pool_Size )
+ {
+ /* Opsa! Keine passende Connection gefunden!? Vermutlich
+ * wurde sie schon wieder geschlossen. */
+ close( r_fd );
+ Log( LOG_DEBUG, "Resolver: Got result for unknown connection!?" );
+ return;
+ }
+
+ Log( LOG_DEBUG, "Resolver: %s is \"%s\".", My_Connections[i].host, result );
+
+ /* Aufraeumen */
+ close( My_Connections[i].res_stat->pipe[0] );
+ close( My_Connections[i].res_stat->pipe[1] );
+ free( My_Connections[i].res_stat );
+ My_Connections[i].res_stat = NULL;
+
+ if( My_Connections[i].sock > NONE )
+ {
+ /* Eingehende Verbindung: Hostnamen setzen */
+ c = Client_GetFromConn( i );
+ assert( c != NULL );
+ strcpy( My_Connections[i].host, result );
+ Client_SetHostname( c, result );
+ }
+ else
+ {
+ /* Ausgehende Verbindung (=Server): IP setzen */
+ assert( My_Connections[i].our_server > NONE );
+ strcpy( Conf_Server[My_Connections[i].our_server].ip, result );
+ }
+
+ /* Penalty-Zeit zurueck setzen */
+ Conn_ResetPenalty( i );
+} /* Read_Resolver_Result */
+
+
+#ifdef USE_ZLIB
+
+LOCAL BOOLEAN
+Zip_Buffer( CONN_ID Idx, CHAR *Data, INT Len )
+{
+ /* Daten zum Komprimieren im "Kompressions-Puffer" sammeln.
+ * Es wird TRUE bei Erfolg, sonst FALSE geliefert. */
+
+ assert( Idx > NONE );
+ assert( Data != NULL );
+ assert( Len > 0 );
+
+ /* Ist noch Platz im Kompressions-Puffer? */
+ if( ZWRITEBUFFER_LEN - My_Connections[Idx].zip.wdatalen < Len + 50 )
+ {
+ /* Nein! Puffer zunaechst leeren ...*/
+ if( ! Zip_Flush( Idx )) return FALSE;
+ }
+
+ /* Daten kopieren */
+ memmove( My_Connections[Idx].zip.wbuf + My_Connections[Idx].zip.wdatalen, Data, Len );
+ My_Connections[Idx].zip.wdatalen += Len;
+
+ return TRUE;
+} /* Zip_Buffer */
+
+
+LOCAL BOOLEAN
+Zip_Flush( CONN_ID Idx )
+{
+ /* Daten komprimieren und in Schreibpuffer kopieren.
+ * Es wird TRUE bei Erfolg, sonst FALSE geliefert. */
+
+ INT result, out_len;
+ z_stream *out;
+
+ out = &My_Connections[Idx].zip.out;
+
+ out->next_in = My_Connections[Idx].zip.wbuf;
+ out->avail_in = My_Connections[Idx].zip.wdatalen;
+ out->next_out = My_Connections[Idx].wbuf + My_Connections[Idx].wdatalen;
+ out->avail_out = WRITEBUFFER_LEN - My_Connections[Idx].wdatalen;
+
+ result = deflate( out, Z_SYNC_FLUSH );
+ if(( result != Z_OK ) || ( out->avail_in > 0 ))
+ {
+ Log( LOG_ALERT, "Compression error: code %d!?", result );
+ Conn_Close( Idx, "Compression error!", NULL, FALSE );
+ return FALSE;
+ }
+
+ out_len = WRITEBUFFER_LEN - My_Connections[Idx].wdatalen - out->avail_out;
+ My_Connections[Idx].wdatalen += out_len;
+ My_Connections[Idx].bytes_out += out_len;
+ My_Connections[Idx].zip.bytes_out += My_Connections[Idx].zip.wdatalen;
+ My_Connections[Idx].zip.wdatalen = 0;
+
+ return TRUE;
+} /* Zip_Flush */
+
+
+LOCAL BOOLEAN
+Unzip_Buffer( CONN_ID Idx )
+{
+ /* Daten entpacken und in Lesepuffer kopieren. Bei Fehlern
+ * wird FALSE geliefert, ansonsten TRUE. Der Fall, dass keine
+ * Daten mehr zu entpacken sind, ist _kein_ Fehler! */
+
+ INT result, in_len, out_len;
+ z_stream *in;
+
+ assert( Idx > NONE );
+
+ if( My_Connections[Idx].zip.rdatalen <= 0 ) return TRUE;
+
+ in = &My_Connections[Idx].zip.in;
+
+ in->next_in = My_Connections[Idx].zip.rbuf;
+ in->avail_in = My_Connections[Idx].zip.rdatalen;
+ in->next_out = My_Connections[Idx].rbuf + My_Connections[Idx].rdatalen;
+ in->avail_out = READBUFFER_LEN - My_Connections[Idx].rdatalen - 1;
+
+ result = inflate( in, Z_SYNC_FLUSH );
+ if( result != Z_OK )
+ {
+ Log( LOG_ALERT, "Decompression error: code %d (ni=%d, ai=%d, no=%d, ao=%d)!?", result, in->next_in, in->avail_in, in->next_out, in->avail_out );
+ Conn_Close( Idx, "Decompression error!", NULL, FALSE );
+ return FALSE;
+ }
+
+ in_len = My_Connections[Idx].zip.rdatalen - in->avail_in;
+ out_len = READBUFFER_LEN - My_Connections[Idx].rdatalen - 1 - in->avail_out;
+ My_Connections[Idx].rdatalen += out_len;
+
+ if( in->avail_in > 0 )
+ {
+ /* es konnten nicht alle Daten entpackt werden, vermutlich war
+ * im Ziel-Puffer kein Platz mehr. Umkopieren ... */
+ My_Connections[Idx].zip.rdatalen -= in_len;
+ memmove( My_Connections[Idx].zip.rbuf, My_Connections[Idx].zip.rbuf + in_len, My_Connections[Idx].zip.rdatalen );
+ }
+ else My_Connections[Idx].zip.rdatalen = 0;
+ My_Connections[Idx].zip.bytes_in += out_len;
+
+ return TRUE;
+} /* Unzip_Buffer */
+
+
+#endif