mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-05-20 19:50:24 +02:00
Have recv!',
send', etc. accept a bytevector.
* libguile/socket.c (scm_recv, scm_send, scm_recvfrom, scm_sendto): Expect the buffer to be a bytevector. Move the string-handling code under `#if SCM_ENABLE_DEPRECATED == 1' and issue a deprecation warning. * test-suite/tests/socket.test ("AF_UNIX/SOCK_DGRAM")["sendto", "sendto/sockaddr"]: Adjust accordingly. * doc/ref/posix.texi (Network Sockets and Communication): Update documentation of `recv!', `send', `recvfrom!', and `sendto'.
This commit is contained in:
parent
0bc2452b55
commit
d21a1dc841
3 changed files with 184 additions and 89 deletions
|
@ -3188,7 +3188,7 @@ Note that on many systems the address of a socket in the
|
||||||
Receive data from a socket port.
|
Receive data from a socket port.
|
||||||
@var{sock} must already
|
@var{sock} must already
|
||||||
be bound to the address from which data is to be received.
|
be bound to the address from which data is to be received.
|
||||||
@var{buf} is a string into which
|
@var{buf} is a bytevector into which
|
||||||
the data will be written. The size of @var{buf} limits
|
the data will be written. The size of @var{buf} limits
|
||||||
the amount of
|
the amount of
|
||||||
data which can be received: in the case of packet
|
data which can be received: in the case of packet
|
||||||
|
@ -3215,7 +3215,7 @@ any unread buffered port data is ignored.
|
||||||
@vindex MSG_OOB
|
@vindex MSG_OOB
|
||||||
@vindex MSG_PEEK
|
@vindex MSG_PEEK
|
||||||
@vindex MSG_DONTROUTE
|
@vindex MSG_DONTROUTE
|
||||||
Transmit the string @var{message} on a socket port @var{sock}.
|
Transmit bytevector @var{message} on socket port @var{sock}.
|
||||||
@var{sock} must already be bound to a destination address. The value
|
@var{sock} must already be bound to a destination address. The value
|
||||||
returned is the number of bytes transmitted---it's possible for this
|
returned is the number of bytes transmitted---it's possible for this
|
||||||
to be less than the length of @var{message} if the socket is set to be
|
to be less than the length of @var{message} if the socket is set to be
|
||||||
|
@ -3227,17 +3227,18 @@ file descriptor:
|
||||||
any unflushed buffered port data is ignored.
|
any unflushed buffered port data is ignored.
|
||||||
@end deffn
|
@end deffn
|
||||||
|
|
||||||
@deffn {Scheme Procedure} recvfrom! sock str [flags [start [end]]]
|
@deffn {Scheme Procedure} recvfrom! sock buf [flags [start [end]]]
|
||||||
@deffnx {C Function} scm_recvfrom (sock, str, flags, start, end)
|
@deffnx {C Function} scm_recvfrom (sock, buf, flags, start, end)
|
||||||
Receive data from socket port @var{sock}, returning the originating
|
Receive data from socket port @var{sock}, returning the originating
|
||||||
address as well as the data. This function is usually for datagram
|
address as well as the data. This function is usually for datagram
|
||||||
sockets, but can be used on stream-oriented sockets too.
|
sockets, but can be used on stream-oriented sockets too.
|
||||||
|
|
||||||
The data received is stored in the given @var{str}, the whole string
|
The data received is stored in bytevector @var{buf}, using
|
||||||
or just the region between the optional @var{start} and @var{end}
|
either the whole bytevector or just the region between the optional
|
||||||
positions. The size of @var{str} limits the amount of data which can
|
@var{start} and @var{end} positions. The size of @var{buf}
|
||||||
be received. For datagram protocols if a packet larger than this is
|
limits the amount of data that can be received. For datagram
|
||||||
received then excess bytes are irrevocably lost.
|
protocols if a packet larger than this is received then excess
|
||||||
|
bytes are irrevocably lost.
|
||||||
|
|
||||||
The return value is a pair. The @code{car} is the number of bytes
|
The return value is a pair. The @code{car} is the number of bytes
|
||||||
read. The @code{cdr} is a socket address object (@pxref{Network
|
read. The @code{cdr} is a socket address object (@pxref{Network
|
||||||
|
@ -3267,7 +3268,7 @@ application may need to use @code{select}, @code{O_NONBLOCK} or
|
||||||
@deffnx {Scheme Procedure} sendto sock message AF_INET6 ipv6addr port [flowinfo [scopeid [flags]]]
|
@deffnx {Scheme Procedure} sendto sock message AF_INET6 ipv6addr port [flowinfo [scopeid [flags]]]
|
||||||
@deffnx {Scheme Procedure} sendto sock message AF_UNIX path [flags]
|
@deffnx {Scheme Procedure} sendto sock message AF_UNIX path [flags]
|
||||||
@deffnx {C Function} scm_sendto (sock, message, fam, address, args_and_flags)
|
@deffnx {C Function} scm_sendto (sock, message, fam, address, args_and_flags)
|
||||||
Transmit the string @var{message} as a datagram on socket port
|
Transmit bytevector @var{message} as a datagram socket port
|
||||||
@var{sock}. The destination is specified either as a socket address
|
@var{sock}. The destination is specified either as a socket address
|
||||||
object, or as arguments the same as would be taken by
|
object, or as arguments the same as would be taken by
|
||||||
@code{make-socket-address} to create such an object (@pxref{Network
|
@code{make-socket-address} to create such an object (@pxref{Network
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/* Copyright (C) 1996,1997,1998,2000,2001, 2002, 2003, 2004, 2005, 2006, 2007, 2009 Free Software Foundation, Inc.
|
/* Copyright (C) 1996, 1997, 1998, 2000, 2001, 2002, 2003, 2004, 2005,
|
||||||
*
|
* 2006, 2007, 2009, 2011 Free Software Foundation, Inc.
|
||||||
|
*
|
||||||
* This library is free software; you can redistribute it and/or
|
* This library is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU Lesser General Public License
|
* modify it under the terms of the GNU Lesser General Public License
|
||||||
* as published by the Free Software Foundation; either version 3 of
|
* as published by the Free Software Foundation; either version 3 of
|
||||||
|
@ -38,6 +39,10 @@
|
||||||
#include "libguile/validate.h"
|
#include "libguile/validate.h"
|
||||||
#include "libguile/socket.h"
|
#include "libguile/socket.h"
|
||||||
|
|
||||||
|
#if SCM_ENABLE_DEPRECATED == 1
|
||||||
|
# include "libguile/deprecation.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef __MINGW32__
|
#ifdef __MINGW32__
|
||||||
#include "win32-socket.h"
|
#include "win32-socket.h"
|
||||||
#endif
|
#endif
|
||||||
|
@ -1352,15 +1357,13 @@ SCM_DEFINE (scm_recv, "recv!", 2, 1, 0,
|
||||||
"Receive data from a socket port.\n"
|
"Receive data from a socket port.\n"
|
||||||
"@var{sock} must already\n"
|
"@var{sock} must already\n"
|
||||||
"be bound to the address from which data is to be received.\n"
|
"be bound to the address from which data is to be received.\n"
|
||||||
"@var{buf} is a string into which\n"
|
"@var{buf} is a bytevector into which\n"
|
||||||
"the data will be written. The size of @var{buf} limits\n"
|
"the data will be written. The size of @var{buf} limits\n"
|
||||||
"the amount of\n"
|
"the amount of\n"
|
||||||
"data which can be received: in the case of packet\n"
|
"data which can be received: in the case of packet\n"
|
||||||
"protocols, if a packet larger than this limit is encountered\n"
|
"protocols, if a packet larger than this limit is encountered\n"
|
||||||
"then some data\n"
|
"then some data\n"
|
||||||
"will be irrevocably lost.\n\n"
|
"will be irrevocably lost.\n\n"
|
||||||
"The data is assumed to be binary, and there is no decoding of\n"
|
|
||||||
"of locale-encoded strings.\n\n"
|
|
||||||
"The optional @var{flags} argument is a value or\n"
|
"The optional @var{flags} argument is a value or\n"
|
||||||
"bitwise OR of MSG_OOB, MSG_PEEK, MSG_DONTROUTE etc.\n\n"
|
"bitwise OR of MSG_OOB, MSG_PEEK, MSG_DONTROUTE etc.\n\n"
|
||||||
"The value returned is the number of bytes read from the\n"
|
"The value returned is the number of bytes read from the\n"
|
||||||
|
@ -1370,38 +1373,55 @@ SCM_DEFINE (scm_recv, "recv!", 2, 1, 0,
|
||||||
"any unread buffered port data is ignored.")
|
"any unread buffered port data is ignored.")
|
||||||
#define FUNC_NAME s_scm_recv
|
#define FUNC_NAME s_scm_recv
|
||||||
{
|
{
|
||||||
int rv;
|
int rv, fd, flg;
|
||||||
int fd;
|
|
||||||
int flg;
|
|
||||||
char *dest;
|
|
||||||
size_t len;
|
|
||||||
SCM msg;
|
|
||||||
|
|
||||||
SCM_VALIDATE_OPFPORT (1, sock);
|
SCM_VALIDATE_OPFPORT (1, sock);
|
||||||
SCM_VALIDATE_STRING (2, buf);
|
|
||||||
if (SCM_UNBNDP (flags))
|
if (SCM_UNBNDP (flags))
|
||||||
flg = 0;
|
flg = 0;
|
||||||
else
|
else
|
||||||
flg = scm_to_int (flags);
|
flg = scm_to_int (flags);
|
||||||
fd = SCM_FPORT_FDES (sock);
|
fd = SCM_FPORT_FDES (sock);
|
||||||
|
|
||||||
len = scm_i_string_length (buf);
|
#if SCM_ENABLE_DEPRECATED == 1
|
||||||
msg = scm_i_make_string (len, &dest);
|
if (SCM_UNLIKELY (scm_is_string (buf)))
|
||||||
SCM_SYSCALL (rv = recv (fd, dest, len, flg));
|
{
|
||||||
scm_string_copy_x (buf, scm_from_int (0),
|
SCM msg;
|
||||||
msg, scm_from_int (0), scm_from_size_t (len));
|
char *dest;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
if (rv == -1)
|
scm_c_issue_deprecation_warning
|
||||||
|
("Passing a string to `recv!' is deprecated, "
|
||||||
|
"use a bytevector instead.");
|
||||||
|
|
||||||
|
len = scm_i_string_length (buf);
|
||||||
|
msg = scm_i_make_string (len, &dest);
|
||||||
|
SCM_SYSCALL (rv = recv (fd, dest, len, flg));
|
||||||
|
scm_string_copy_x (buf, scm_from_int (0),
|
||||||
|
msg, scm_from_int (0), scm_from_size_t (len));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
SCM_VALIDATE_BYTEVECTOR (1, buf);
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = recv (fd,
|
||||||
|
SCM_BYTEVECTOR_CONTENTS (buf),
|
||||||
|
SCM_BYTEVECTOR_LENGTH (buf),
|
||||||
|
flg));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (SCM_UNLIKELY (rv == -1))
|
||||||
SCM_SYSERROR;
|
SCM_SYSERROR;
|
||||||
|
|
||||||
scm_remember_upto_here_2 (buf, msg);
|
scm_remember_upto_here (buf);
|
||||||
return scm_from_int (rv);
|
return scm_from_int (rv);
|
||||||
}
|
}
|
||||||
#undef FUNC_NAME
|
#undef FUNC_NAME
|
||||||
|
|
||||||
SCM_DEFINE (scm_send, "send", 2, 1, 0,
|
SCM_DEFINE (scm_send, "send", 2, 1, 0,
|
||||||
(SCM sock, SCM message, SCM flags),
|
(SCM sock, SCM message, SCM flags),
|
||||||
"Transmit the string @var{message} on a socket port @var{sock}.\n"
|
"Transmit bytevector @var{message} on socket port @var{sock}.\n"
|
||||||
"@var{sock} must already be bound to a destination address. The\n"
|
"@var{sock} must already be bound to a destination address. The\n"
|
||||||
"value returned is the number of bytes transmitted --\n"
|
"value returned is the number of bytes transmitted --\n"
|
||||||
"it's possible for\n"
|
"it's possible for\n"
|
||||||
|
@ -1417,34 +1437,47 @@ SCM_DEFINE (scm_send, "send", 2, 1, 0,
|
||||||
"zero to 255.")
|
"zero to 255.")
|
||||||
#define FUNC_NAME s_scm_send
|
#define FUNC_NAME s_scm_send
|
||||||
{
|
{
|
||||||
int rv;
|
int rv, fd, flg;
|
||||||
int fd;
|
|
||||||
int flg;
|
|
||||||
char *src;
|
|
||||||
size_t len;
|
|
||||||
|
|
||||||
sock = SCM_COERCE_OUTPORT (sock);
|
sock = SCM_COERCE_OUTPORT (sock);
|
||||||
SCM_VALIDATE_OPFPORT (1, sock);
|
SCM_VALIDATE_OPFPORT (1, sock);
|
||||||
SCM_VALIDATE_STRING (2, message);
|
|
||||||
|
|
||||||
/* If the string is wide, see if it can be coerced into
|
|
||||||
a narrow string. */
|
|
||||||
if (!scm_i_is_narrow_string (message)
|
|
||||||
|| scm_i_try_narrow_string (message))
|
|
||||||
SCM_MISC_ERROR ("the message string is not 8-bit: ~s",
|
|
||||||
scm_list_1 (message));
|
|
||||||
|
|
||||||
if (SCM_UNBNDP (flags))
|
if (SCM_UNBNDP (flags))
|
||||||
flg = 0;
|
flg = 0;
|
||||||
else
|
else
|
||||||
flg = scm_to_int (flags);
|
flg = scm_to_int (flags);
|
||||||
|
|
||||||
fd = SCM_FPORT_FDES (sock);
|
fd = SCM_FPORT_FDES (sock);
|
||||||
|
|
||||||
len = scm_i_string_length (message);
|
#if SCM_ENABLE_DEPRECATED == 1
|
||||||
message = scm_i_string_start_writing (message);
|
if (SCM_UNLIKELY (scm_is_string (message)))
|
||||||
src = scm_i_string_writable_chars (message);
|
{
|
||||||
SCM_SYSCALL (rv = send (fd, src, len, flg));
|
scm_c_issue_deprecation_warning
|
||||||
scm_i_string_stop_writing ();
|
("Passing a string to `send' is deprecated, "
|
||||||
|
"use a bytevector instead.");
|
||||||
|
|
||||||
|
/* If the string is wide, see if it can be coerced into a narrow
|
||||||
|
string. */
|
||||||
|
if (!scm_i_is_narrow_string (message)
|
||||||
|
|| !scm_i_try_narrow_string (message))
|
||||||
|
SCM_MISC_ERROR ("the message string is not 8-bit: ~s",
|
||||||
|
scm_list_1 (message));
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = send (fd,
|
||||||
|
scm_i_string_chars (message),
|
||||||
|
scm_i_string_length (message),
|
||||||
|
flg));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
SCM_VALIDATE_BYTEVECTOR (1, message);
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = send (fd,
|
||||||
|
SCM_BYTEVECTOR_CONTENTS (message),
|
||||||
|
SCM_BYTEVECTOR_LENGTH (message),
|
||||||
|
flg));
|
||||||
|
}
|
||||||
|
|
||||||
if (rv == -1)
|
if (rv == -1)
|
||||||
SCM_SYSERROR;
|
SCM_SYSERROR;
|
||||||
|
@ -1455,22 +1488,22 @@ SCM_DEFINE (scm_send, "send", 2, 1, 0,
|
||||||
#undef FUNC_NAME
|
#undef FUNC_NAME
|
||||||
|
|
||||||
SCM_DEFINE (scm_recvfrom, "recvfrom!", 2, 3, 0,
|
SCM_DEFINE (scm_recvfrom, "recvfrom!", 2, 3, 0,
|
||||||
(SCM sock, SCM str, SCM flags, SCM start, SCM end),
|
(SCM sock, SCM buf, SCM flags, SCM start, SCM end),
|
||||||
"Receive data from socket port @var{sock} (which must be already\n"
|
"Receive data from socket port @var{sock} (which must be already\n"
|
||||||
"bound), returning the originating address as well as the data.\n"
|
"bound), returning the originating address as well as the data.\n"
|
||||||
"This is usually for use on datagram sockets, but can be used on\n"
|
"This is usually for use on datagram sockets, but can be used on\n"
|
||||||
"stream-oriented sockets too.\n"
|
"stream-oriented sockets too.\n"
|
||||||
"\n"
|
"\n"
|
||||||
"The data received is stored in the given @var{str}, using\n"
|
"The data received is stored in bytevector @var{buf}, using\n"
|
||||||
"either the whole string or just the region between the optional\n"
|
"either the whole bytevector or just the region between the optional\n"
|
||||||
"@var{start} and @var{end} positions. The size of @var{str}\n"
|
"@var{start} and @var{end} positions. The size of @var{buf}\n"
|
||||||
"limits the amount of data which can be received. For datagram\n"
|
"limits the amount of data that can be received. For datagram\n"
|
||||||
"protocols, if a packet larger than this is received then excess\n"
|
"protocols, if a packet larger than this is received then excess\n"
|
||||||
"bytes are irrevocably lost.\n"
|
"bytes are irrevocably lost.\n"
|
||||||
"\n"
|
"\n"
|
||||||
"The return value is a pair. The @code{car} is the number of\n"
|
"The return value is a pair. The @code{car} is the number of\n"
|
||||||
"bytes read. The @code{cdr} is a socket address object which is\n"
|
"bytes read. The @code{cdr} is a socket address object which is\n"
|
||||||
"where the data come from, or @code{#f} if the origin is\n"
|
"where the data came from, or @code{#f} if the origin is\n"
|
||||||
"unknown.\n"
|
"unknown.\n"
|
||||||
"\n"
|
"\n"
|
||||||
"The optional @var{flags} argument is a or bitwise OR\n"
|
"The optional @var{flags} argument is a or bitwise OR\n"
|
||||||
|
@ -1486,46 +1519,79 @@ SCM_DEFINE (scm_recvfrom, "recvfrom!", 2, 3, 0,
|
||||||
"or @code{MSG_DONTWAIT} to avoid this.")
|
"or @code{MSG_DONTWAIT} to avoid this.")
|
||||||
#define FUNC_NAME s_scm_recvfrom
|
#define FUNC_NAME s_scm_recvfrom
|
||||||
{
|
{
|
||||||
int rv;
|
int rv, fd, flg;
|
||||||
int fd;
|
|
||||||
int flg;
|
|
||||||
char *buf;
|
|
||||||
size_t offset;
|
|
||||||
size_t cend;
|
|
||||||
SCM address;
|
SCM address;
|
||||||
|
size_t offset, cend;
|
||||||
socklen_t addr_size = MAX_ADDR_SIZE;
|
socklen_t addr_size = MAX_ADDR_SIZE;
|
||||||
scm_t_max_sockaddr addr;
|
scm_t_max_sockaddr addr;
|
||||||
|
|
||||||
SCM_VALIDATE_OPFPORT (1, sock);
|
SCM_VALIDATE_OPFPORT (1, sock);
|
||||||
fd = SCM_FPORT_FDES (sock);
|
fd = SCM_FPORT_FDES (sock);
|
||||||
|
|
||||||
SCM_VALIDATE_STRING (2, str);
|
|
||||||
scm_i_get_substring_spec (scm_i_string_length (str),
|
|
||||||
start, &offset, end, &cend);
|
|
||||||
|
|
||||||
if (SCM_UNBNDP (flags))
|
if (SCM_UNBNDP (flags))
|
||||||
flg = 0;
|
flg = 0;
|
||||||
else
|
else
|
||||||
SCM_VALIDATE_ULONG_COPY (3, flags, flg);
|
SCM_VALIDATE_ULONG_COPY (3, flags, flg);
|
||||||
|
|
||||||
/* recvfrom will not necessarily return an address. usually nothing
|
|
||||||
is returned for stream sockets. */
|
|
||||||
str = scm_i_string_start_writing (str);
|
|
||||||
buf = scm_i_string_writable_chars (str);
|
|
||||||
((struct sockaddr *) &addr)->sa_family = AF_UNSPEC;
|
((struct sockaddr *) &addr)->sa_family = AF_UNSPEC;
|
||||||
SCM_SYSCALL (rv = recvfrom (fd, buf + offset,
|
|
||||||
cend - offset, flg,
|
#if SCM_ENABLE_DEPRECATED == 1
|
||||||
(struct sockaddr *) &addr, &addr_size));
|
if (SCM_UNLIKELY (scm_is_string (buf)))
|
||||||
scm_i_string_stop_writing ();
|
{
|
||||||
|
char *cbuf;
|
||||||
|
|
||||||
|
scm_c_issue_deprecation_warning
|
||||||
|
("Passing a string to `recvfrom!' is deprecated, "
|
||||||
|
"use a bytevector instead.");
|
||||||
|
|
||||||
|
scm_i_get_substring_spec (scm_i_string_length (buf),
|
||||||
|
start, &offset, end, &cend);
|
||||||
|
|
||||||
|
buf = scm_i_string_start_writing (buf);
|
||||||
|
cbuf = scm_i_string_writable_chars (buf);
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = recvfrom (fd, cbuf + offset,
|
||||||
|
cend - offset, flg,
|
||||||
|
(struct sockaddr *) &addr, &addr_size));
|
||||||
|
scm_i_string_stop_writing ();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
SCM_VALIDATE_BYTEVECTOR (1, buf);
|
||||||
|
|
||||||
|
if (SCM_UNBNDP (start))
|
||||||
|
offset = 0;
|
||||||
|
else
|
||||||
|
offset = scm_to_size_t (start);
|
||||||
|
|
||||||
|
if (SCM_UNBNDP (end))
|
||||||
|
cend = SCM_BYTEVECTOR_LENGTH (buf);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cend = scm_to_size_t (end);
|
||||||
|
if (SCM_UNLIKELY (cend >= SCM_BYTEVECTOR_LENGTH (buf)
|
||||||
|
|| cend < offset))
|
||||||
|
scm_out_of_range (FUNC_NAME, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = recvfrom (fd,
|
||||||
|
SCM_BYTEVECTOR_CONTENTS (buf) + offset,
|
||||||
|
cend - offset, flg,
|
||||||
|
(struct sockaddr *) &addr, &addr_size));
|
||||||
|
}
|
||||||
|
|
||||||
if (rv == -1)
|
if (rv == -1)
|
||||||
SCM_SYSERROR;
|
SCM_SYSERROR;
|
||||||
|
|
||||||
|
/* `recvfrom' does not necessarily return an address. Usually nothing
|
||||||
|
is returned for stream sockets. */
|
||||||
if (((struct sockaddr *) &addr)->sa_family != AF_UNSPEC)
|
if (((struct sockaddr *) &addr)->sa_family != AF_UNSPEC)
|
||||||
address = _scm_from_sockaddr (&addr, addr_size, FUNC_NAME);
|
address = _scm_from_sockaddr (&addr, addr_size, FUNC_NAME);
|
||||||
else
|
else
|
||||||
address = SCM_BOOL_F;
|
address = SCM_BOOL_F;
|
||||||
|
|
||||||
scm_remember_upto_here_1 (str);
|
scm_remember_upto_here_1 (buf);
|
||||||
|
|
||||||
return scm_cons (scm_from_int (rv), address);
|
return scm_cons (scm_from_int (rv), address);
|
||||||
}
|
}
|
||||||
|
@ -1533,7 +1599,7 @@ SCM_DEFINE (scm_recvfrom, "recvfrom!", 2, 3, 0,
|
||||||
|
|
||||||
SCM_DEFINE (scm_sendto, "sendto", 3, 1, 1,
|
SCM_DEFINE (scm_sendto, "sendto", 3, 1, 1,
|
||||||
(SCM sock, SCM message, SCM fam_or_sockaddr, SCM address, SCM args_and_flags),
|
(SCM sock, SCM message, SCM fam_or_sockaddr, SCM address, SCM args_and_flags),
|
||||||
"Transmit the string @var{message} on the socket port\n"
|
"Transmit bytevector @var{message} on socket port\n"
|
||||||
"@var{sock}. The\n"
|
"@var{sock}. The\n"
|
||||||
"destination address is specified using the @var{fam},\n"
|
"destination address is specified using the @var{fam},\n"
|
||||||
"@var{address} and\n"
|
"@var{address} and\n"
|
||||||
|
@ -1555,15 +1621,12 @@ SCM_DEFINE (scm_sendto, "sendto", 3, 1, 1,
|
||||||
"zero to 255.")
|
"zero to 255.")
|
||||||
#define FUNC_NAME s_scm_sendto
|
#define FUNC_NAME s_scm_sendto
|
||||||
{
|
{
|
||||||
int rv;
|
int rv, fd, flg;
|
||||||
int fd;
|
|
||||||
int flg;
|
|
||||||
struct sockaddr *soka;
|
struct sockaddr *soka;
|
||||||
size_t size;
|
size_t size;
|
||||||
|
|
||||||
sock = SCM_COERCE_OUTPORT (sock);
|
sock = SCM_COERCE_OUTPORT (sock);
|
||||||
SCM_VALIDATE_FPORT (1, sock);
|
SCM_VALIDATE_FPORT (1, sock);
|
||||||
SCM_VALIDATE_STRING (2, message);
|
|
||||||
fd = SCM_FPORT_FDES (sock);
|
fd = SCM_FPORT_FDES (sock);
|
||||||
|
|
||||||
if (!scm_is_number (fam_or_sockaddr))
|
if (!scm_is_number (fam_or_sockaddr))
|
||||||
|
@ -1586,10 +1649,37 @@ SCM_DEFINE (scm_sendto, "sendto", 3, 1, 1,
|
||||||
SCM_VALIDATE_CONS (5, args_and_flags);
|
SCM_VALIDATE_CONS (5, args_and_flags);
|
||||||
flg = SCM_NUM2ULONG (5, SCM_CAR (args_and_flags));
|
flg = SCM_NUM2ULONG (5, SCM_CAR (args_and_flags));
|
||||||
}
|
}
|
||||||
SCM_SYSCALL (rv = sendto (fd,
|
|
||||||
scm_i_string_chars (message),
|
#if SCM_ENABLE_DEPRECATED == 1
|
||||||
scm_i_string_length (message),
|
if (SCM_UNLIKELY (scm_is_string (message)))
|
||||||
flg, soka, size));
|
{
|
||||||
|
scm_c_issue_deprecation_warning
|
||||||
|
("Passing a string to `sendto' is deprecated, "
|
||||||
|
"use a bytevector instead.");
|
||||||
|
|
||||||
|
/* If the string is wide, see if it can be coerced into a narrow
|
||||||
|
string. */
|
||||||
|
if (!scm_i_is_narrow_string (message)
|
||||||
|
|| !scm_i_try_narrow_string (message))
|
||||||
|
SCM_MISC_ERROR ("the message string is not 8-bit: ~s",
|
||||||
|
scm_list_1 (message));
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = sendto (fd,
|
||||||
|
scm_i_string_chars (message),
|
||||||
|
scm_i_string_length (message),
|
||||||
|
flg, soka, size));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
SCM_VALIDATE_BYTEVECTOR (1, message);
|
||||||
|
|
||||||
|
SCM_SYSCALL (rv = sendto (fd,
|
||||||
|
SCM_BYTEVECTOR_CONTENTS (message),
|
||||||
|
SCM_BYTEVECTOR_LENGTH (message),
|
||||||
|
flg, soka, size));
|
||||||
|
}
|
||||||
|
|
||||||
if (rv == -1)
|
if (rv == -1)
|
||||||
{
|
{
|
||||||
int save_errno = errno;
|
int save_errno = errno;
|
||||||
|
|
|
@ -1,22 +1,24 @@
|
||||||
;;;; socket.test --- test socket functions -*- scheme -*-
|
;;;; socket.test --- test socket functions -*- scheme -*-
|
||||||
;;;;
|
;;;;
|
||||||
;;;; Copyright (C) 2004, 2005, 2006, 2007, 2008, 2009, 2010 Free Software Foundation, Inc.
|
;;;; Copyright (C) 2004, 2005, 2006, 2007, 2008, 2009, 2010,
|
||||||
|
;;;; 2011 Free Software Foundation, Inc.
|
||||||
;;;;
|
;;;;
|
||||||
;;;; This library is free software; you can redistribute it and/or
|
;;;; This library is free software; you can redistribute it and/or
|
||||||
;;;; modify it under the terms of the GNU Lesser General Public
|
;;;; modify it under the terms of the GNU Lesser General Public
|
||||||
;;;; License as published by the Free Software Foundation; either
|
;;;; License as published by the Free Software Foundation; either
|
||||||
;;;; version 3 of the License, or (at your option) any later version.
|
;;;; version 3 of the License, or (at your option) any later version.
|
||||||
;;;;
|
;;;;
|
||||||
;;;; This library is distributed in the hope that it will be useful,
|
;;;; This library is distributed in the hope that it will be useful,
|
||||||
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
;;;; Lesser General Public License for more details.
|
;;;; Lesser General Public License for more details.
|
||||||
;;;;
|
;;;;
|
||||||
;;;; You should have received a copy of the GNU Lesser General Public
|
;;;; You should have received a copy of the GNU Lesser General Public
|
||||||
;;;; License along with this library; if not, write to the Free Software
|
;;;; License along with this library; if not, write to the Free Software
|
||||||
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
|
||||||
(define-module (test-suite test-socket)
|
(define-module (test-suite test-socket)
|
||||||
|
#:use-module (rnrs bytevectors)
|
||||||
#:use-module (test-suite lib))
|
#:use-module (test-suite lib))
|
||||||
|
|
||||||
|
|
||||||
|
@ -235,15 +237,17 @@
|
||||||
(pass-if "sendto"
|
(pass-if "sendto"
|
||||||
(if (not server-bound?)
|
(if (not server-bound?)
|
||||||
(throw 'unresolved)
|
(throw 'unresolved)
|
||||||
(let ((client (socket AF_UNIX SOCK_DGRAM 0)))
|
(let ((client (socket AF_UNIX SOCK_DGRAM 0))
|
||||||
(> (sendto client "hello" AF_UNIX path) 0))))
|
(message (string->utf8 "hello")))
|
||||||
|
(> (sendto client message AF_UNIX path) 0))))
|
||||||
|
|
||||||
(pass-if "sendto/sockaddr"
|
(pass-if "sendto/sockaddr"
|
||||||
(if (not server-bound?)
|
(if (not server-bound?)
|
||||||
(throw 'unresolved)
|
(throw 'unresolved)
|
||||||
(let ((client (socket AF_UNIX SOCK_DGRAM 0))
|
(let ((client (socket AF_UNIX SOCK_DGRAM 0))
|
||||||
|
(message (string->utf8 "hello"))
|
||||||
(sockaddr (make-socket-address AF_UNIX path)))
|
(sockaddr (make-socket-address AF_UNIX path)))
|
||||||
(> (sendto client "hello" sockaddr) 0))))
|
(> (sendto client message sockaddr) 0))))
|
||||||
|
|
||||||
(false-if-exception (delete-file path)))))
|
(false-if-exception (delete-file path)))))
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue