/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* vim:expandtab:shiftwidth=4:tabstop=4: */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Christopher Blizzard * . Portions created by the Initial Developer * are Copyright (C) 2001 the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include #include #include #include #include #include "nsAppShell.h" #include "nsWindow.h" #include "prlog.h" #include "prenv.h" #include "mozilla/HangMonitor.h" #define NOTIFY_TOKEN 0xFA #ifdef PR_LOGGING PRLogModuleInfo *gWidgetLog = nsnull; PRLogModuleInfo *gWidgetFocusLog = nsnull; PRLogModuleInfo *gWidgetDragLog = nsnull; PRLogModuleInfo *gWidgetDrawLog = nsnull; #endif static GPollFunc sPollFunc; // Wrapper function to disable hang monitoring while waiting in poll(). static gint PollWrapper(GPollFD *ufds, guint nfsd, gint timeout_) { mozilla::HangMonitor::Suspend(); gint result = (*sPollFunc)(ufds, nfsd, timeout_); mozilla::HangMonitor::NotifyActivity(); return result; } /*static*/ gboolean nsAppShell::EventProcessorCallback(GIOChannel *source, GIOCondition condition, gpointer data) { nsAppShell *self = static_cast(data); unsigned char c; read(self->mPipeFDs[0], &c, 1); NS_ASSERTION(c == (unsigned char) NOTIFY_TOKEN, "wrong token"); self->NativeEventCallback(); return TRUE; } nsAppShell::~nsAppShell() { if (mTag) g_source_remove(mTag); if (mPipeFDs[0]) close(mPipeFDs[0]); if (mPipeFDs[1]) close(mPipeFDs[1]); } nsresult nsAppShell::Init() { #ifdef PR_LOGGING if (!gWidgetLog) gWidgetLog = PR_NewLogModule("Widget"); if (!gWidgetFocusLog) gWidgetFocusLog = PR_NewLogModule("WidgetFocus"); if (!gWidgetDragLog) gWidgetDragLog = PR_NewLogModule("WidgetDrag"); if (!gWidgetDrawLog) gWidgetDrawLog = PR_NewLogModule("WidgetDraw"); #endif if (!sPollFunc) { sPollFunc = g_main_context_get_poll_func(NULL); g_main_context_set_poll_func(NULL, &PollWrapper); } GIOChannel *ioc; if (PR_GetEnv("MOZ_DEBUG_PAINTS")) gdk_window_set_debug_updates(TRUE); int err = pipe(mPipeFDs); if (err) return NS_ERROR_OUT_OF_MEMORY; // make the pipe nonblocking int flags = fcntl(mPipeFDs[0], F_GETFL, 0); if (flags == -1) goto failed; err = fcntl(mPipeFDs[0], F_SETFL, flags | O_NONBLOCK); if (err == -1) goto failed; flags = fcntl(mPipeFDs[1], F_GETFL, 0); if (flags == -1) goto failed; err = fcntl(mPipeFDs[1], F_SETFL, flags | O_NONBLOCK); if (err == -1) goto failed; ioc = g_io_channel_unix_new(mPipeFDs[0]); mTag = g_io_add_watch_full(ioc, G_PRIORITY_DEFAULT, G_IO_IN, EventProcessorCallback, this, nsnull); g_io_channel_unref(ioc); return nsBaseAppShell::Init(); failed: close(mPipeFDs[0]); close(mPipeFDs[1]); mPipeFDs[0] = mPipeFDs[1] = 0; return NS_ERROR_FAILURE; } void nsAppShell::ScheduleNativeEventCallback() { unsigned char buf[] = { NOTIFY_TOKEN }; write(mPipeFDs[1], buf, 1); } bool nsAppShell::ProcessNextNativeEvent(bool mayWait) { return g_main_context_iteration(NULL, mayWait); }