home *** CD-ROM | disk | FTP | other *** search
/ Geek Gadgets 1 / ADE-1.bin / ade-dist / inetutils-1.2-src.tgz / tar.out / fsf / inetutils / talk / ctl_transact.c next >
C/C++ Source or Header  |  1996-09-28  |  4KB  |  121 lines

  1. /*
  2.  * Copyright (c) 1983, 1993
  3.  *    The Regents of the University of California.  All rights reserved.
  4.  *
  5.  * Redistribution and use in source and binary forms, with or without
  6.  * modification, are permitted provided that the following conditions
  7.  * are met:
  8.  * 1. Redistributions of source code must retain the above copyright
  9.  *    notice, this list of conditions and the following disclaimer.
  10.  * 2. Redistributions in binary form must reproduce the above copyright
  11.  *    notice, this list of conditions and the following disclaimer in the
  12.  *    documentation and/or other materials provided with the distribution.
  13.  * 3. All advertising materials mentioning features or use of this software
  14.  *    must display the following acknowledgement:
  15.  *    This product includes software developed by the University of
  16.  *    California, Berkeley and its contributors.
  17.  * 4. Neither the name of the University nor the names of its contributors
  18.  *    may be used to endorse or promote products derived from this software
  19.  *    without specific prior written permission.
  20.  *
  21.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  22.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  23.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  24.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  25.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  26.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  27.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  28.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  29.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  30.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  31.  * SUCH DAMAGE.
  32.  */
  33.  
  34. #ifndef lint
  35. static char sccsid[] = "@(#)ctl_transact.c    8.1 (Berkeley) 6/6/93";
  36. #endif /* not lint */
  37.  
  38. #ifdef HAVE_CONFIG_H
  39. #include <config.h>
  40. #endif
  41.  
  42. #include <sys/types.h>
  43. #include <sys/socket.h>
  44. #include <sys/time.h>
  45. #include <netinet/in.h>
  46. #ifdef HAVE_OSOCKADDR_H
  47. #include <osockaddr.h>
  48. #endif
  49. #include <protocols/talkd.h>
  50. #include <errno.h>
  51. #include "talk_ctl.h"
  52.  
  53. #define CTL_WAIT 2    /* time to wait for a response, in seconds */
  54.  
  55. /*
  56.  * SOCKDGRAM is unreliable, so we must repeat messages if we have
  57.  * not recieved an acknowledgement within a reasonable amount
  58.  * of time
  59.  */
  60. ctl_transact(target, msg, type, rp)
  61.     struct in_addr target;
  62.     CTL_MSG msg;
  63.     int type;
  64.     CTL_RESPONSE *rp;
  65. {
  66.     int read_mask, ctl_mask, nready, cc;
  67.     struct timeval wait;
  68.  
  69.     msg.type = type;
  70.     daemon_addr.sin_addr = target;
  71.     daemon_addr.sin_port = daemon_port;
  72.     ctl_mask = 1 << ctl_sockt;
  73.  
  74.     /*
  75.      * Keep sending the message until a response of
  76.      * the proper type is obtained.
  77.      */
  78.     do {
  79.         wait.tv_sec = CTL_WAIT;
  80.         wait.tv_usec = 0;
  81.         /* resend message until a response is obtained */
  82.         do {
  83.             cc = sendto(ctl_sockt, (char *)&msg, sizeof (msg), 0,
  84.                 (struct sockaddr *)&daemon_addr,
  85.                 sizeof (daemon_addr));
  86.             if (cc != sizeof (msg)) {
  87.                 if (errno == EINTR)
  88.                     continue;
  89.                 p_error("Error on write to talk daemon");
  90.             }
  91.             read_mask = ctl_mask;
  92.             nready = select(32, &read_mask, 0, 0, &wait);
  93.             if (nready < 0) {
  94.                 if (errno == EINTR)
  95.                     continue;
  96.                 p_error("Error waiting for daemon response");
  97.             }
  98.         } while (nready == 0);
  99.         /*
  100.          * Keep reading while there are queued messages 
  101.          * (this is not necessary, it just saves extra
  102.          * request/acknowledgements being sent)
  103.          */
  104.         do {
  105.             cc = recv(ctl_sockt, (char *)rp, sizeof (*rp), 0);
  106.             if (cc < 0) {
  107.                 if (errno == EINTR)
  108.                     continue;
  109.                 p_error("Error on read from talk daemon");
  110.             }
  111.             read_mask = ctl_mask;
  112.             /* an immediate poll */
  113.             timerclear(&wait);
  114.             nready = select(32, &read_mask, 0, 0, &wait);
  115.         } while (nready > 0 && (rp->vers != TALK_VERSION ||
  116.             rp->type != type));
  117.     } while (rp->vers != TALK_VERSION || rp->type != type);
  118.     rp->id_num = ntohl(rp->id_num);
  119.     rp->addr.sa_family = ntohs(rp->addr.sa_family);
  120. }
  121.