mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
* ext/readline/readline.c (Init_readline, readline_s_set_output)
(clear_rl_outstream, readline_s_set_input, clear_rl_instream) (readline_readline): fix causing SEGV if closed IO object that is set Readline.input or Readline.output. Patched by akr [ruby-dev:47509] [Bug #8644] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@42402 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
b7bc4706fa
commit
d2fa5dff12
2 changed files with 119 additions and 20 deletions
|
@ -1,3 +1,11 @@
|
||||||
|
Tue Aug 6 21:14:11 2013 Kouji Takao <kouji.takao@gmail.com>
|
||||||
|
|
||||||
|
* ext/readline/readline.c (Init_readline, readline_s_set_output)
|
||||||
|
(clear_rl_outstream, readline_s_set_input, clear_rl_instream)
|
||||||
|
(readline_readline): fix causing SEGV if closed IO object that is
|
||||||
|
set Readline.input or Readline.output. Patched by akr
|
||||||
|
[ruby-dev:47509] [Bug #8644]
|
||||||
|
|
||||||
Tue Aug 6 17:56:40 2013 Koichi Sasada <ko1@atdot.net>
|
Tue Aug 6 17:56:40 2013 Koichi Sasada <ko1@atdot.net>
|
||||||
|
|
||||||
* vm_insnhelper.c (vm_push_frame): change type of stack_max to size_t.
|
* vm_insnhelper.c (vm_push_frame): change type of stack_max to size_t.
|
||||||
|
|
|
@ -132,8 +132,10 @@ static char **readline_attempted_completion_function(const char *text,
|
||||||
* Documented by TAKAO Kouji <kouji at takao7 dot net>.
|
* Documented by TAKAO Kouji <kouji at takao7 dot net>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#if defined HAVE_RL_GETC_FUNCTION
|
|
||||||
static VALUE readline_instream;
|
static VALUE readline_instream;
|
||||||
|
static VALUE readline_outstream;
|
||||||
|
|
||||||
|
#if defined HAVE_RL_GETC_FUNCTION
|
||||||
|
|
||||||
#ifndef HAVE_RL_GETC
|
#ifndef HAVE_RL_GETC
|
||||||
#define rl_getc(f) EOF
|
#define rl_getc(f) EOF
|
||||||
|
@ -302,9 +304,9 @@ readline_get(VALUE prompt)
|
||||||
* Returns nil when the inputted line is empty and user inputs EOF
|
* Returns nil when the inputted line is empty and user inputs EOF
|
||||||
* (Presses ^D on UNIX).
|
* (Presses ^D on UNIX).
|
||||||
*
|
*
|
||||||
* Raises IOError exception if below conditions are satisfied.
|
* Raises IOError exception if one of below conditions are satisfied.
|
||||||
* 1. stdin is not tty.
|
* 1. stdin was closed.
|
||||||
* 2. stdin was closed. (errno is EBADF after called isatty(2).)
|
* 2. stdout was closed.
|
||||||
*
|
*
|
||||||
* This method supports thread. Switches the thread context when waits
|
* This method supports thread. Switches the thread context when waits
|
||||||
* inputting line.
|
* inputting line.
|
||||||
|
@ -395,13 +397,30 @@ readline_readline(int argc, VALUE *argv, VALUE self)
|
||||||
prompt = RSTRING_PTR(tmp);
|
prompt = RSTRING_PTR(tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isatty(fileno(rl_instream)) && errno == EBADF) rb_raise(rb_eIOError, "closed stdin");
|
if (readline_instream) {
|
||||||
if (rl_outstream) {
|
rb_io_t *ifp;
|
||||||
struct stat stbuf;
|
GetOpenFile(readline_instream, ifp);
|
||||||
int fd = fileno(rl_outstream);
|
if (ifp->fd < 0) {
|
||||||
if (fd < 0 || fstat(fd, &stbuf) != 0) {
|
if (rl_instream) {
|
||||||
rb_raise(rb_eIOError, "closed stdout");
|
fclose(rl_instream);
|
||||||
}
|
rl_instream = NULL;
|
||||||
|
}
|
||||||
|
readline_instream = Qfalse;
|
||||||
|
rb_raise(rb_eIOError, "closed stdin");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (readline_outstream) {
|
||||||
|
rb_io_t *ofp;
|
||||||
|
GetOpenFile(readline_outstream, ofp);
|
||||||
|
if (ofp->fd < 0) {
|
||||||
|
if (rl_outstream) {
|
||||||
|
fclose(rl_outstream);
|
||||||
|
rl_outstream = NULL;
|
||||||
|
}
|
||||||
|
readline_outstream = Qfalse;
|
||||||
|
rb_raise(rb_eIOError, "closed stdout");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
|
@ -443,6 +462,23 @@ readline_readline(int argc, VALUE *argv, VALUE self)
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
clear_rl_instream(void)
|
||||||
|
{
|
||||||
|
rb_io_t *ifp;
|
||||||
|
if (rl_instream) {
|
||||||
|
if (readline_instream) {
|
||||||
|
rb_io_check_initialized(ifp = RFILE(rb_io_taint_check(readline_instream))->fptr);
|
||||||
|
if (ifp->fd < 0 || fileno(rl_instream) == ifp->fd) {
|
||||||
|
fclose(rl_instream);
|
||||||
|
rl_instream = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
readline_instream = Qfalse;
|
||||||
|
rl_instream = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* Readline.input = input
|
* Readline.input = input
|
||||||
|
@ -456,16 +492,49 @@ static VALUE
|
||||||
readline_s_set_input(VALUE self, VALUE input)
|
readline_s_set_input(VALUE self, VALUE input)
|
||||||
{
|
{
|
||||||
rb_io_t *ifp;
|
rb_io_t *ifp;
|
||||||
|
int fd;
|
||||||
|
FILE *f;
|
||||||
|
|
||||||
Check_Type(input, T_FILE);
|
if (NIL_P(input)) {
|
||||||
GetOpenFile(input, ifp);
|
clear_rl_instream();
|
||||||
rl_instream = rb_io_stdio_file(ifp);
|
}
|
||||||
#ifdef HAVE_RL_GETC_FUNCTION
|
else {
|
||||||
readline_instream = input;
|
Check_Type(input, T_FILE);
|
||||||
#endif
|
GetOpenFile(input, ifp);
|
||||||
|
clear_rl_instream();
|
||||||
|
fd = rb_cloexec_dup(ifp->fd);
|
||||||
|
if (fd == -1)
|
||||||
|
rb_sys_fail("dup");
|
||||||
|
f = fdopen(fd, "r");
|
||||||
|
if (f == NULL) {
|
||||||
|
int save_errno = errno;
|
||||||
|
close(fd);
|
||||||
|
errno = save_errno;
|
||||||
|
rb_sys_fail("fdopen");
|
||||||
|
}
|
||||||
|
rl_instream = f;
|
||||||
|
readline_instream = input;
|
||||||
|
}
|
||||||
return input;
|
return input;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
clear_rl_outstream(void)
|
||||||
|
{
|
||||||
|
rb_io_t *ofp;
|
||||||
|
if (rl_outstream) {
|
||||||
|
if (readline_outstream) {
|
||||||
|
rb_io_check_initialized(ofp = RFILE(rb_io_taint_check(readline_outstream))->fptr);
|
||||||
|
if (ofp->fd < 0 || fileno(rl_outstream) == ofp->fd) {
|
||||||
|
fclose(rl_outstream);
|
||||||
|
rl_outstream = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
readline_outstream = Qfalse;
|
||||||
|
rl_outstream = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* Readline.output = output
|
* Readline.output = output
|
||||||
|
@ -479,10 +548,29 @@ static VALUE
|
||||||
readline_s_set_output(VALUE self, VALUE output)
|
readline_s_set_output(VALUE self, VALUE output)
|
||||||
{
|
{
|
||||||
rb_io_t *ofp;
|
rb_io_t *ofp;
|
||||||
|
int fd;
|
||||||
|
FILE *f;
|
||||||
|
|
||||||
Check_Type(output, T_FILE);
|
if (NIL_P(output)) {
|
||||||
GetOpenFile(output, ofp);
|
clear_rl_outstream();
|
||||||
rl_outstream = rb_io_stdio_file(ofp);
|
}
|
||||||
|
else {
|
||||||
|
Check_Type(output, T_FILE);
|
||||||
|
GetOpenFile(output, ofp);
|
||||||
|
clear_rl_outstream();
|
||||||
|
fd = rb_cloexec_dup(ofp->fd);
|
||||||
|
if (fd == -1)
|
||||||
|
rb_sys_fail("dup");
|
||||||
|
f = fdopen(fd, "w");
|
||||||
|
if (f == NULL) {
|
||||||
|
int save_errno = errno;
|
||||||
|
close(fd);
|
||||||
|
errno = save_errno;
|
||||||
|
rb_sys_fail("fdopen");
|
||||||
|
}
|
||||||
|
rl_outstream = f;
|
||||||
|
readline_outstream = output;
|
||||||
|
}
|
||||||
return output;
|
return output;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1906,5 +1994,8 @@ Init_readline()
|
||||||
rl_clear_signals();
|
rl_clear_signals();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
rb_gc_register_address(&readline_instream);
|
||||||
|
rb_gc_register_address(&readline_outstream);
|
||||||
|
|
||||||
readline_s_set_input(mReadline, rb_stdin);
|
readline_s_set_input(mReadline, rb_stdin);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue