Class: NIO::Selector

Inherits:
Object
  • Object
show all
Defined in:
lib/nio/selector.rb,
ext/nio4r/selector.c

Overview

Selectors monitor IO objects for events of interest

Instance Method Summary collapse

Constructor Details

#initializeObject

Methods



5
6
7
8
9
10
11
12
# File 'lib/nio/selector.rb', line 5

def initialize
  @selectables = {}
  @lock = Mutex.new

  # Other threads can wake up a selector
  @wakeup, @waker = IO.pipe
  @closed = false
end

Instance Method Details

#closeObject

Close this selector and free its resources



110
111
112
113
114
115
116
117
118
# File 'lib/nio/selector.rb', line 110

def close
  @lock.synchronize do
    return if @closed

    @wakeup.close rescue nil
    @waker.close rescue nil
    @closed = true
  end
end

#closed?Boolean

Is this selector closed?

Returns:

  • (Boolean)


121
# File 'lib/nio/selector.rb', line 121

def closed?; @closed end

#deregister(io) ⇒ Object

Deregister the given IO object from the selector



33
34
35
36
37
38
39
# File 'lib/nio/selector.rb', line 33

def deregister(io)
  @lock.synchronize do
    monitor = @selectables.delete io
    monitor.close(false) if monitor and not monitor.closed?
    monitor
  end
end

#empty?Boolean

Returns:

  • (Boolean)


31
32
33
# File 'ext/nio4r/selector.c', line 31

def empty?
  @selectables.empty?
end

#register(io, interest) ⇒ Object

Register interest in an IO object with the selector for the given types of events. Valid event types for interest are:

  • :r - is the IO readable?

  • :w - is the IO writeable?

  • :rw - is the IO either readable or writeable?



19
20
21
22
23
24
25
26
27
28
29
30
# File 'lib/nio/selector.rb', line 19

def register(io, interest)
  @lock.synchronize do
    if monitor = @selectables[io]
      raise ArgumentError, "this IO is already registered with the selector as #{monitor.interests.inspect}"
    end

    monitor = Monitor.new(io, interest, self)
    @selectables[io] = monitor

    monitor
  end
end

#registered?(io) ⇒ Boolean

Is the given IO object registered with the selector?

Returns:

  • (Boolean)


42
43
44
# File 'lib/nio/selector.rb', line 42

def registered?(io)
  @lock.synchronize { @selectables.has_key? io }
end

#select(timeout = nil) ⇒ Object

Select which monitors are ready



47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
# File 'lib/nio/selector.rb', line 47

def select(timeout = nil)
  @lock.synchronize do
    readers, writers = [@wakeup], []

    @selectables.each do |io, monitor|
      readers << io if monitor.interests == :r || monitor.interests == :rw
      writers << io if monitor.interests == :w || monitor.interests == :rw
      monitor.readiness = nil
    end

    ready_readers, ready_writers = Kernel.select readers, writers, [], timeout
    return unless ready_readers # timeout or wakeup
    
    selected_monitors = Set.new

    ready_readers.each do |io|
      if io == @wakeup
        # Clear all wakeup signals we've received by reading them
        # Wakeups should have level triggered behavior
        @wakeup.read(@wakeup.stat.size)
        return
      else
        monitor = @selectables[io]
        monitor.readiness = :r
        selected_monitors << monitor
      end
    end
    
    ready_writers.each do |io|
      monitor = @selectables[io]
      monitor.readiness = case monitor.readiness
      when :r
        :rw
      else
        :w
      end
      selected_monitors << monitor
    end
    
    if block_given?
      selected_monitors.each do |m|
        yield m
      end
      selected_monitors.size
    else
      selected_monitors
    end
  end
end

#wakeupObject

Wake up a thread that’s in the middle of selecting on this selector, if any such thread exists.

Invoking this method more than once between two successive select calls has the same effect as invoking it just once. In other words, it provides level-triggered behavior.



103
104
105
106
107
# File 'lib/nio/selector.rb', line 103

def wakeup
  # Send the selector a signal in the form of writing data to a pipe
  @waker.write "\0"
  nil
end