Blame sunrpc/tst-udp-nonblocking.c

Packit Service 82fcde
/* Test non-blocking use of the UDP client.
Packit Service 82fcde
   Copyright (C) 2017-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <netinet/in.h>
Packit Service 82fcde
#include <rpc/clnt.h>
Packit Service 82fcde
#include <rpc/svc.h>
Packit Service 82fcde
#include <stdbool.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
#include <support/check.h>
Packit Service 82fcde
#include <support/namespace.h>
Packit Service 82fcde
#include <support/test-driver.h>
Packit Service 82fcde
#include <support/xsocket.h>
Packit Service 82fcde
#include <support/xunistd.h>
Packit Service 82fcde
#include <sys/socket.h>
Packit Service 82fcde
#include <time.h>
Packit Service 82fcde
#include <unistd.h>
Packit Service 82fcde
Packit Service 82fcde
/* Test data serialization and deserialization.   */
Packit Service 82fcde
Packit Service 82fcde
struct test_query
Packit Service 82fcde
{
Packit Service 82fcde
  uint32_t a;
Packit Service 82fcde
  uint32_t b;
Packit Service 82fcde
  uint32_t timeout_ms;
Packit Service 82fcde
};
Packit Service 82fcde
Packit Service 82fcde
static bool_t
Packit Service 82fcde
xdr_test_query (XDR *xdrs, void *data, ...)
Packit Service 82fcde
{
Packit Service 82fcde
  struct test_query *p = data;
Packit Service 82fcde
  return xdr_uint32_t (xdrs, &p->a)
Packit Service 82fcde
    && xdr_uint32_t (xdrs, &p->b)
Packit Service 82fcde
    && xdr_uint32_t (xdrs, &p->timeout_ms);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
struct test_response
Packit Service 82fcde
{
Packit Service 82fcde
  uint32_t server_id;
Packit Service 82fcde
  uint32_t seq;
Packit Service 82fcde
  uint32_t sum;
Packit Service 82fcde
};
Packit Service 82fcde
Packit Service 82fcde
static bool_t
Packit Service 82fcde
xdr_test_response (XDR *xdrs, void *data, ...)
Packit Service 82fcde
{
Packit Service 82fcde
  struct test_response *p = data;
Packit Service 82fcde
  return xdr_uint32_t (xdrs, &p->server_id)
Packit Service 82fcde
    && xdr_uint32_t (xdrs, &p->seq)
Packit Service 82fcde
    && xdr_uint32_t (xdrs, &p->sum);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Implementation of the test server.  */
Packit Service 82fcde
Packit Service 82fcde
enum
Packit Service 82fcde
  {
Packit Service 82fcde
    /* Number of test servers to run. */
Packit Service 82fcde
    SERVER_COUNT = 3,
Packit Service 82fcde
Packit Service 82fcde
    /* RPC parameters, chosen at random.  */
Packit Service 82fcde
    PROGNUM = 8242,
Packit Service 82fcde
    VERSNUM = 19654,
Packit Service 82fcde
Packit Service 82fcde
    /* Main RPC operation.  */
Packit Service 82fcde
    PROC_ADD = 1,
Packit Service 82fcde
Packit Service 82fcde
    /* Request process termination.  */
Packit Service 82fcde
    PROC_EXIT,
Packit Service 82fcde
Packit Service 82fcde
    /* Special exit status to mark successful processing.  */
Packit Service 82fcde
    EXIT_MARKER = 55,
Packit Service 82fcde
  };
Packit Service 82fcde
Packit Service 82fcde
/* Set by the parent process to tell test servers apart.  */
Packit Service 82fcde
static int server_id;
Packit Service 82fcde
Packit Service 82fcde
/* Implementation of the test server.  */
Packit Service 82fcde
static void
Packit Service 82fcde
server_dispatch (struct svc_req *request, SVCXPRT *transport)
Packit Service 82fcde
{
Packit Service 82fcde
  /* Query sequence number.  */
Packit Service 82fcde
  static uint32_t seq = 0;
Packit Service 82fcde
  ++seq;
Packit Service 82fcde
  static bool proc_add_seen;
Packit Service 82fcde
Packit Service 82fcde
  if (test_verbose)
Packit Service 82fcde
    printf ("info: server_dispatch server_id=%d seq=%u rq_proc=%lu\n",
Packit Service 82fcde
            server_id, seq, request->rq_proc);
Packit Service 82fcde
Packit Service 82fcde
  switch (request->rq_proc)
Packit Service 82fcde
    {
Packit Service 82fcde
    case PROC_ADD:
Packit Service 82fcde
      {
Packit Service 82fcde
        struct test_query query;
Packit Service 82fcde
        memset (&query, 0xc0, sizeof (query));
Packit Service 82fcde
        TEST_VERIFY_EXIT
Packit Service 82fcde
          (svc_getargs (transport, xdr_test_query,
Packit Service 82fcde
                        (void *) &query));
Packit Service 82fcde
Packit Service 82fcde
        if (test_verbose)
Packit Service 82fcde
          printf ("  a=%u b=%u timeout_ms=%u\n",
Packit Service 82fcde
                  query.a, query.b, query.timeout_ms);
Packit Service 82fcde
Packit Service 82fcde
        usleep (query.timeout_ms * 1000);
Packit Service 82fcde
Packit Service 82fcde
        struct test_response response =
Packit Service 82fcde
          {
Packit Service 82fcde
            .server_id = server_id,
Packit Service 82fcde
            .seq = seq,
Packit Service 82fcde
            .sum = query.a + query.b,
Packit Service 82fcde
          };
Packit Service 82fcde
        TEST_VERIFY (svc_sendreply (transport, xdr_test_response,
Packit Service 82fcde
                                    (void *) &response));
Packit Service 82fcde
        if (test_verbose)
Packit Service 82fcde
          printf ("  server id %d response seq=%u sent\n", server_id, seq);
Packit Service 82fcde
        proc_add_seen = true;
Packit Service 82fcde
      }
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case PROC_EXIT:
Packit Service 82fcde
      TEST_VERIFY (proc_add_seen);
Packit Service 82fcde
      TEST_VERIFY (svc_sendreply (transport, (xdrproc_t) xdr_void, NULL));
Packit Service 82fcde
      _exit (EXIT_MARKER);
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    default:
Packit Service 82fcde
      FAIL_EXIT1 ("invalid rq_proc value: %lu", request->rq_proc);
Packit Service 82fcde
      break;
Packit Service 82fcde
    }
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
/* Return the number seconds since an arbitrary point in time.  */
Packit Service 82fcde
static double
Packit Service 82fcde
get_ticks (void)
Packit Service 82fcde
{
Packit Service 82fcde
  {
Packit Service 82fcde
    struct timespec ts;
Packit Service 82fcde
    if (clock_gettime (CLOCK_MONOTONIC, &ts) == 0)
Packit Service 82fcde
      return ts.tv_sec + ts.tv_nsec * 1e-9;
Packit Service 82fcde
  }
Packit Service 82fcde
  {
Packit Service 82fcde
    struct timeval tv;
Packit Service 82fcde
    TEST_VERIFY_EXIT (gettimeofday (&tv, NULL) == 0);
Packit Service 82fcde
    return tv.tv_sec + tv.tv_usec * 1e-6;
Packit Service 82fcde
  }
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
static int
Packit Service 82fcde
do_test (void)
Packit Service 82fcde
{
Packit Service 82fcde
  support_become_root ();
Packit Service 82fcde
  support_enter_network_namespace ();
Packit Service 82fcde
Packit Service 82fcde
  /* Information about the test servers.  */
Packit Service 82fcde
  struct
Packit Service 82fcde
  {
Packit Service 82fcde
    SVCXPRT *transport;
Packit Service 82fcde
    struct sockaddr_in address;
Packit Service 82fcde
    pid_t pid;
Packit Service 82fcde
    uint32_t xid;
Packit Service 82fcde
  } servers[SERVER_COUNT];
Packit Service 82fcde
Packit Service 82fcde
  /* Spawn the test servers.  */
Packit Service 82fcde
  for (int i = 0; i < SERVER_COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      servers[i].transport = svcudp_create (RPC_ANYSOCK);
Packit Service 82fcde
      TEST_VERIFY_EXIT (servers[i].transport != NULL);
Packit Service 82fcde
      servers[i].address = (struct sockaddr_in)
Packit Service 82fcde
        {
Packit Service 82fcde
          .sin_family = AF_INET,
Packit Service 82fcde
          .sin_addr.s_addr = htonl (INADDR_LOOPBACK),
Packit Service 82fcde
          .sin_port = htons (servers[i].transport->xp_port),
Packit Service 82fcde
        };
Packit Service 82fcde
      servers[i].xid = 0xabcd0101 + i;
Packit Service 82fcde
      if (test_verbose)
Packit Service 82fcde
        printf ("info: setting up server %d xid=%x on port %d\n",
Packit Service 82fcde
                i, servers[i].xid, servers[i].transport->xp_port);
Packit Service 82fcde
Packit Service 82fcde
      server_id = i;
Packit Service 82fcde
      servers[i].pid = xfork ();
Packit Service 82fcde
      if (servers[i].pid == 0)
Packit Service 82fcde
        {
Packit Service 82fcde
          TEST_VERIFY (svc_register (servers[i].transport,
Packit Service 82fcde
                                     PROGNUM, VERSNUM, server_dispatch, 0));
Packit Service 82fcde
          svc_run ();
Packit Service 82fcde
          FAIL_EXIT1 ("supposed to be unreachable");
Packit Service 82fcde
        }
Packit Service 82fcde
      /* We need to close the socket so that we do not accidentally
Packit Service 82fcde
         consume the request.  */
Packit Service 82fcde
      TEST_VERIFY (close (servers[i].transport->xp_sock) == 0);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
  /* The following code mirrors what ypbind does.  */
Packit Service 82fcde
Packit Service 82fcde
  /* Copied from clnt_udp.c (like ypbind).  */
Packit Service 82fcde
  struct cu_data
Packit Service 82fcde
  {
Packit Service 82fcde
    int cu_sock;
Packit Service 82fcde
    bool_t cu_closeit;
Packit Service 82fcde
    struct sockaddr_in cu_raddr;
Packit Service 82fcde
    int cu_rlen;
Packit Service 82fcde
    struct timeval cu_wait;
Packit Service 82fcde
    struct timeval cu_total;
Packit Service 82fcde
    struct rpc_err cu_error;
Packit Service 82fcde
    XDR cu_outxdrs;
Packit Service 82fcde
    u_int cu_xdrpos;
Packit Service 82fcde
    u_int cu_sendsz;
Packit Service 82fcde
    char *cu_outbuf;
Packit Service 82fcde
    u_int cu_recvsz;
Packit Service 82fcde
    char cu_inbuf[1];
Packit Service 82fcde
  };
Packit Service 82fcde
Packit Service 82fcde
  int client_socket = xsocket (AF_INET, SOCK_DGRAM | SOCK_NONBLOCK, 0);
Packit Service 82fcde
  CLIENT *clnt = clntudp_create (&servers[0].address, PROGNUM, VERSNUM,
Packit Service 82fcde
                                 /* 5 seconds per-response timeout.  */
Packit Service 82fcde
                                 ((struct timeval) { 5, 0 }),
Packit Service 82fcde
                                 &client_socket);
Packit Service 82fcde
  TEST_VERIFY (clnt != NULL);
Packit Service 82fcde
  clnt->cl_auth = authunix_create_default ();
Packit Service 82fcde
  {
Packit Service 82fcde
    struct timeval zero = { 0, 0 };
Packit Service 82fcde
    TEST_VERIFY (clnt_control (clnt, CLSET_TIMEOUT, (void *) &zero));
Packit Service 82fcde
  }
Packit Service 82fcde
Packit Service 82fcde
  /* Poke at internal data structures (like ypbind).  */
Packit Service 82fcde
  struct cu_data *cu = (struct cu_data *) clnt->cl_private;
Packit Service 82fcde
Packit Service 82fcde
  /* Send a ping to each server.  */
Packit Service 82fcde
  double before_pings = get_ticks ();
Packit Service 82fcde
  for (int i = 0; i < SERVER_COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (test_verbose)
Packit Service 82fcde
        printf ("info: sending server %d ping\n", i);
Packit Service 82fcde
      /* Reset the xid because it is changed by each invocation of
Packit Service 82fcde
         clnt_call.  Subtract one to compensate for the xid update
Packit Service 82fcde
         during the call.  */
Packit Service 82fcde
      *((uint32_t *) (cu->cu_outbuf)) = servers[i].xid - 1;
Packit Service 82fcde
      cu->cu_raddr = servers[i].address;
Packit Service 82fcde
Packit Service 82fcde
      struct test_query query = { .a = 100, .b = i + 1 };
Packit Service 82fcde
      if (i == 1)
Packit Service 82fcde
        /* Shorter timeout to prefer this server.  These timeouts must
Packit Service 82fcde
           be much shorter than the 5-second per-response timeout
Packit Service 82fcde
           configured with clntudp_create.  */
Packit Service 82fcde
        query.timeout_ms = 750;
Packit Service 82fcde
      else
Packit Service 82fcde
        query.timeout_ms = 1500;
Packit Service 82fcde
      struct test_response response = { 0 };
Packit Service 82fcde
      /* NB: Do not check the return value.  The server reply will
Packit Service 82fcde
         prove that the call worked.  */
Packit Service 82fcde
      double before_one_ping = get_ticks ();
Packit Service 82fcde
      clnt_call (clnt, PROC_ADD,
Packit Service 82fcde
                 xdr_test_query, (void *) &query,
Packit Service 82fcde
                 xdr_test_response, (void *) &response,
Packit Service 82fcde
                 ((struct timeval) { 0, 0 }));
Packit Service 82fcde
      double after_one_ping = get_ticks ();
Packit Service 82fcde
      if (test_verbose)
Packit Service 82fcde
        printf ("info: non-blocking send took %f seconds\n",
Packit Service 82fcde
                after_one_ping - before_one_ping);
Packit Service 82fcde
      /* clnt_call should return immediately.  Accept some delay in
Packit Service 82fcde
         case the process is descheduled.  */
Packit Service 82fcde
      TEST_VERIFY (after_one_ping - before_one_ping < 0.3);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Collect the non-blocking response.  */
Packit Service 82fcde
  if (test_verbose)
Packit Service 82fcde
    printf ("info: collecting response\n");
Packit Service 82fcde
  struct test_response response = { 0 };
Packit Service 82fcde
  TEST_VERIFY
Packit Service 82fcde
    (clnt_call (clnt, PROC_ADD, NULL, NULL,
Packit Service 82fcde
                xdr_test_response, (void *) &response,
Packit Service 82fcde
                ((struct timeval) { 0, 0 })) == RPC_SUCCESS);
Packit Service 82fcde
  double after_pings = get_ticks ();
Packit Service 82fcde
  if (test_verbose)
Packit Service 82fcde
    printf ("info: send/receive took %f seconds\n",
Packit Service 82fcde
            after_pings - before_pings);
Packit Service 82fcde
  /* Expected timeout is 0.75 seconds.  */
Packit Service 82fcde
  TEST_VERIFY (0.75 <= after_pings - before_pings);
Packit Service 82fcde
  TEST_VERIFY (after_pings - before_pings < 1.2);
Packit Service 82fcde
Packit Service 82fcde
  uint32_t xid;
Packit Service 82fcde
  memcpy (&xid, &cu->cu_inbuf, sizeof (xid));
Packit Service 82fcde
  if (test_verbose)
Packit Service 82fcde
    printf ("info: non-blocking response: xid=%x server_id=%u seq=%u sum=%u\n",
Packit Service 82fcde
            xid, response.server_id, response.seq, response.sum);
Packit Service 82fcde
  /* Check that the reply from the preferred server was used.  */
Packit Service 82fcde
  TEST_VERIFY (servers[1].xid == xid);
Packit Service 82fcde
  TEST_VERIFY (response.server_id == 1);
Packit Service 82fcde
  TEST_VERIFY (response.seq == 1);
Packit Service 82fcde
  TEST_VERIFY (response.sum == 102);
Packit Service 82fcde
Packit Service 82fcde
  auth_destroy (clnt->cl_auth);
Packit Service 82fcde
  clnt_destroy (clnt);
Packit Service 82fcde
Packit Service 82fcde
  for (int i = 0; i < SERVER_COUNT; ++i)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (test_verbose)
Packit Service 82fcde
        printf ("info: requesting server %d termination\n", i);
Packit Service 82fcde
      client_socket = RPC_ANYSOCK;
Packit Service 82fcde
      clnt = clntudp_create (&servers[i].address, PROGNUM, VERSNUM,
Packit Service 82fcde
                             ((struct timeval) { 5, 0 }),
Packit Service 82fcde
                             &client_socket);
Packit Service 82fcde
      TEST_VERIFY_EXIT (clnt != NULL);
Packit Service 82fcde
      TEST_VERIFY (clnt_call (clnt, PROC_EXIT,
Packit Service 82fcde
                              (xdrproc_t) xdr_void, NULL,
Packit Service 82fcde
                              (xdrproc_t) xdr_void, NULL,
Packit Service 82fcde
                              ((struct timeval) { 3, 0 })) == RPC_SUCCESS);
Packit Service 82fcde
      clnt_destroy (clnt);
Packit Service 82fcde
Packit Service 82fcde
      int status;
Packit Service 82fcde
      xwaitpid (servers[i].pid, &status, 0);
Packit Service 82fcde
      TEST_VERIFY (WIFEXITED (status) && WEXITSTATUS (status) == EXIT_MARKER);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
#include <support/test-driver.c>