// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "remoting/host/curtain_mode.h" #include #include #include #include #include "base/logging.h" #include "base/mac/mac_util.h" #include "base/mac/scoped_cftyperef.h" namespace { const char* kCGSessionPath = "/System/Library/CoreServices/Menu Extras/User.menu/Contents/Resources/" "CGSession"; } namespace remoting { class CurtainModeMac : public CurtainMode { public: CurtainModeMac(const base::Closure& on_session_activate, const base::Closure& on_error); virtual ~CurtainModeMac(); // Overriden from CurtainMode. virtual void SetActivated(bool activated) OVERRIDE; private: // If the current session is attached to the console and is not showing // the logon screen then switch it out to ensure privacy. bool ActivateCurtain(); // Add or remove the switch-in event handler. bool InstallEventHandler(); bool RemoveEventHandler(); // Handlers for the switch-in event. static OSStatus SessionActivateHandler(EventHandlerCallRef handler, EventRef event, void* user_data); void OnSessionActivate(); base::Closure on_session_activate_; base::Closure on_error_; EventHandlerRef event_handler_; DISALLOW_COPY_AND_ASSIGN(CurtainModeMac); }; CurtainModeMac::CurtainModeMac(const base::Closure& on_session_activate, const base::Closure& on_error) : on_session_activate_(on_session_activate), on_error_(on_error), event_handler_(NULL) { } CurtainModeMac::~CurtainModeMac() { SetActivated(false); } void CurtainModeMac::SetActivated(bool activated) { if (activated) { if (!ActivateCurtain()) { on_error_.Run(); } } else { RemoveEventHandler(); } } bool CurtainModeMac::ActivateCurtain() { // Curtain mode causes problems with the login screen on Lion only (starting // with 10.7.3), so disable it on that platform. There is a work-around, but // it involves modifying a system Plist pertaining to power-management, so // it's not something that should be done automatically. For more details, // see https://discussions.apple.com/thread/3209415?start=690&tstart=0 // // TODO(jamiewalch): If the underlying OS bug is ever fixed, we should support // curtain mode on suitable versions of Lion. if (base::mac::IsOSLion()) { LOG(ERROR) << "Host curtaining is not supported on Mac OS X 10.7."; return false; } // Try to install the switch-in handler. Do this before switching out the // current session so that the console session is not affected if it fails. if (!InstallEventHandler()) { return false; } base::mac::ScopedCFTypeRef session( CGSessionCopyCurrentDictionary()); // CGSessionCopyCurrentDictionary has been observed to return NULL in some // cases. Once the system is in this state, curtain mode will fail as the // CGSession command thinks the session is not attached to the console. The // only known remedy is logout or reboot. Since we're not sure what causes // this, or how common it is, a crash report is useful in this case (note // that the connection would have to be refused in any case, so this is no // loss of functionality). CHECK(session != NULL); const void* on_console = CFDictionaryGetValue(session, kCGSessionOnConsoleKey); const void* logged_in = CFDictionaryGetValue(session, kCGSessionLoginDoneKey); if (logged_in == kCFBooleanTrue && on_console == kCFBooleanTrue) { pid_t child = fork(); if (child == 0) { execl(kCGSessionPath, kCGSessionPath, "-suspend", NULL); _exit(1); } else if (child > 0) { int status = 0; waitpid(child, &status, 0); if (status != 0) { LOG(ERROR) << kCGSessionPath << " failed."; return false; } } else { LOG(ERROR) << "fork() failed."; return false; } } return true; } OSStatus CurtainModeMac::SessionActivateHandler(EventHandlerCallRef handler, EventRef event, void* user_data) { CurtainModeMac* self = static_cast(user_data); self->OnSessionActivate(); return noErr; } void CurtainModeMac::OnSessionActivate() { on_session_activate_.Run(); } bool CurtainModeMac::InstallEventHandler() { OSStatus result = noErr; if (!event_handler_) { EventTypeSpec event; event.eventClass = kEventClassSystem; event.eventKind = kEventSystemUserSessionActivated; result = ::InstallApplicationEventHandler( NewEventHandlerUPP(SessionActivateHandler), 1, &event, this, &event_handler_); } return result == noErr; } bool CurtainModeMac::RemoveEventHandler() { OSStatus result = noErr; if (event_handler_) { result = ::RemoveEventHandler(event_handler_); event_handler_ = NULL; } return result == noErr; } // static scoped_ptr CurtainMode::Create( const base::Closure& on_session_activate, const base::Closure& on_error) { return scoped_ptr( new CurtainModeMac(on_session_activate, on_error)); } } // namespace remoting