diff options
author | milo <milo@r0ot.me> | 2011-04-20 04:13:01 +0200 |
---|---|---|
committer | milo <milo@r0ot.me> | 2011-05-02 17:41:51 +0200 |
commit | 833cc00014f35d4c0fcf7086529a84f2500947b1 (patch) | |
tree | 48cfac95648b815f5d643f7cb55681a17b18c738 /src | |
parent | 09e8cf33d7d12db2b2e7f23be7638999bb557c0d (diff) | |
download | libssh-833cc00014f35d4c0fcf7086529a84f2500947b1.tar.gz libssh-833cc00014f35d4c0fcf7086529a84f2500947b1.tar.xz libssh-833cc00014f35d4c0fcf7086529a84f2500947b1.zip |
[socket] fix a segfault at disconnect
(cherry picked from commit 7d2064c2894fc298cc8923cae657200900b032e5)
Diffstat (limited to 'src')
-rw-r--r-- | src/socket.c | 19 |
1 files changed, 15 insertions, 4 deletions
diff --git a/src/socket.c b/src/socket.c index 6b15e6d5..f3da4280 100644 --- a/src/socket.c +++ b/src/socket.c @@ -246,8 +246,9 @@ int ssh_socket_pollcallback(struct ssh_poll_handle_struct *p, socket_t fd, int r s->read_wontblock=1; r=ssh_socket_unbuffered_read(s,buffer,sizeof(buffer)); if(r<0){ - if(p != NULL) - ssh_poll_set_events(p,ssh_poll_get_events(p) & ~POLLIN); + if(p != NULL) { + ssh_poll_remove_events(p, POLLIN); + } if(s->callbacks && s->callbacks->exception){ s->callbacks->exception( SSH_SOCKET_EXCEPTION_ERROR, @@ -255,7 +256,12 @@ int ssh_socket_pollcallback(struct ssh_poll_handle_struct *p, socket_t fd, int r } } if(r==0){ - ssh_poll_set_events(p,ssh_poll_get_events(p) & ~POLLIN); + if(p != NULL) { + ssh_poll_remove_events(p, POLLIN); + } + if(p != NULL) { + ssh_poll_remove_events(p, POLLIN); + } if(s->callbacks && s->callbacks->exception){ s->callbacks->exception( SSH_SOCKET_EXCEPTION_EOF, @@ -270,6 +276,9 @@ int ssh_socket_pollcallback(struct ssh_poll_handle_struct *p, socket_t fd, int r buffer_get_rest_len(s->in_buffer), s->callbacks->userdata); buffer_pass_bytes(s->in_buffer,r); + /* p may have been freed, so don't use it + * anymore in this function */ + p = NULL; } } } @@ -290,7 +299,9 @@ int ssh_socket_pollcallback(struct ssh_poll_handle_struct *p, socket_t fd, int r } /* So, we can write data */ s->write_wontblock=1; - ssh_poll_remove_events(p,POLLOUT); + if(p != NULL) { + ssh_poll_remove_events(p, POLLOUT); + } /* If buffered data is pending, write it */ if(buffer_get_rest_len(s->out_buffer) > 0){ |