d595ba2b25
This option can be used for the performance optimization if sendfile() is not applicable.
342 lines
7.4 KiB
C
342 lines
7.4 KiB
C
/****************************************************************************
|
|
* apps/netutils/netcat/netcat_main.c
|
|
* netcat networking application
|
|
*
|
|
* 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 <nuttx/config.h>
|
|
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
|
|
#include <sys/sendfile.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <arpa/inet.h>
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
#ifndef NETCAT_PORT
|
|
# define NETCAT_PORT 31337
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
int do_io(int infd,
|
|
int outfd,
|
|
char *buf,
|
|
size_t buf_size)
|
|
{
|
|
ssize_t avail;
|
|
ssize_t written;
|
|
|
|
while (true)
|
|
{
|
|
avail = read(infd, buf, buf_size);
|
|
if (avail == 0)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (avail == -1)
|
|
{
|
|
perror("do_io: read error");
|
|
return 5;
|
|
}
|
|
|
|
written = write(outfd, buf, avail);
|
|
if (written == -1)
|
|
{
|
|
perror("do_io: write error");
|
|
return 6;
|
|
}
|
|
}
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
#ifdef CONFIG_NETUTILS_NETCAT_SENDFILE
|
|
int do_io_over_sendfile(int infd, int outfd, ssize_t len)
|
|
{
|
|
off_t offset = 0;
|
|
ssize_t written;
|
|
|
|
while (len > 0)
|
|
{
|
|
written = sendfile(outfd, infd, &offset, len);
|
|
|
|
if (written == -1 && errno == EAGAIN)
|
|
{
|
|
continue;
|
|
}
|
|
else if (written == -1)
|
|
{
|
|
perror("do_io: sendfile error");
|
|
return 5;
|
|
}
|
|
|
|
len -= written;
|
|
}
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
int netcat_server(int argc, char * argv[])
|
|
{
|
|
int id = -1;
|
|
int outfd = STDOUT_FILENO;
|
|
struct sockaddr_in server;
|
|
struct sockaddr_in client;
|
|
int port = NETCAT_PORT;
|
|
int result = EXIT_SUCCESS;
|
|
int conn;
|
|
socklen_t addrlen;
|
|
char *preallocated_iobuf = NULL;
|
|
|
|
if ((1 < argc) && (0 == strcmp("-l", argv[1])))
|
|
{
|
|
if (2 < argc)
|
|
{
|
|
port = atoi(argv[2]);
|
|
}
|
|
|
|
if (3 < argc)
|
|
{
|
|
outfd = open(argv[3], O_WRONLY | O_CREAT | O_TRUNC, 0777);
|
|
if (outfd == -1)
|
|
{
|
|
perror("error: io: Failed to create file");
|
|
outfd = STDOUT_FILENO;
|
|
result = 1;
|
|
goto out;
|
|
}
|
|
}
|
|
}
|
|
|
|
preallocated_iobuf = (char *)malloc(CONFIG_NETUTILS_NETCAT_BUFSIZE);
|
|
if (preallocated_iobuf == NULL)
|
|
{
|
|
perror("error: malloc: Failed to allocate I/O buffer\n");
|
|
result = 2;
|
|
goto out;
|
|
}
|
|
|
|
id = socket(AF_INET , SOCK_STREAM , 0);
|
|
if (0 > id)
|
|
{
|
|
perror("error: net: Failed to create socket");
|
|
result = 2;
|
|
goto out;
|
|
}
|
|
|
|
server.sin_family = AF_INET;
|
|
server.sin_addr.s_addr = INADDR_ANY;
|
|
server.sin_port = htons(port);
|
|
if (0 > bind(id, (struct sockaddr *)&server , sizeof(server)))
|
|
{
|
|
perror("error: net: Failed to bind");
|
|
result = 3;
|
|
goto out;
|
|
}
|
|
|
|
fprintf(stderr, "log: net: listening on :%d\n", port);
|
|
if (listen(id , 3) == -1)
|
|
{
|
|
perror("error: net: Failed to listen");
|
|
result = 7;
|
|
goto out;
|
|
}
|
|
|
|
if ((conn = accept(id, (struct sockaddr *)&client, &addrlen)) != -1)
|
|
{
|
|
result = do_io(conn, outfd,
|
|
preallocated_iobuf, CONFIG_NETUTILS_NETCAT_BUFSIZE);
|
|
}
|
|
|
|
if (0 > conn)
|
|
{
|
|
perror("accept failed");
|
|
result = 4;
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
if (id != -1)
|
|
{
|
|
close(id);
|
|
}
|
|
|
|
if (preallocated_iobuf != NULL)
|
|
{
|
|
free(preallocated_iobuf);
|
|
}
|
|
|
|
if (outfd != STDOUT_FILENO)
|
|
{
|
|
close(outfd);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
int netcat_client(int argc, char * argv[])
|
|
{
|
|
int id = -1;
|
|
int infd = STDIN_FILENO;
|
|
char *host = "127.0.0.1";
|
|
int port = NETCAT_PORT;
|
|
int result = EXIT_SUCCESS;
|
|
struct sockaddr_in server;
|
|
char *preallocated_iobuf = NULL;
|
|
#ifdef CONFIG_NETUTILS_NETCAT_SENDFILE
|
|
struct stat stat_buf;
|
|
#endif
|
|
|
|
if (argc > 1)
|
|
{
|
|
host = argv[1];
|
|
}
|
|
|
|
if (argc > 2)
|
|
{
|
|
port = atoi(argv[2]);
|
|
}
|
|
|
|
if (argc > 3)
|
|
{
|
|
infd = open(argv[3], O_RDONLY);
|
|
if (infd == -1)
|
|
{
|
|
perror("error: io: Failed to open file");
|
|
infd = STDIN_FILENO;
|
|
result = 1;
|
|
goto out;
|
|
}
|
|
|
|
#ifdef CONFIG_NETUTILS_NETCAT_SENDFILE
|
|
if (fstat(infd, &stat_buf) == -1)
|
|
{
|
|
perror("error: fstat: Could not get the input file size");
|
|
infd = STDIN_FILENO;
|
|
result = 1;
|
|
goto out;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
id = socket(AF_INET , SOCK_STREAM , 0);
|
|
if (0 > id)
|
|
{
|
|
perror("error: net: Failed to create socket");
|
|
result = 2;
|
|
goto out;
|
|
}
|
|
|
|
server.sin_family = AF_INET;
|
|
server.sin_port = htons(port);
|
|
if (1 != inet_pton(AF_INET, host, &server.sin_addr))
|
|
{
|
|
perror("error: net: Invalid host");
|
|
result = 3;
|
|
goto out;
|
|
}
|
|
|
|
if (connect(id, (struct sockaddr *) &server, sizeof(server)) < 0)
|
|
{
|
|
perror("error: net: Failed to connect");
|
|
result = 4;
|
|
goto out;
|
|
}
|
|
|
|
#ifdef CONFIG_NETUTILS_NETCAT_SENDFILE
|
|
if (argc > 3)
|
|
{
|
|
result = do_io_over_sendfile(infd, id, stat_buf.st_size);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
preallocated_iobuf = (char *)malloc(CONFIG_NETUTILS_NETCAT_BUFSIZE);
|
|
|
|
if (preallocated_iobuf == NULL)
|
|
{
|
|
perror("error: malloc: Failed to allocate I/O buffer\n");
|
|
result = 2;
|
|
goto out;
|
|
}
|
|
|
|
result = do_io(infd, id,
|
|
preallocated_iobuf, CONFIG_NETUTILS_NETCAT_BUFSIZE);
|
|
}
|
|
|
|
out:
|
|
if (id != -1)
|
|
{
|
|
close(id);
|
|
}
|
|
|
|
if (preallocated_iobuf != NULL)
|
|
{
|
|
free(preallocated_iobuf);
|
|
}
|
|
|
|
if (infd != STDIN_FILENO)
|
|
{
|
|
close(infd);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* netcat_main
|
|
****************************************************************************/
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
{
|
|
int status = EXIT_SUCCESS;
|
|
if (2 > argc)
|
|
{
|
|
fprintf(stderr,
|
|
"Usage: netcat <destination> [port] [file]\n"
|
|
"Usage: netcat -l [port] [file]\n");
|
|
}
|
|
else if ((1 < argc) && (0 == strcmp("-l", argv[1])))
|
|
{
|
|
status = netcat_server(argc, argv);
|
|
}
|
|
else
|
|
{
|
|
status = netcat_client(argc, argv);
|
|
}
|
|
|
|
return status;
|
|
}
|