mirror of
https://repo.or.cz/socat.git
synced 2025-01-10 22:52:33 +00:00
228 lines
6.5 KiB
C
228 lines
6.5 KiB
C
/* source: xio-test.c */
|
|
/* Copyright Gerhard Rieger 2007-2009 */
|
|
/* Published under the GNU General Public License V.2, see file COPYING */
|
|
|
|
/* this file contains the source for an intermediate test address that appends
|
|
'>' to every block transferred from right to left, and '<' in the other
|
|
direction */
|
|
|
|
#include "xiosysincludes.h"
|
|
#include "xioopen.h"
|
|
|
|
#include "xio-test.h"
|
|
|
|
|
|
#if WITH_TEST
|
|
|
|
static int xioopen_test(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy1, int dummy2,
|
|
int dummy3);
|
|
static int xioopen_testuni(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy1, int dummy2,
|
|
int dummy3);
|
|
static int xioopen_testrev(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy1, int dummy2,
|
|
int dummy3);
|
|
|
|
static const struct xioaddr_inter_desc xiointer_test0ro = { XIOADDR_PROT, "test", 0, XIOBIT_RDONLY, 0/*groups*/, XIOSHUT_UNSPEC, XIOCLOSE_UNSPEC, xioopen_test, 0, 0, 0, XIOBIT_WRONLY HELP("") };
|
|
static const struct xioaddr_inter_desc xiointer_test0wo = { XIOADDR_PROT, "test", 0, XIOBIT_WRONLY, 0/*groups*/, XIOSHUT_UNSPEC, XIOCLOSE_UNSPEC, xioopen_test, 0, 0, 0, XIOBIT_RDONLY HELP("") };
|
|
static const struct xioaddr_inter_desc xiointer_test0rw = { XIOADDR_PROT, "test", 0, XIOBIT_RDWR, 0/*groups*/, XIOSHUT_UNSPEC, XIOCLOSE_UNSPEC, xioopen_test, 0, 0, 0, XIOBIT_RDWR HELP("") };
|
|
|
|
const union xioaddr_desc *xioaddrs_test[] = {
|
|
(union xioaddr_desc *)&xiointer_test0ro,
|
|
(union xioaddr_desc *)&xiointer_test0wo,
|
|
(union xioaddr_desc *)&xiointer_test0rw,
|
|
NULL };
|
|
|
|
|
|
static const struct xioaddr_inter_desc xiointer_testuni = { XIOADDR_PROT, "testuni", 0, XIOBIT_WRONLY, 0/*groups*/, XIOSHUT_CLOSE, XIOCLOSE_NONE, xioopen_testuni, 0, 0, 0, XIOBIT_RDONLY HELP("") };
|
|
|
|
const union xioaddr_desc *xioaddrs_testuni[] = {
|
|
(union xioaddr_desc *)&xiointer_testuni,
|
|
NULL };
|
|
|
|
|
|
static const struct xioaddr_inter_desc xiointer_testrev = { XIOADDR_PROT, "testrev", 0, XIOBIT_WRONLY, 0/*groups*/, XIOSHUT_CLOSE, XIOCLOSE_NONE, xioopen_testrev, 0, 0, 0, XIOBIT_RDONLY HELP("") };
|
|
|
|
const union xioaddr_desc *xioaddrs_testrev[] = {
|
|
(union xioaddr_desc *)&xiointer_testrev,
|
|
NULL };
|
|
|
|
static int xioopen_test(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy, int dummy2,
|
|
int dummy3) {
|
|
struct single *xfd = &xxfd->stream;
|
|
int result;
|
|
|
|
assert(argc == 1);
|
|
assert(!(xfd->rfd < 0 && xfd->wfd < 0)); /*!!!*/
|
|
|
|
applyopts(-1, opts, PH_INIT);
|
|
if (applyopts_single(xfd, opts, PH_INIT) < 0) return -1;
|
|
|
|
Notice("opening TEST");
|
|
xfd->dtype = XIODATA_TEST;
|
|
applyopts(xfd->rfd, opts, PH_ALL);
|
|
if ((result = _xio_openlate(xfd, opts)) < 0)
|
|
return result;
|
|
return 0;
|
|
}
|
|
|
|
static int xioopen_testuni(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy, int dummy2,
|
|
int dummy3) {
|
|
struct single *xfd = &xxfd->stream;
|
|
int result;
|
|
|
|
assert(argc == 1);
|
|
assert(!(xfd->rfd < 0 && xfd->wfd < 0)); /*!!!*/
|
|
|
|
applyopts(-1, opts, PH_INIT);
|
|
if (applyopts_single(xfd, opts, PH_INIT) < 0) return -1;
|
|
|
|
Notice("opening TESTUNI");
|
|
xfd->dtype = XIODATA_TESTUNI;
|
|
applyopts(xfd->rfd, opts, PH_ALL);
|
|
if ((result = _xio_openlate(xfd, opts)) < 0)
|
|
return result;
|
|
return 0;
|
|
}
|
|
|
|
static int xioopen_testrev(int argc, const char *argv[], struct opt *opts,
|
|
int xioflags, xiofile_t *xxfd,
|
|
unsigned groups, int dummy, int dummy2,
|
|
int dummy3) {
|
|
struct single *xfd = &xxfd->stream;
|
|
int result;
|
|
|
|
assert(argc == 1);
|
|
assert(!(xfd->rfd < 0 && xfd->wfd < 0)); /*!!!*/
|
|
|
|
applyopts(-1, opts, PH_INIT);
|
|
if (applyopts_single(xfd, opts, PH_INIT) < 0) return -1;
|
|
|
|
Notice("opening TESTREV");
|
|
xfd->dtype = XIODATA_TESTREV;
|
|
applyopts(xfd->rfd, opts, PH_ALL);
|
|
if ((result = _xio_openlate(xfd, opts)) < 0)
|
|
return result;
|
|
return 0;
|
|
}
|
|
|
|
size_t xioread_test(struct single *sfd, void *buff, size_t bufsiz) {
|
|
int fd = sfd->rfd;
|
|
ssize_t bytes;
|
|
int _errno;
|
|
|
|
do {
|
|
bytes = Read(fd, buff, bufsiz-1);
|
|
} while (bytes < 0 && errno == EINTR);
|
|
if (bytes < 0) {
|
|
_errno = errno;
|
|
switch (_errno) {
|
|
case EPIPE: case ECONNRESET:
|
|
Warn4("read(%d, %p, "F_Zu"): %s",
|
|
fd, buff, bufsiz-1, strerror(_errno));
|
|
break;
|
|
default:
|
|
Error4("read(%d, %p, "F_Zu"): %s",
|
|
fd, buff, bufsiz-1, strerror(_errno));
|
|
}
|
|
return -1;
|
|
}
|
|
if (bytes == 0) {
|
|
return 0;
|
|
}
|
|
((char *)buff)[bytes] = '<';
|
|
return bytes+1;
|
|
}
|
|
|
|
size_t xiowrite_test(struct single *sfd, const void *buff, size_t bytes) {
|
|
int fd = sfd->wfd;
|
|
void *buff1;
|
|
ssize_t writt;
|
|
int _errno;
|
|
|
|
if ((buff1 = Malloc(bytes+1)) == NULL) {
|
|
return -1;
|
|
}
|
|
memcpy(buff1, buff, bytes);
|
|
((char *)buff1)[bytes] = '>';
|
|
do {
|
|
writt = Write(fd, buff1, bytes+1);
|
|
} while (writt < 0 && errno == EINTR);
|
|
if (writt < 0) {
|
|
_errno = errno;
|
|
switch (_errno) {
|
|
case EPIPE:
|
|
case ECONNRESET:
|
|
if (sfd->cool_write) {
|
|
Notice4("write(%d, %p, "F_Zu"): %s",
|
|
fd, buff1, bytes+1, strerror(_errno));
|
|
break;
|
|
}
|
|
/*PASSTHROUGH*/
|
|
default:
|
|
Error4("write(%d, %p, "F_Zu"): %s",
|
|
fd, buff1, bytes+1, strerror(_errno));
|
|
}
|
|
errno = _errno;
|
|
free(buff1);
|
|
return -1;
|
|
}
|
|
if ((size_t)writt < bytes) {
|
|
Warn2("write() only wrote "F_Zu" of "F_Zu" bytes",
|
|
writt, bytes+1);
|
|
}
|
|
free(buff1);
|
|
return writt;
|
|
}
|
|
|
|
size_t xiowrite_testrev(struct single *sfd, const void *buff, size_t bytes) {
|
|
int fd = sfd->wfd;
|
|
void *buff1;
|
|
ssize_t writt;
|
|
int _errno;
|
|
|
|
if ((buff1 = Malloc(bytes+1)) == NULL) {
|
|
return -1;
|
|
}
|
|
memcpy(buff1, buff, bytes);
|
|
((char *)buff1)[bytes] = '<';
|
|
do {
|
|
writt = Write(fd, buff1, bytes+1);
|
|
} while (writt < 0 && errno == EINTR);
|
|
if (writt < 0) {
|
|
_errno = errno;
|
|
switch (_errno) {
|
|
case EPIPE:
|
|
case ECONNRESET:
|
|
if (sfd->cool_write) {
|
|
Notice4("write(%d, %p, "F_Zu"): %s",
|
|
fd, buff1, bytes+1, strerror(_errno));
|
|
break;
|
|
}
|
|
/*PASSTHROUGH*/
|
|
default:
|
|
Error4("write(%d, %p, "F_Zu"): %s",
|
|
fd, buff1, bytes+1, strerror(_errno));
|
|
}
|
|
errno = _errno;
|
|
free(buff1);
|
|
return -1;
|
|
}
|
|
if ((size_t)writt < bytes) {
|
|
Warn2("write() only wrote "F_Zu" of "F_Zu" bytes",
|
|
writt, bytes+1);
|
|
}
|
|
free(buff1);
|
|
return writt;
|
|
}
|
|
|
|
#endif /* WITH_TEST */
|
|
|