File: sctp.c

package info (click to toggle)
valgrind 1%3A3.24.0-3
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 176,332 kB
  • sloc: ansic: 795,029; exp: 26,134; xml: 23,472; asm: 14,393; cpp: 9,397; makefile: 7,464; sh: 6,122; perl: 5,446; python: 1,498; javascript: 981; awk: 166; csh: 1
file content (122 lines) | stat: -rw-r--r-- 2,646 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/* This is based on /src/tools/test/stress2/misc/sctp.sh
 * but it uses fork and the server exits rather than
 * loops indefinitely */

#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <libgen.h>
#include <netinet/in.h>
#include <netinet/sctp.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

static int my_port_num = 62324;

static void
die(const char *s)
{
   perror(s);
   exit(1);
}

static void
server(void)
{
   struct sctp_sndrcvinfo sndrcvinfo;
   struct sockaddr_in servaddr = {
      .sin_family = AF_INET,
      .sin_addr.s_addr = htonl(INADDR_ANY),
      .sin_port = htons(my_port_num),
   };
   struct sctp_initmsg initmsg = {
      .sinit_num_ostreams = 5,
      .sinit_max_instreams = 5,
      .sinit_max_attempts = 4,
   };
   int listen_fd, conn_fd, flags, ret, in;

   listen_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_SCTP);
   if (listen_fd < 0)
      die("socket");

   ret = bind(listen_fd, (struct sockaddr *) &servaddr, sizeof(servaddr));
   if (ret < 0)
      die("bind");

   ret = setsockopt(listen_fd, IPPROTO_SCTP, SCTP_INITMSG, &initmsg,
         sizeof(initmsg));
   if (ret < 0)
      die("setsockopt");

   ret = listen(listen_fd, initmsg.sinit_max_instreams);
   if (ret < 0)
      die("listen");

   char buffer[1024];

   printf("Waiting for connection\n");
   fflush(stdout);

   conn_fd = accept(listen_fd, (struct sockaddr *) NULL, NULL);
   if(conn_fd < 0)
      die("accept()");

   printf("New client connected\n");
   fflush(stdout);

   /* Note that flags is uninitialized here */
   in = sctp_recvmsg(conn_fd, buffer, sizeof(buffer), NULL, 0,
         &sndrcvinfo, &flags);
   if (in > 0) {
      printf("Received data: %s\n", buffer);
      fflush(stdout);
   }

   close(conn_fd);
}

static void
client(void)
{
   struct sockaddr_in servaddr = {
      .sin_family = AF_INET,
      .sin_port = htons(my_port_num),
      .sin_addr.s_addr = inet_addr("127.0.0.1"),
   };
   int conn_fd, ret;
   const char *msg = "Hello, Server!";

   conn_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_SCTP);
   if (conn_fd < 0)
      die("socket()");

   ret = connect(conn_fd, (struct sockaddr *) &servaddr, sizeof(servaddr));
   if (ret < 0)
      die("connect()");

   ret = sctp_sendmsg(conn_fd, (void *) msg, strlen(msg) + 1, NULL, 0, 0, 0, 0, 0, 0 );
   if (ret < 0)
      die("sctp_sendmsg");

   close(conn_fd);
}

int
main(int argc __unused, char *argv[])
{
   switch (fork())
   {
   case 0:
      sleep(1);
      client();
      exit(EXIT_SUCCESS);
   default:
      server();
      exit(EXIT_SUCCESS);
   case -1:
      exit(EXIT_FAILURE);
   }
}