Subject: Re: select call implementation and threads
To: Andrey Petrov <petrov@netbsd.org>
From: malleswararao venkatanaga <bobbyavn@yahoo.com>
List: tech-kern
Date: 05/24/2003 04:26:11
Here is the code i tested with.
/* Recv.c This is the code that receives messages */
#include <stdio.h>
#include<fcntl.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/select.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include <pthread.h>

void * stub_test (void *p );
pthread_t  threads[3];
int main()
{
  int rc;
  stub_test2();
  rc = pthread_create(&threads[0], NULL , stub_test
,(void *) &threads[0]);
  if(rc)
  {
    printf("Error in creating thread\n");
    exit(0);
  }

  rc = pthread_join( &threads[0], NULL);
  if(rc)
  {
    printf("Error in creating thread\n");
    exit(0);
  }
}


void * stub_test (void *p )
{
 struct sockaddr_in     saddr;
 int sock_fd, i, nread, ndfs, ret_val;
 fd_set   read_set;
 char buff[1000];
 struct timeval timeval;
 
 saddr.sin_family =    AF_INET;
 saddr.sin_port   =    45000;
 saddr.sin_addr.s_addr = htonl ( INADDR_ANY);
 if( (sock_fd = socket( AF_INET , SOCK_DGRAM , 0 ) ) <
0 )
     return -1;

 if ( bind ( sock_fd , (struct sockaddr * ) &saddr,
sizeof(saddr ) ) < 0)
     return -1;

  ndfs = sock_fd + 1;
  while(1)
  {
     FD_ZERO ( &read_set);
     FD_SET (sock_fd , &read_set);
     ret_val = select(ndfs, &read_set, NULL, NULL,
NULL);
     if(ret_val >0)
     {
          if(FD_ISSET(sock_fd, &read_set))
          {

           nread = recvfrom ( sock_fd , buff , 10, 0
,NULL, NULL);
           printf("Read bytes thr 2 %d\n", nread);
           for(i = 0; i <5; i++)
           {
              printf("Buff contents = %c\n", buff[i]);
           }
           }
      }
      else
     {
        printf("\n failed in select");
     }
   } 
  return 0;
}
void stub_test2()
{
   struct sockaddr_in   saddr2;
   fd_set  read_set;
   int sock_fd, ret_val, ndfs;

 struct timeval timeval;

 timeval.tv_sec  = 10;
 timeval.tv_usec = 0;

 saddr2.sin_family =    AF_INET;
 saddr2.sin_port   =    45001;
 saddr2.sin_addr.s_addr = htonl ( INADDR_ANY);

  if( (sock_fd = socket( AF_INET , SOCK_DGRAM , 0 ) )
< 0 )
     return -1;

  if ( bind ( sock_fd , (struct sockaddr * ) &saddr2,
sizeof(saddr2 ) ) < 0)
               return -1;

 ndfs = sock_fd + 1;
 while(1)
 {
   FD_ZERO ( &read_set);
   FD_SET (sock_fd , &read_set);
    ret_val = select(ndfs, &read_set, NULL, NULL,NULL
);
    if(ret_val >0)
    {
       if(FD_ISSET(sock_fd, &read_set))
       {
            int nread;
            char buf[1000];
            printf("\n Recvd some message  main thread
\n");
            nread = read(sock_fd, buf, sizeof(buf));
            if(nread>0)
                  {
              printf("\n read %i bytes", nread);
            }
       }
    }
    else
     {
        printf("\n failed in select 1");
     }
 }   
 
}


/* The code that sends messages on both the ports */

#include <stdio.h>
#include <sys/errno.h>
#include <sys/types.h>
#include<sys/socket.h>
#include<sys/select.h>
#include<netinet/in.h>



main()
{
        int                             sockfd,
nwritten , i=1;
        const char                      on = 1;
        struct sockaddr_in               saddr;
        unsigned char                    *pbuf;

        pbuf = (unsigned char*)malloc(5);
        strcpy(pbuf, "hello");

        saddr.sin_family                = AF_INET;
        saddr.sin_addr.s_addr   = INADDR_ANY;

        if ( ( sockfd = socket ( AF_INET, SOCK_DGRAM,
0 ) ) < 0 )
        {
                return -1;
        }
        for ( ;  ; )
        {
         if(i==0)
         {
        saddr.sin_port                  = 45000;
         i = 1;
         }
         else
         {
         {
         saddr.sin_port                  = 45001;
           i=0;
         }
        nwritten        = sendto ( sockfd, pbuf, 5,
0,(struct sockaddr *)&saddr,

                                               
(size_t)(sizeof (struct sockaddr_in)) );

       if ( nwritten < 0 )
        {
                return -1;
        }
           printf("\n sent message ");
           sleep(10);
        }

   return;
}



--- Andrey Petrov <petrov@netbsd.org> wrote:
> On Fri, May 23, 2003 at 09:50:02AM -0700,
> malleswararao venkatanaga wrote:
> > Hi,
> >  I'm facing a problem and it is somewhat like
> this,
> >   1)I spawn a thread using pthread_create()
> >   2) In the thread I open a socket, bind it to a
> port
> > say 10001 and block it in the select call for
> > receiving a UDP message.
> >   Similarly 
> >   In my main thread i open a socket and bind it to
> a 
> > port say 10000 and block the main thread in a
> select
> > call.
> >   Both the main thread and the other thread are
> > essentially waiting for different messages on
> > different ports and different socket fds with
> their
> > respective select calls.
> >   I observe that if i send a messages to both of
> them
> > only one of them receives while the other one is
> still
> > blocked in the select call though it has messages
> to
> > read. 
> > 
> 
> Good thing about userland's threads is that it's
> easy to debug:
> build pth library and your applicarion with '-g' and
> you'll
> get the full picture of what's going on.
> 
> The behavouir you described is strange: select is
> normally
> redefined to pth_select and that supposed to take
> care of swiching
> betwean threads. Well, you can post your application
> here...
> 
> 	Andrey
> 


__________________________________
Do you Yahoo!?
The New Yahoo! Search - Faster. Easier. Bingo.
http://search.yahoo.com