mod.rs 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. //! poll implementation for Redox, following http://pubs.opengroup.org/onlinepubs/7908799/xsh/poll.h.html
  2. use core::{mem, slice};
  3. use fs::File;
  4. use header::sys_epoll::{
  5. epoll_create1, EPOLL_CLOEXEC,
  6. epoll_ctl, EPOLL_CTL_ADD,
  7. epoll_wait,
  8. EPOLLIN, EPOLLPRI, EPOLLOUT, EPOLLERR, EPOLLHUP, EPOLLNVAL,
  9. epoll_data, epoll_event
  10. };
  11. use platform::types::*;
  12. pub const POLLIN: c_short = 0x001;
  13. pub const POLLPRI: c_short = 0x002;
  14. pub const POLLOUT: c_short = 0x004;
  15. pub const POLLERR: c_short = 0x008;
  16. pub const POLLHUP: c_short = 0x010;
  17. pub const POLLNVAL: c_short = 0x020;
  18. pub type nfds_t = c_ulong;
  19. #[repr(C)]
  20. pub struct pollfd {
  21. pub fd: c_int,
  22. pub events: c_short,
  23. pub revents: c_short,
  24. }
  25. pub fn poll_epoll(fds: &mut [pollfd], timeout: c_int) -> c_int {
  26. let event_map = [
  27. (POLLIN, EPOLLIN),
  28. (POLLPRI, EPOLLPRI),
  29. (POLLOUT, EPOLLOUT),
  30. (POLLERR, EPOLLERR),
  31. (POLLHUP, EPOLLHUP),
  32. (POLLNVAL, EPOLLNVAL)
  33. ];
  34. let ep = {
  35. let epfd = epoll_create1(EPOLL_CLOEXEC);
  36. if epfd < 0 {
  37. return -1;
  38. }
  39. File::new(epfd)
  40. };
  41. for i in 0..fds.len() {
  42. let mut pfd = &mut fds[i];
  43. let mut event = epoll_event {
  44. events: 0,
  45. data: epoll_data {
  46. u64: i as u64,
  47. },
  48. };
  49. for (p, ep) in event_map.iter() {
  50. if pfd.events & p > 0 {
  51. event.events |= ep;
  52. }
  53. }
  54. pfd.revents = 0;
  55. if epoll_ctl(*ep, EPOLL_CTL_ADD, pfd.fd, &mut event) < 0 {
  56. return -1;
  57. }
  58. }
  59. let mut events: [epoll_event; 32] = unsafe { mem::zeroed() };
  60. let res = epoll_wait(
  61. *ep,
  62. events.as_mut_ptr(),
  63. events.len() as c_int,
  64. timeout
  65. );
  66. if res < 0 {
  67. return -1;
  68. }
  69. for i in 0..res as usize {
  70. let event = &events[i];
  71. let pi = unsafe { event.data.u64 as usize };
  72. // TODO: Error status when fd does not match?
  73. if let Some(pfd) = fds.get_mut(pi) {
  74. for (p, ep) in event_map.iter() {
  75. if event.events & ep > 0 {
  76. pfd.revents |= p;
  77. }
  78. }
  79. }
  80. }
  81. let mut count = 0;
  82. for pfd in fds.iter() {
  83. if pfd.revents > 0 {
  84. count += 1;
  85. }
  86. }
  87. count
  88. }
  89. #[no_mangle]
  90. pub unsafe extern "C" fn poll(fds: *mut pollfd, nfds: nfds_t, timeout: c_int) -> c_int {
  91. trace_expr!(
  92. poll_epoll(
  93. slice::from_raw_parts_mut(fds, nfds as usize),
  94. timeout
  95. ),
  96. "poll({:p}, {}, {})",
  97. fds,
  98. nfds,
  99. timeout
  100. )
  101. }