+} /* Init_Socket */
+
+
+LOCAL RES_STAT *ResolveAddr( struct sockaddr_in *Addr )
+{
+ /* IP (asyncron!) aufloesen. Bei Fehler, z.B. wenn der
+ * Child-Prozess nicht erzeugt werden kann, wird NULL geliefert.
+ * Der Host kann dann nicht aufgeloest werden. */
+
+ RES_STAT *s;
+ INT pid;
+
+ /* Speicher anfordern */
+ s = malloc( sizeof( RES_STAT ));
+ if( ! s )
+ {
+ Log( LOG_EMERG, "Resolver: Can't allocate memory!" );
+ return NULL;
+ }
+
+ /* Pipe fuer Antwort initialisieren */
+ if( pipe( s->pipe ) != 0 )
+ {
+ free( s );
+ Log( LOG_ALERT, "Resolver: Can't create output pipe: %s!", strerror( errno ));
+ return NULL;
+ }
+
+ /* Sub-Prozess erzeugen */
+ pid = fork( );
+ if( pid > 0 )
+ {
+ /* Haupt-Prozess */
+ Log( LOG_DEBUG, "Resolver for %s created (PID %d).", inet_ntoa( Addr->sin_addr ), pid );
+ FD_SET( s->pipe[0], &My_Resolvers );
+ if( s->pipe[0] > My_Max_Fd ) My_Max_Fd = s->pipe[0];
+ s->pid = pid;
+ return s;
+ }
+ else if( pid == 0 )
+ {
+ /* Sub-Prozess */
+ Log_Init_Resolver( );
+ Do_ResolveAddr( Addr, s->pipe[1] );
+ Log_Exit_Resolver( );
+ exit( 0 );
+ }
+ else
+ {
+ /* Fehler */
+ free( s );
+ Log( LOG_CRIT, "Resolver: Can't fork: %s!", strerror( errno ));
+ return NULL;
+ }
+} /* ResolveAddr */
+
+
+LOCAL RES_STAT *ResolveName( CHAR *Host )
+{
+ /* Hostnamen (asyncron!) aufloesen. Bei Fehler, z.B. wenn der
+ * Child-Prozess nicht erzeugt werden kann, wird NULL geliefert.
+ * Der Host kann dann nicht aufgeloest werden. */
+
+ RES_STAT *s;
+ INT pid;
+
+ /* Speicher anfordern */
+ s = malloc( sizeof( RES_STAT ));
+ if( ! s )
+ {
+ Log( LOG_EMERG, "Resolver: Can't allocate memory!" );
+ return NULL;
+ }
+
+ /* Pipe fuer Antwort initialisieren */
+ if( pipe( s->pipe ) != 0 )
+ {
+ free( s );
+ Log( LOG_ALERT, "Resolver: Can't create output pipe: %s!", strerror( errno ));
+ return NULL;
+ }
+
+ /* Sub-Prozess erzeugen */
+ pid = fork( );
+ if( pid > 0 )
+ {
+ /* Haupt-Prozess */
+ Log( LOG_DEBUG, "Resolver for \"%s\" created (PID %d).", Host, pid );
+ FD_SET( s->pipe[0], &My_Resolvers );
+ if( s->pipe[0] > My_Max_Fd ) My_Max_Fd = s->pipe[0];
+ s->pid = pid;
+ return s;
+ }
+ else if( pid == 0 )
+ {
+ /* Sub-Prozess */
+ Log_Init_Resolver( );
+ Do_ResolveName( Host, s->pipe[1] );
+ Log_Exit_Resolver( );
+ exit( 0 );
+ }
+ else
+ {
+ /* Fehler */
+ free( s );
+ Log( LOG_CRIT, "Resolver: Can't fork: %s!", strerror( errno ));
+ return NULL;
+ }
+} /* ResolveName */
+
+
+LOCAL VOID Do_ResolveAddr( struct sockaddr_in *Addr, INT w_fd )
+{
+ /* Resolver Sub-Prozess: IP aufloesen und Ergebnis in Pipe schreiben. */
+
+ CHAR hostname[HOST_LEN];
+ struct hostent *h;
+
+ Log_Resolver( LOG_DEBUG, "Now resolving %s ...", inet_ntoa( Addr->sin_addr ));
+
+ /* Namen aufloesen */
+ h = gethostbyaddr( (CHAR *)&Addr->sin_addr, sizeof( Addr->sin_addr ), AF_INET );
+ if( h ) strcpy( hostname, h->h_name );
+ else
+ {
+ Log_Resolver( LOG_WARNING, "Can't resolve address %s: code %s!", inet_ntoa( Addr->sin_addr ), Resolv_Error( h_errno ));
+ strcpy( hostname, inet_ntoa( Addr->sin_addr ));
+ }
+
+ /* Antwort an Parent schreiben */
+ if( write( w_fd, hostname, strlen( hostname ) + 1 ) != ( strlen( hostname ) + 1 ))
+ {
+ Log_Resolver( LOG_CRIT, "Resolver: Can't write to parent: %s!", strerror( errno ));
+ close( w_fd );
+ return;
+ }
+
+ Log_Resolver( LOG_DEBUG, "Ok, translated %s to \"%s\".", inet_ntoa( Addr->sin_addr ), hostname );
+} /* Do_ResolveAddr */
+
+
+LOCAL VOID Do_ResolveName( CHAR *Host, INT w_fd )
+{
+ /* Resolver Sub-Prozess: Name aufloesen und Ergebnis in Pipe schreiben. */
+
+ CHAR ip[16];
+ struct hostent *h;
+ struct in_addr *addr;
+
+ Log_Resolver( LOG_DEBUG, "Now resolving \"%s\" ...", Host );
+
+ /* Namen aufloesen */
+ h = gethostbyname( Host );
+ if( h )
+ {
+ addr = (struct in_addr *)h->h_addr;
+ strcpy( ip, inet_ntoa( *addr ));
+ }
+ else
+ {
+ Log_Resolver( LOG_WARNING, "Can't resolve \"%s\": %s!", Host, Resolv_Error( h_errno ));
+ strcpy( ip, "" );
+ }
+
+ /* Antwort an Parent schreiben */
+ if( write( w_fd, ip, strlen( ip ) + 1 ) != ( strlen( ip ) + 1 ))
+ {
+ Log_Resolver( LOG_CRIT, "Resolver: Can't write to parent: %s!", strerror( errno ));
+ close( w_fd );
+ return;
+ }
+
+ if( ip[0] ) Log_Resolver( LOG_DEBUG, "Ok, translated \"%s\" to %s.", Host, ip );
+} /* Do_ResolveName */
+
+
+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, &My_Resolvers );
+
+ /* Anfrage vom Parent lesen */
+ len = read( r_fd, result, HOST_LEN);
+ 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 < MAX_CONNECTIONS; i++ )
+ {
+ if(( My_Connections[i].sock != NONE ) && ( My_Connections[i].res_stat ) && ( My_Connections[i].res_stat->pipe[0] == r_fd )) break;
+ }
+ if( i >= MAX_CONNECTIONS )
+ {
+ /* Opsa! Keine passende Connection gefunden!? Vermutlich
+ * wurde sie schon wieder geschlossen. */
+ close( r_fd );
+ Log( LOG_DEBUG, "Resolver: Got result for unknown connection!?" );
+ return;
+ }
+
+ /* 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 >= 0 );
+ strcpy( Conf_Server[My_Connections[i].our_server].ip, result );
+ }
+} /* Read_Resolver_Result */
+
+
+LOCAL CHAR *Resolv_Error( INT H_Error )
+{
+ /* Fehlerbeschreibung fuer H_Error liefern */
+
+ switch( H_Error )
+ {
+ case HOST_NOT_FOUND:
+ return "host not found";
+ case NO_DATA:
+ return "name valid but no IP address defined";
+ case NO_RECOVERY:
+ return "name server error";
+ case TRY_AGAIN:
+ return "name server temporary not available";
+ default:
+ return "unknown error";
+ }
+} /* Resolv_Error */