| // Copyright 2016 The Fuchsia Authors | 
 | // | 
 | // Use of this source code is governed by a MIT-style | 
 | // license that can be found in the LICENSE file or at | 
 | // https://opensource.org/licenses/MIT | 
 |  | 
 | #include <object/wait_state_observer.h> | 
 |  | 
 | #include <assert.h> | 
 |  | 
 | #include <kernel/event.h> | 
 |  | 
 | #include <object/handle.h> | 
 | #include <object/dispatcher.h> | 
 |  | 
 | #include <fbl/type_support.h> | 
 |  | 
 | WaitStateObserver::~WaitStateObserver() { | 
 |     DEBUG_ASSERT(!dispatcher_); | 
 | } | 
 |  | 
 | zx_status_t WaitStateObserver::Begin(Event* event, | 
 |                                      Handle* handle, | 
 |                                      zx_signals_t watched_signals) { | 
 |     canary_.Assert(); | 
 |     DEBUG_ASSERT(!dispatcher_); | 
 |  | 
 |     event_ = event; | 
 |     handle_ = handle; | 
 |     watched_signals_ = watched_signals; | 
 |     dispatcher_ = handle->dispatcher(); | 
 |     wakeup_reasons_ = 0u; | 
 |  | 
 |     auto status = dispatcher_->add_observer(this); | 
 |     if (status != ZX_OK) { | 
 |         dispatcher_.reset(); | 
 |         return status; | 
 |     } | 
 |     return ZX_OK; | 
 | } | 
 |  | 
 | zx_signals_t WaitStateObserver::End() { | 
 |     canary_.Assert(); | 
 |     DEBUG_ASSERT(dispatcher_); | 
 |     DEBUG_ASSERT(dispatcher_->has_state_tracker()); | 
 |  | 
 |     dispatcher_->RemoveObserver(this); | 
 |     dispatcher_.reset(); | 
 |  | 
 |     // Return the set of reasons that we may have been woken.  Basically, this | 
 |     // is set of satisfied bits which were ever set while we were waiting on the list. | 
 |     return wakeup_reasons_; | 
 | } | 
 |  | 
 | StateObserver::Flags WaitStateObserver::OnInitialize(zx_signals_t initial_state, | 
 |                                                      const StateObserver::CountInfo* cinfo) { | 
 |     canary_.Assert(); | 
 |  | 
 |     // Record the initial state of the state tracker as our wakeup reason.  If | 
 |     // we are going to become immediately signaled, the reason is contained | 
 |     // somewhere in this initial state. | 
 |     wakeup_reasons_ = initial_state; | 
 |  | 
 |     if (initial_state & watched_signals_) { | 
 |         event_->Signal(); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | StateObserver::Flags WaitStateObserver::OnStateChange(zx_signals_t new_state) { | 
 |     canary_.Assert(); | 
 |  | 
 |     // If we are still on our StateTracker's list of observers, and the | 
 |     // StateTracker's state has changed, accumulate the reasons that we may have | 
 |     // woken up.  In particular any satisfied bits which have become set | 
 |     // while we were on the list may have been reasons to wake up. | 
 |     wakeup_reasons_ |= new_state; | 
 |  | 
 |     if (new_state & watched_signals_) { | 
 |         event_->Signal(); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | StateObserver::Flags WaitStateObserver::OnCancel(const Handle* handle) { | 
 |     canary_.Assert(); | 
 |  | 
 |     if (handle == handle_) { | 
 |         wakeup_reasons_ |= ZX_SIGNAL_HANDLE_CLOSED; | 
 |         event_->Signal(ZX_ERR_CANCELED); | 
 |         return kHandled; | 
 |     } else { | 
 |         return 0; | 
 |     } | 
 | } |