/**************************************************************************** * apps/examples/netloop/lo_main.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "netloop.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define IOBUFFER_SIZE 80 /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: lo_client ****************************************************************************/ static int lo_client(void) { struct sockaddr_in myaddr; char outbuf[IOBUFFER_SIZE]; char inbuf[IOBUFFER_SIZE]; int sockfd; int len; int nbytessent; int nbytesrecvd; int ret; int i; #ifdef CONFIG_EXAMPLES_NETLOOP_KEEPALIVE struct timeval tv; int value; #endif /* Create a new TCP socket */ sockfd = socket(PF_INET, SOCK_STREAM, 0); if (sockfd < 0) { ret = -errno; printf("lo_client: socket failure %d\n", ret); return ret; } #ifdef CONFIG_EXAMPLES_NETLOOP_KEEPALIVE /* Enable TCP KeepAlive */ value = TRUE; ret = setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, &value, sizeof(int)); if (ret < 0) { ret = -errno; fprintf(stderr, "setsockopt(SO_KEEPALIVE) failed: %d\n", ret); return ret; } tv.tv_sec = 5; tv.tv_usec = 0; ret = setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPIDLE, &tv, sizeof(struct timeval)); if (ret < 0) { ret = -errno; fprintf(stderr, "setsockopt(TCP_KEEPIDLE) failed: %d\n", ret); return ret; } tv.tv_sec = 1; tv.tv_usec = 0; ret = setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPINTVL, &tv, sizeof(struct timeval)); if (ret < 0) { ret = -errno; fprintf(stderr, "setsockopt(TCP_KEEPIDLE) failed: %d\n", ret); return ret; } value = 3; ret = setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPCNT, &value, sizeof(int)); if (ret < 0) { ret = -errno; fprintf(stderr, "setsockopt(SO_KEEPALIVE) failed: %d\n", ret); return ret; } #endif /* Connect the socket to the server */ myaddr.sin_family = AF_INET; myaddr.sin_port = htons(LISTENER_PORT); myaddr.sin_addr.s_addr = htonl(LO_ADDRESS); printf("lo_client: Connecting to %08x:%d...\n", LO_ADDRESS, LISTENER_PORT); if (connect(sockfd, (struct sockaddr *)&myaddr, sizeof(struct sockaddr_in)) < 0) { ret = -errno; printf("lo_client: connect failure: %d\n", ret); goto errout_with_socket; } printf("lo_client: Connected\n"); /* Then send and receive messages */ for (i = 0; ; i++) { snprintf(outbuf, sizeof(outbuf), "Loopback message %d", i); len = strlen(outbuf); printf("lo_client: Sending '%s' (%d bytes)\n", outbuf, len); nbytessent = send(sockfd, outbuf, len, 0); printf("lo_client: Sent %d bytes\n", nbytessent); if (nbytessent < 0) { ret = -errno; printf("lo_client: send failed: %d\n", ret); goto errout_with_socket; } else if (nbytessent != len) { ret = -EINVAL; printf("lo_client: Bad send length: %d Expected: %d\n", nbytessent, len); goto errout_with_socket; } printf("lo_client: Receiving...\n"); nbytesrecvd = recv(sockfd, inbuf, IOBUFFER_SIZE, 0); if (nbytesrecvd < 0) { ret = -errno; printf("lo_client: recv failed: %d\n", ret); goto errout_with_socket; } else if (nbytesrecvd == 0) { ret = -ENOTCONN; printf("lo_client: The server broke the connections\n"); goto errout_with_socket; } inbuf[nbytesrecvd] = '\0'; printf("lo_client: Received '%s' (%d bytes)\n", inbuf, nbytesrecvd); if (nbytesrecvd != len) { ret = -EINVAL; printf("lo_client: Bad recv length: %d Expected: %d\n", nbytesrecvd, len); goto errout_with_socket; } else if (memcmp(inbuf, outbuf, len) != 0) { ret = -EINVAL; printf("lo_client: Received outbuf does not match sent outbuf\n"); goto errout_with_socket; } #ifdef CONFIG_EXAMPLES_NETLOOP_KEEPALIVE /* Send four messages, then wait a longer amount of time ... * longer than: * TCP_KEEPIDLE + TCP_KEEPINTVL * TCP_KEEPCNT = 5 + 3 * 1 = 8 seconds. */ if (i != 0 && (i & 3) == 0) { printf("lo_client: Long sleep\n"); sleep(12); } else #endif { printf("lo_client: Sleeping\n"); sleep(6); } } close(sockfd); return 0; errout_with_socket: close(sockfd); return ret; } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: netloop_main ****************************************************************************/ int main(int argc, FAR char *argv[]) { pthread_t tid; int ret; /* Start the listeners */ printf("netloop_main: Starting lo_listener thread\n"); ret = pthread_create(&tid, NULL, lo_listener, NULL); if (ret != 0) { printf("netloop_main: Failed to create lo_listener thread: %d\n", ret); } /* Run the client */ ret = lo_client(); return 0; }