(index<- )        ./librustuv/signal.rs

    git branch:    * master           5200215 auto merge of #14035 : alexcrichton/rust/experimental, r=huonw
    modified:    Fri May  9 13:02:28 2014
  1  // Copyright 2013 The Rust Project Developers. See the COPYRIGHT
  2  // file at the top-level directory of this distribution and at
  3  // http://rust-lang.org/COPYRIGHT.
  4  //
  5  // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
  6  // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
  7  // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
  8  // option. This file may not be copied, modified, or distributed
  9  // except according to those terms.
 10  
 11  use libc::c_int;
 12  use std::io::signal::Signum;
 13  use std::rt::rtio::RtioSignal;
 14  
 15  use homing::{HomingIO, HomeHandle};
 16  use super::{UvError, UvHandle};
 17  use uvll;
 18  use uvio::UvIoFactory;
 19  
 20  pub struct SignalWatcher {
 21      handle: *uvll::uv_signal_t,
 22      home: HomeHandle,
 23  
 24      channel: Sender<Signum>,
 25      signal: Signum,
 26  }
 27  
 28  impl SignalWatcher {
 29      pub fn new(io&mut UvIoFactory, signumSignum, channelSender<Signum>)
 30                 -> Result<Box<SignalWatcher>, UvError> {
 31          let s = box SignalWatcher {
 32              handle: UvHandle::alloc(None::<SignalWatcher>, uvll::UV_SIGNAL),
 33              home: io.make_handle(),
 34              channel: channel,
 35              signal: signum,
 36          };
 37          assert_eq!(unsafe {
 38              uvll::uv_signal_init(io.uv_loop(), s.handle)
 39          }, 0);
 40  
 41          match unsafe {
 42              uvll::uv_signal_start(s.handle, signal_cb, signum as c_int)
 43          } {
 44              0 => Ok(s.install()),
 45              n => Err(UvError(n)),
 46          }
 47  
 48      }
 49  }
 50  
 51  extern fn signal_cb(handle: *uvll::uv_signal_t, signumc_int) {
 52      let s&mut SignalWatcher = unsafe { UvHandle::from_uv_handle(&handle) };
 53      assert_eq!(signum as int, s.signal as int);
 54      let _ = s.channel.send_opt(s.signal);
 55  }
 56  
 57  impl HomingIO for SignalWatcher {
 58      fn home<'r>(&'r mut self) -> &'r mut HomeHandle { &mut self.home }
 59  }
 60  
 61  impl UvHandle<uvll::uv_signal_t> for SignalWatcher {
 62      fn uv_handle(&self) -> *uvll::uv_signal_t { self.handle }
 63  }
 64  
 65  impl RtioSignal for SignalWatcher {}
 66  
 67  impl Drop for SignalWatcher {
 68      fn drop(&mut self) {
 69          let _m = self.fire_homing_missile();
 70          self.close();
 71      }
 72  }
 73  
 74  #[cfg(test)]
 75  mod test {
 76      use super::super::local_loop;
 77      use std::io::signal;
 78      use super::SignalWatcher;
 79  
 80      #[test]
 81      fn closing_channel_during_drop_doesnt_kill_everything() {
 82          // see issue #10375, relates to timers as well.
 83          let (tx, rx) = channel();
 84          let _signal = SignalWatcher::new(local_loop(), signal::Interrupt,
 85                                           tx);
 86  
 87          spawn(proc() {
 88              let _ = rx.recv_opt();
 89          });
 90  
 91          // when we drop the SignalWatcher we're going to destroy the channel,
 92          // which must wake up the task on the other end
 93      }
 94  }