123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 |
- /* Copyright 2001,2006,2009,2011,2014,2018
- Free Software Foundation, Inc.
- This file is part of Guile.
- Guile is free software: you can redistribute it and/or modify it
- under the terms of the GNU Lesser General Public License as published
- by the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
- Guile is distributed in the hope that it will be useful, but WITHOUT
- ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
- FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
- License for more details.
- You should have received a copy of the GNU Lesser General Public
- License along with Guile. If not, see
- <https://www.gnu.org/licenses/>. */
- /* This is the C part of the (ice-9 rw) module. */
- #ifdef HAVE_CONFIG_H
- # include <config.h>
- #endif
- #include <errno.h>
- #include <string.h>
- #include <unistd.h>
- #ifdef HAVE_IO_H
- #include <io.h>
- #endif
- #include "async.h"
- #include "fports.h"
- #include "gsubr.h"
- #include "modules.h"
- #include "numbers.h"
- #include "ports-internal.h"
- #include "ports.h"
- #include "strings.h"
- #include "strports.h"
- #include "syscalls.h"
- #include "rw.h"
- #if defined (EAGAIN)
- #define SCM_MAYBE_EAGAIN || errno == EAGAIN
- #else
- #define SCM_MAYBE_EAGAIN
- #endif
- #if defined (EWOULDBLOCK)
- #define SCM_MAYBE_EWOULDBLOCK || errno == EWOULDBLOCK
- #else
- #define SCM_MAYBE_EWOULDBLOCK
- #endif
- /* MAYBE there is EAGAIN way of defining this macro but now I EWOULDBLOCK. */
- #define SCM_EBLOCK(errno) \
- (0 SCM_MAYBE_EAGAIN SCM_MAYBE_EWOULDBLOCK)
- SCM_DEFINE (scm_read_string_x_partial, "read-string!/partial", 1, 3, 0,
- (SCM str, SCM port_or_fdes, SCM start, SCM end),
- "Read characters from a port or file descriptor into a\n"
- "string @var{str}. A port must have an underlying file\n"
- "descriptor --- a so-called fport. This procedure is\n"
- "scsh-compatible and can efficiently read large strings.\n"
- "It will:\n\n"
- "@itemize\n"
- "@item\n"
- "attempt to fill the entire string, unless the @var{start}\n"
- "and/or @var{end} arguments are supplied. i.e., @var{start}\n"
- "defaults to 0 and @var{end} defaults to\n"
- "@code{(string-length str)}\n"
- "@item\n"
- "use the current input port if @var{port_or_fdes} is not\n"
- "supplied.\n"
- "@item\n"
- "return fewer than the requested number of characters in some\n"
- "cases, e.g., on end of file, if interrupted by a signal, or if\n"
- "not all the characters are immediately available.\n"
- "@item\n"
- "wait indefinitely for some input if no characters are\n"
- "currently available,\n"
- "unless the port is in non-blocking mode.\n"
- "@item\n"
- "read characters from the port's input buffers if available,\n"
- "instead from the underlying file descriptor.\n"
- "@item\n"
- "return @code{#f} if end-of-file is encountered before reading\n"
- "any characters, otherwise return the number of characters\n"
- "read.\n"
- "@item\n"
- "return 0 if the port is in non-blocking mode and no characters\n"
- "are immediately available.\n"
- "@item\n"
- "return 0 if the request is for 0 bytes, with no\n"
- "end-of-file check.\n"
- "@end itemize")
- #define FUNC_NAME s_scm_read_string_x_partial
- {
- char *dest;
- size_t offset;
- long read_len;
- long chars_read = 0;
- int fdes;
- {
- size_t last;
- SCM_VALIDATE_STRING (1, str);
- scm_i_get_substring_spec (scm_i_string_length (str),
- start, &offset, end, &last);
- read_len = last - offset;
- }
- if (scm_is_integer (port_or_fdes))
- fdes = scm_to_int (port_or_fdes);
- else
- {
- SCM port = (SCM_UNBNDP (port_or_fdes)?
- scm_current_input_port () : port_or_fdes);
- SCM_VALIDATE_OPFPORT (2, port);
- SCM_VALIDATE_INPUT_PORT (2, port);
- /* if there's anything in the port buffers, use it, but then
- don't touch the file descriptor. otherwise the
- "return immediately if something is available" rule may
- be violated. */
- str = scm_i_string_start_writing (str);
- dest = scm_i_string_writable_chars (str) + offset;
- chars_read = scm_take_from_input_buffers (port, dest, read_len);
- scm_i_string_stop_writing ();
- fdes = SCM_FPORT_FDES (port);
- }
- if (chars_read == 0 && read_len > 0) /* don't confuse read_len == 0 with
- EOF. */
- {
- str = scm_i_string_start_writing (str);
- dest = scm_i_string_writable_chars (str) + offset;
- SCM_SYSCALL (chars_read = read (fdes, dest, read_len));
- scm_i_string_stop_writing ();
- if (chars_read == -1)
- {
- if (SCM_EBLOCK (errno))
- chars_read = 0;
- else
- SCM_SYSERROR;
- }
- else if (chars_read == 0)
- {
- scm_remember_upto_here_1 (str);
- return SCM_BOOL_F;
- }
- }
- scm_remember_upto_here_1 (str);
- return scm_from_long (chars_read);
- }
- #undef FUNC_NAME
- SCM_DEFINE (scm_write_string_partial, "write-string/partial", 1, 3, 0,
- (SCM str, SCM port_or_fdes, SCM start, SCM end),
- "Write characters from a string @var{str} to a port or file\n"
- "descriptor. A port must have an underlying file descriptor\n"
- "--- a so-called fport. This procedure is\n"
- "scsh-compatible and can efficiently write large strings.\n"
- "It will:\n\n"
- "@itemize\n"
- "@item\n"
- "attempt to write the entire string, unless the @var{start}\n"
- "and/or @var{end} arguments are supplied. i.e., @var{start}\n"
- "defaults to 0 and @var{end} defaults to\n"
- "@code{(string-length str)}\n"
- "@item\n"
- "use the current output port if @var{port_of_fdes} is not\n"
- "supplied.\n"
- "@item\n"
- "in the case of a buffered port, store the characters in the\n"
- "port's output buffer, if all will fit. If they will not fit\n"
- "then any existing buffered characters will be flushed\n"
- "before attempting\n"
- "to write the new characters directly to the underlying file\n"
- "descriptor. If the port is in non-blocking mode and\n"
- "buffered characters can not be flushed immediately, then an\n"
- "@code{EAGAIN} system-error exception will be raised (Note:\n"
- "scsh does not support the use of non-blocking buffered ports.)\n"
- "@item\n"
- "write fewer than the requested number of\n"
- "characters in some cases, e.g., if interrupted by a signal or\n"
- "if not all of the output can be accepted immediately.\n"
- "@item\n"
- "wait indefinitely for at least one character\n"
- "from @var{str} to be accepted by the port, unless the port is\n"
- "in non-blocking mode.\n"
- "@item\n"
- "return the number of characters accepted by the port.\n"
- "@item\n"
- "return 0 if the port is in non-blocking mode and can not accept\n"
- "at least one character from @var{str} immediately\n"
- "@item\n"
- "return 0 immediately if the request size is 0 bytes.\n"
- "@end itemize")
- #define FUNC_NAME s_scm_write_string_partial
- {
- const char *src;
- scm_t_off write_len;
- int fdes;
- {
- size_t offset;
- size_t last;
- SCM_VALIDATE_STRING (1, str);
- src = scm_i_string_chars (str);
- scm_i_get_substring_spec (scm_i_string_length (str),
- start, &offset, end, &last);
- src += offset;
- write_len = last - offset;
- }
- if (write_len == 0)
- return SCM_INUM0;
- if (scm_is_integer (port_or_fdes))
- fdes = scm_to_int (port_or_fdes);
- else
- {
- SCM port = (SCM_UNBNDP (port_or_fdes)?
- scm_current_output_port () : port_or_fdes);
- SCM write_buf;
- size_t end;
- SCM_VALIDATE_OPFPORT (2, port);
- SCM_VALIDATE_OUTPUT_PORT (2, port);
- write_buf = SCM_PORT (port)->write_buf;
- /* Filling the last character in the buffer would require a
- flush. */
- if (write_len < scm_port_buffer_can_put (write_buf, &end))
- {
- scm_c_write (port, src, write_len);
- return scm_from_long (write_len);
- }
- scm_flush (port);
- fdes = SCM_FPORT_FDES (port);
- }
- {
- long rv;
- SCM_SYSCALL (rv = write (fdes, src, write_len));
- if (rv == -1)
- {
- if (SCM_EBLOCK (errno))
- rv = 0;
- else
- SCM_SYSERROR;
- }
- scm_remember_upto_here_1 (str);
- return scm_from_long (rv);
- }
- }
- #undef FUNC_NAME
- SCM
- scm_init_rw_builtins ()
- {
- #include "rw.x"
- return SCM_UNSPECIFIED;
- }
- void
- scm_init_rw ()
- {
- scm_c_define_gsubr ("%init-rw-builtins", 0, 0, 0, scm_init_rw_builtins);
- }
|