| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103 |
- // Copyright (c) 2014-2015 The Notify Authors. All rights reserved.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- package notify
- // EventDiff describes a change to an event set - EventDiff[0] is an old state,
- // while EventDiff[1] is a new state. If event set has not changed (old == new),
- // functions typically return the None value.
- type eventDiff [2]Event
- func (diff eventDiff) Event() Event {
- return diff[1] &^ diff[0]
- }
- // Watchpoint
- //
- // The nil key holds total event set - logical sum for all registered events.
- // It speeds up computing EventDiff for Add method.
- //
- // The rec key holds an event set for a watchpoints created by RecursiveWatch
- // for a Watcher implementation which is not natively recursive.
- type watchpoint map[chan<- EventInfo]Event
- // None is an empty event diff, think null object.
- var none eventDiff
- // rec is just a placeholder
- var rec = func() (ch chan<- EventInfo) {
- ch = make(chan<- EventInfo)
- close(ch)
- return
- }()
- func (wp watchpoint) dryAdd(ch chan<- EventInfo, e Event) eventDiff {
- if e &^= internal; wp[ch]&e == e {
- return none
- }
- total := wp[ch] &^ internal
- return eventDiff{total, total | e}
- }
- // Add assumes neither c nor e are nil or zero values.
- func (wp watchpoint) Add(c chan<- EventInfo, e Event) (diff eventDiff) {
- wp[c] |= e
- diff[0] = wp[nil]
- diff[1] = diff[0] | e
- wp[nil] = diff[1] &^ omit
- // Strip diff from internal events.
- diff[0] &^= internal
- diff[1] &^= internal
- if diff[0] == diff[1] {
- return none
- }
- return
- }
- func (wp watchpoint) Del(c chan<- EventInfo, e Event) (diff eventDiff) {
- wp[c] &^= e
- if wp[c] == 0 {
- delete(wp, c)
- }
- diff[0] = wp[nil]
- delete(wp, nil)
- if len(wp) != 0 {
- // Recalculate total event set.
- for _, e := range wp {
- diff[1] |= e
- }
- wp[nil] = diff[1] &^ omit
- }
- // Strip diff from internal events.
- diff[0] &^= internal
- diff[1] &^= internal
- if diff[0] == diff[1] {
- return none
- }
- return
- }
- func (wp watchpoint) Dispatch(ei EventInfo, extra Event) {
- e := eventmask(ei, extra)
- if !matches(wp[nil], e) {
- return
- }
- for ch, eset := range wp {
- if ch != nil && matches(eset, e) {
- select {
- case ch <- ei:
- default: // Drop event if receiver is too slow
- dbgprintf("dropped %s on %q: receiver too slow", ei.Event(), ei.Path())
- }
- }
- }
- }
- func (wp watchpoint) Total() Event {
- return wp[nil] &^ internal
- }
- func (wp watchpoint) IsRecursive() bool {
- return wp[nil]&recursive != 0
- }
|