LCOV - code coverage report
Current view: top level - vcl/unx/generic/app - saldata.cxx (source / functions) Hit Total Coverage
Test: commit c8344322a7af75b84dd3ca8f78b05543a976dfd5 Lines: 1 271 0.4 %
Date: 2015-06-13 12:38:46 Functions: 2 33 6.1 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <unistd.h>
      21             : #include <fcntl.h>
      22             : 
      23             : #include <cstdio>
      24             : #include <cstring>
      25             : #include <cstdlib>
      26             : #include <stdio.h>
      27             : #include <limits.h>
      28             : #include <errno.h>
      29             : #include <pthread.h>
      30             : #include <sys/resource.h>
      31             : #ifdef SUN
      32             : #include <sys/systeminfo.h>
      33             : #endif
      34             : #ifdef AIX
      35             : #include <strings.h>
      36             : #endif
      37             : #ifdef FREEBSD
      38             : #include <sys/types.h>
      39             : #include <sys/time.h>
      40             : #endif
      41             : 
      42             : #include <osl/process.h>
      43             : #include <osl/mutex.hxx>
      44             : 
      45             : #include "unx/saldisp.hxx"
      46             : #include "unx/saldata.hxx"
      47             : #include "unx/salframe.h"
      48             : #include "unx/sm.hxx"
      49             : #include "unx/i18n_im.hxx"
      50             : #include "unx/i18n_xkb.hxx"
      51             : #include "unx/x11/x11display.hxx"
      52             : #include "salinst.hxx"
      53             : 
      54             : #include <osl/signal.h>
      55             : #include <osl/thread.h>
      56             : #include <rtl/strbuf.hxx>
      57             : #include <rtl/bootstrap.hxx>
      58             : 
      59             : #include <tools/debug.hxx>
      60             : #include <vcl/svapp.hxx>
      61             : 
      62             : #ifndef UNX
      63             : #ifndef SIGBUS
      64             : #define SIGBUS 10
      65             : #endif
      66             : #ifndef SIGSEGV
      67             : #define SIGSEGV 11
      68             : #endif
      69             : #ifndef SIGIOT
      70             : #define SIGIOT SIGABRT
      71             : #endif
      72             : #endif
      73             : 
      74           0 : X11SalData* GetX11SalData()
      75             : {
      76           0 :     SalData * p1 = ImplGetSVData()->mpSalData;
      77             :     OSL_ASSERT(p1 != 0);
      78           0 :     X11SalData * p2 = dynamic_cast< X11SalData * >(p1);
      79             :     OSL_ASSERT(p2 != 0);
      80           0 :     return p2;
      81             : }
      82             : 
      83             : extern "C" {
      84             : 
      85           0 : static int XErrorHdl( Display *pDisplay, XErrorEvent *pEvent )
      86             : {
      87           0 :     GetX11SalData()->XError( pDisplay, pEvent );
      88           0 :     return 0;
      89             : }
      90             : 
      91           0 : static int XIOErrorHdl( Display * )
      92             : {
      93           0 :     if (::osl::Thread::getCurrentIdentifier() == Application::GetMainThreadIdentifier())
      94             :     {
      95             :         /*  #106197# hack: until a real shutdown procedure exists
      96             :          *  _exit ASAP
      97             :          */
      98           0 :         if( ImplGetSVData()->maAppData.mbAppQuit )
      99           0 :             _exit(1);
     100             : 
     101             :         // really bad hack
     102           0 :         if( ! SessionManagerClient::checkDocumentsSaved() )
     103           0 :             /* oslSignalAction eToDo = */ osl_raiseSignal (OSL_SIGNAL_USER_X11SUBSYSTEMERROR, NULL);
     104             :     }
     105             : 
     106           0 :     std::fprintf( stderr, "X IO Error\n" );
     107           0 :     std::fflush( stdout );
     108           0 :     std::fflush( stderr );
     109             : 
     110             :     /*  #106197# the same reasons to use _exit instead of exit in salmain
     111             :      *  do apply here. Since there is nothing to be done after an XIO
     112             :      *  error we have to _exit immediately.
     113             :      */
     114           0 :     _exit(1);
     115             :     return 0;
     116             : }
     117             : 
     118             : }
     119             : 
     120             : static const struct timeval noyield__ = { 0, 0 };
     121             : static const struct timeval yield__   = { 0, 10000 };
     122             : 
     123             : static const char* XRequest[] = {
     124             :     // see /usr/lib/X11/XErrorDB, /usr/openwin/lib/XErrorDB ...
     125             :     NULL,
     126             :     "X_CreateWindow",
     127             :     "X_ChangeWindowAttributes",
     128             :     "X_GetWindowAttributes",
     129             :     "X_DestroyWindow",
     130             :     "X_DestroySubwindows",
     131             :     "X_ChangeSaveSet",
     132             :     "X_ReparentWindow",
     133             :     "X_MapWindow",
     134             :     "X_MapSubwindows",
     135             :     "X_UnmapWindow",
     136             :     "X_UnmapSubwindows",
     137             :     "X_ConfigureWindow",
     138             :     "X_CirculateWindow",
     139             :     "X_GetGeometry",
     140             :     "X_QueryTree",
     141             :     "X_InternAtom",
     142             :     "X_GetAtomName",
     143             :     "X_ChangeProperty",
     144             :     "X_DeleteProperty",
     145             :     "X_GetProperty",
     146             :     "X_ListProperties",
     147             :     "X_SetSelectionOwner",
     148             :     "X_GetSelectionOwner",
     149             :     "X_ConvertSelection",
     150             :     "X_SendEvent",
     151             :     "X_GrabPointer",
     152             :     "X_UngrabPointer",
     153             :     "X_GrabButton",
     154             :     "X_UngrabButton",
     155             :     "X_ChangeActivePointerGrab",
     156             :     "X_GrabKeyboard",
     157             :     "X_UngrabKeyboard",
     158             :     "X_GrabKey",
     159             :     "X_UngrabKey",
     160             :     "X_AllowEvents",
     161             :     "X_GrabServer",
     162             :     "X_UngrabServer",
     163             :     "X_QueryPointer",
     164             :     "X_GetMotionEvents",
     165             :     "X_TranslateCoords",
     166             :     "X_WarpPointer",
     167             :     "X_SetInputFocus",
     168             :     "X_GetInputFocus",
     169             :     "X_QueryKeymap",
     170             :     "X_OpenFont",
     171             :     "X_CloseFont",
     172             :     "X_QueryFont",
     173             :     "X_QueryTextExtents",
     174             :     "X_ListFonts",
     175             :     "X_ListFontsWithInfo",
     176             :     "X_SetFontPath",
     177             :     "X_GetFontPath",
     178             :     "X_CreatePixmap",
     179             :     "X_FreePixmap",
     180             :     "X_CreateGC",
     181             :     "X_ChangeGC",
     182             :     "X_CopyGC",
     183             :     "X_SetDashes",
     184             :     "X_SetClipRectangles",
     185             :     "X_FreeGC",
     186             :     "X_ClearArea",
     187             :     "X_CopyArea",
     188             :     "X_CopyPlane",
     189             :     "X_PolyPoint",
     190             :     "X_PolyLine",
     191             :     "X_PolySegment",
     192             :     "X_PolyRectangle",
     193             :     "X_PolyArc",
     194             :     "X_FillPoly",
     195             :     "X_PolyFillRectangle",
     196             :     "X_PolyFillArc",
     197             :     "X_PutImage",
     198             :     "X_GetImage",
     199             :     "X_PolyText8",
     200             :     "X_PolyText16",
     201             :     "X_ImageText8",
     202             :     "X_ImageText16",
     203             :     "X_CreateColormap",
     204             :     "X_FreeColormap",
     205             :     "X_CopyColormapAndFree",
     206             :     "X_InstallColormap",
     207             :     "X_UninstallColormap",
     208             :     "X_ListInstalledColormaps",
     209             :     "X_AllocColor",
     210             :     "X_AllocNamedColor",
     211             :     "X_AllocColorCells",
     212             :     "X_AllocColorPlanes",
     213             :     "X_FreeColors",
     214             :     "X_StoreColors",
     215             :     "X_StoreNamedColor",
     216             :     "X_QueryColors",
     217             :     "X_LookupColor",
     218             :     "X_CreateCursor",
     219             :     "X_CreateGlyphCursor",
     220             :     "X_FreeCursor",
     221             :     "X_RecolorCursor",
     222             :     "X_QueryBestSize",
     223             :     "X_QueryExtension",
     224             :     "X_ListExtensions",
     225             :     "X_ChangeKeyboardMapping",
     226             :     "X_GetKeyboardMapping",
     227             :     "X_ChangeKeyboardControl",
     228             :     "X_GetKeyboardControl",
     229             :     "X_Bell",
     230             :     "X_ChangePointerControl",
     231             :     "X_GetPointerControl",
     232             :     "X_SetScreenSaver",
     233             :     "X_GetScreenSaver",
     234             :     "X_ChangeHosts",
     235             :     "X_ListHosts",
     236             :     "X_SetAccessControl",
     237             :     "X_SetCloseDownMode",
     238             :     "X_KillClient",
     239             :     "X_RotateProperties",
     240             :     "X_ForceScreenSaver",
     241             :     "X_SetPointerMapping",
     242             :     "X_GetPointerMapping",
     243             :     "X_SetModifierMapping",
     244             :     "X_GetModifierMapping",
     245             :     NULL,
     246             :     NULL,
     247             :     NULL,
     248             :     NULL,
     249             :     NULL,
     250             :     NULL,
     251             :     NULL,
     252             :     "X_NoOperation"
     253             : };
     254             : 
     255           0 : X11SalData::X11SalData( SalGenericDataType t, SalInstance *pInstance )
     256           0 :     : SalGenericData( t, pInstance )
     257             : {
     258           0 :     pXLib_          = NULL;
     259           0 :     m_pPlugin       = NULL;
     260             : 
     261           0 :     m_aOrigXIOErrorHandler = XSetIOErrorHandler ( XIOErrorHdl );
     262           0 :     PushXErrorLevel( !!getenv( "SAL_IGNOREXERRORS" ) );
     263           0 : }
     264             : 
     265           0 : X11SalData::~X11SalData()
     266             : {
     267           0 :     DeleteDisplay();
     268           0 :     PopXErrorLevel();
     269           0 :     XSetIOErrorHandler (m_aOrigXIOErrorHandler);
     270           0 : }
     271             : 
     272           0 : void X11SalData::Dispose()
     273             : {
     274           0 :     deInitNWF();
     275           0 :     delete GetDisplay();
     276           0 :     SetSalData( NULL );
     277           0 : }
     278             : 
     279           0 : void X11SalData::DeleteDisplay()
     280             : {
     281           0 :     delete GetDisplay();
     282           0 :     SetDisplay( NULL );
     283           0 :     delete pXLib_;
     284           0 :     pXLib_ = NULL;
     285           0 : }
     286             : 
     287           0 : void X11SalData::Init()
     288             : {
     289           0 :     pXLib_ = new SalXLib();
     290           0 :     pXLib_->Init();
     291           0 : }
     292             : 
     293           0 : void X11SalData::initNWF()
     294             : {
     295           0 : }
     296             : 
     297           0 : void X11SalData::deInitNWF()
     298             : {
     299           0 : }
     300             : 
     301           0 : void X11SalData::ErrorTrapPush()
     302             : {
     303           0 :     PushXErrorLevel( true );
     304           0 : }
     305             : 
     306           0 : bool X11SalData::ErrorTrapPop( bool bIgnoreError )
     307             : {
     308           0 :     bool err = false;
     309           0 :     if( !bIgnoreError )
     310           0 :         err = HasXErrorOccurred();
     311           0 :     ResetXErrorOccurred();
     312           0 :     PopXErrorLevel();
     313           0 :     return err;
     314             : }
     315             : 
     316           0 : void X11SalData::PushXErrorLevel( bool bIgnore )
     317             : {
     318           0 :     m_aXErrorHandlerStack.push_back( XErrorStackEntry() );
     319           0 :     XErrorStackEntry& rEnt = m_aXErrorHandlerStack.back();
     320           0 :     rEnt.m_bWas = false;
     321           0 :     rEnt.m_bIgnore = bIgnore;
     322           0 :     rEnt.m_nLastErrorRequest = 0;
     323           0 :     rEnt.m_aHandler = XSetErrorHandler( XErrorHdl );
     324           0 : }
     325             : 
     326           0 : void X11SalData::PopXErrorLevel()
     327             : {
     328           0 :     if( m_aXErrorHandlerStack.size() )
     329             :     {
     330           0 :         XSetErrorHandler( m_aXErrorHandlerStack.back().m_aHandler );
     331           0 :         m_aXErrorHandlerStack.pop_back();
     332             :     }
     333           0 : }
     334             : 
     335           0 : SalXLib::SalXLib()
     336           0 :     : blockIdleTimeout( false )
     337             : {
     338           0 :     m_aTimeout.tv_sec       = 0;
     339           0 :     m_aTimeout.tv_usec      = 0;
     340           0 :     m_nTimeoutMS            = 0;
     341             : 
     342           0 :     nFDs_                   = 0;
     343           0 :     FD_ZERO( &aReadFDS_ );
     344           0 :     FD_ZERO( &aExceptionFDS_ );
     345             : 
     346           0 :     m_pTimeoutFDS[0] = m_pTimeoutFDS[1] = -1;
     347           0 :     if (pipe (m_pTimeoutFDS) != -1)
     348             :     {
     349             :         // initialize 'wakeup' pipe.
     350             :         int flags;
     351             : 
     352             :         // set close-on-exec descriptor flag.
     353           0 :         if ((flags = fcntl (m_pTimeoutFDS[0], F_GETFD)) != -1)
     354             :         {
     355           0 :             flags |= FD_CLOEXEC;
     356           0 :             (void)fcntl(m_pTimeoutFDS[0], F_SETFD, flags);
     357             :         }
     358           0 :         if ((flags = fcntl (m_pTimeoutFDS[1], F_GETFD)) != -1)
     359             :         {
     360           0 :             flags |= FD_CLOEXEC;
     361           0 :             (void)fcntl(m_pTimeoutFDS[1], F_SETFD, flags);
     362             :         }
     363             : 
     364             :         // set non-blocking I/O flag.
     365           0 :         if ((flags = fcntl (m_pTimeoutFDS[0], F_GETFL)) != -1)
     366             :         {
     367           0 :             flags |= O_NONBLOCK;
     368           0 :             (void)fcntl(m_pTimeoutFDS[0], F_SETFL, flags);
     369             :         }
     370           0 :         if ((flags = fcntl (m_pTimeoutFDS[1], F_GETFL)) != -1)
     371             :         {
     372           0 :             flags |= O_NONBLOCK;
     373           0 :             (void)fcntl(m_pTimeoutFDS[1], F_SETFL, flags);
     374             :         }
     375             : 
     376             :         // insert [0] into read descriptor set.
     377           0 :         FD_SET( m_pTimeoutFDS[0], &aReadFDS_ );
     378           0 :         nFDs_ = m_pTimeoutFDS[0] + 1;
     379             :     }
     380           0 : }
     381             : 
     382           0 : SalXLib::~SalXLib()
     383             : {
     384             :     // close 'wakeup' pipe.
     385           0 :     close (m_pTimeoutFDS[0]);
     386           0 :     close (m_pTimeoutFDS[1]);
     387           0 : }
     388             : 
     389           0 : void SalXLib::Init()
     390             : {
     391           0 :     SalI18N_InputMethod* pInputMethod = new SalI18N_InputMethod;
     392           0 :     pInputMethod->SetLocale();
     393           0 :     XrmInitialize();
     394             : 
     395           0 :     OString aDisplay;
     396           0 :     Display *pDisp = OpenX11Display(aDisplay);
     397             : 
     398           0 :     if ( !pDisp )
     399             :     {
     400           0 :         OUString aProgramFileURL;
     401           0 :         osl_getExecutableFile( &aProgramFileURL.pData );
     402           0 :         OUString aProgramSystemPath;
     403           0 :         osl_getSystemPathFromFileURL (aProgramFileURL.pData, &aProgramSystemPath.pData);
     404             :         OString  aProgramName = OUStringToOString(
     405             :                                             aProgramSystemPath,
     406           0 :                                             osl_getThreadTextEncoding() );
     407             :         std::fprintf( stderr, "%s X11 error: Can't open display: %s\n",
     408           0 :                 aProgramName.getStr(), aDisplay.getStr());
     409           0 :         std::fprintf( stderr, "   Set DISPLAY environment variable, use -display option\n");
     410           0 :         std::fprintf( stderr, "   or check permissions of your X-Server\n");
     411           0 :         std::fprintf( stderr, "   (See \"man X\" resp. \"man xhost\" for details)\n");
     412           0 :         std::fflush( stderr );
     413           0 :         exit(0);
     414             :     }
     415             : 
     416           0 :     SalX11Display *pSalDisplay = new SalX11Display( pDisp );
     417             : 
     418           0 :     pInputMethod->CreateMethod( pDisp );
     419           0 :     pSalDisplay->SetupInput( pInputMethod );
     420           0 : }
     421             : 
     422             : extern "C" {
     423           0 : void EmitFontpathWarning()
     424             : {
     425             :     static Bool bOnce = False;
     426           0 :     if ( !bOnce )
     427             :     {
     428           0 :         bOnce = True;
     429             :         std::fprintf( stderr, "Please verify your fontpath settings\n"
     430             :                 "\t(See \"man xset\" for details"
     431           0 :                 " or ask your system administrator)\n" );
     432             :     }
     433           0 : }
     434             : 
     435             : } /* extern "C" */
     436             : 
     437           0 : static void PrintXError( Display *pDisplay, XErrorEvent *pEvent )
     438             : {
     439           0 :     char msg[ 120 ] = "";
     440             : #if ! ( defined LINUX && defined PPC )
     441           0 :     XGetErrorText( pDisplay, pEvent->error_code, msg, sizeof( msg ) );
     442             : #endif
     443           0 :     std::fprintf( stderr, "X-Error: %s\n", msg );
     444           0 :     if( pEvent->request_code < SAL_N_ELEMENTS( XRequest ) )
     445             :     {
     446           0 :         const char* pName = XRequest[pEvent->request_code];
     447           0 :         if( !pName )
     448           0 :             pName = "BadRequest?";
     449           0 :         std::fprintf( stderr, "\tMajor opcode: %d (%s)\n", pEvent->request_code, pName );
     450             :     }
     451             :     else
     452             :     {
     453           0 :         std::fprintf( stderr, "\tMajor opcode: %d\n", pEvent->request_code );
     454             :         // TODO: also display extension name?
     455           0 :         std::fprintf( stderr, "\tMinor opcode: %d\n", pEvent->minor_code );
     456             :     }
     457             : 
     458             :     std::fprintf( stderr, "\tResource ID:  0x%lx\n",
     459           0 :              pEvent->resourceid );
     460             :     std::fprintf( stderr, "\tSerial No:    %ld (%ld)\n",
     461           0 :              pEvent->serial, LastKnownRequestProcessed(pDisplay) );
     462             : 
     463           0 :     if( !getenv( "SAL_SYNCHRONIZE" ) )
     464             :     {
     465           0 :         std::fprintf( stderr, "These errors are reported asynchronously,\n");
     466           0 :         std::fprintf( stderr, "set environment variable SAL_SYNCHRONIZE to 1 to help debugging\n");
     467             :     }
     468             : 
     469           0 :     std::fflush( stdout );
     470           0 :     std::fflush( stderr );
     471           0 : }
     472             : 
     473           0 : void X11SalData::XError( Display *pDisplay, XErrorEvent *pEvent )
     474             : {
     475           0 :     if( ! m_aXErrorHandlerStack.back().m_bIgnore )
     476             :     {
     477           0 :         if (   (pEvent->error_code   == BadAlloc)
     478           0 :             && (pEvent->request_code == X_OpenFont) )
     479             :         {
     480             :             static Bool bOnce = False;
     481           0 :             if ( !bOnce )
     482             :             {
     483           0 :                 std::fprintf(stderr, "X-Error occurred in a request for X_OpenFont\n");
     484           0 :                 EmitFontpathWarning();
     485             : 
     486           0 :                 bOnce = True ;
     487             :             }
     488           0 :             return;
     489             :         }
     490             :         /* ignore
     491             :         * X_SetInputFocus: it's a hint only anyway
     492             :         * X_GetProperty: this is part of the XGetWindowProperty call and will
     493             :         *                be handled by the return value of that function
     494             :         */
     495           0 :         else if( pEvent->request_code == X_SetInputFocus ||
     496           0 :                  pEvent->request_code == X_GetProperty
     497             :             )
     498           0 :             return;
     499             : 
     500           0 :         if( pDisplay != vcl_sal::getSalDisplay(GetGenericData())->GetDisplay() )
     501           0 :             return;
     502             : 
     503           0 :         PrintXError( pDisplay, pEvent );
     504             : 
     505           0 :         oslSignalAction eToDo = osl_raiseSignal (OSL_SIGNAL_USER_X11SUBSYSTEMERROR, NULL);
     506           0 :         switch (eToDo)
     507             :         {
     508             :             case osl_Signal_ActIgnore       :
     509           0 :                 return;
     510             :             case osl_Signal_ActAbortApp     :
     511           0 :                 abort();
     512             :             case osl_Signal_ActKillApp      :
     513           0 :                 exit(0);
     514             :             case osl_Signal_ActCallNextHdl  :
     515           0 :                 break;
     516             :             default :
     517           0 :                 break;
     518             :         }
     519             : 
     520             :     }
     521             : 
     522           0 :     m_aXErrorHandlerStack.back().m_bWas = true;
     523             : }
     524             : 
     525             : struct YieldEntry
     526             : {
     527             :     YieldEntry* next;       // pointer to next entry
     528             :     int         fd;         // file descriptor for reading
     529             :     void*           data;       // data for predicate and callback
     530             :     YieldFunc       pending;    // predicate (determins pending events)
     531             :     YieldFunc       queued;     // read and queue up events
     532             :     YieldFunc       handle;     // handle pending events
     533             : 
     534           0 :     inline int  HasPendingEvent()   const { return pending( fd, data ); }
     535           0 :     inline int  IsEventQueued()     const { return queued( fd, data ); }
     536           0 :     inline void HandleNextEvent()   const { handle( fd, data ); }
     537             : };
     538             : 
     539             : #define MAX_NUM_DESCRIPTORS 128
     540             : 
     541             : static YieldEntry yieldTable[ MAX_NUM_DESCRIPTORS ];
     542             : 
     543           0 : void SalXLib::Insert( int nFD, void* data,
     544             :                       YieldFunc     pending,
     545             :                       YieldFunc     queued,
     546             :                       YieldFunc     handle )
     547             : {
     548             :     DBG_ASSERT( nFD, "can not insert stdin descriptor" );
     549             :     DBG_ASSERT( !yieldTable[nFD].fd, "SalXLib::Insert fd twice" );
     550             : 
     551           0 :     yieldTable[nFD].fd      = nFD;
     552           0 :     yieldTable[nFD].data    = data;
     553           0 :     yieldTable[nFD].pending = pending;
     554           0 :     yieldTable[nFD].queued  = queued;
     555           0 :     yieldTable[nFD].handle  = handle;
     556             : 
     557           0 :     FD_SET( nFD, &aReadFDS_ );
     558           0 :     FD_SET( nFD, &aExceptionFDS_ );
     559             : 
     560           0 :     if( nFD >= nFDs_ )
     561           0 :         nFDs_ = nFD + 1;
     562           0 : }
     563             : 
     564           0 : void SalXLib::Remove( int nFD )
     565             : {
     566           0 :     FD_CLR( nFD, &aReadFDS_ );
     567           0 :     FD_CLR( nFD, &aExceptionFDS_ );
     568             : 
     569           0 :     yieldTable[nFD].fd = 0;
     570             : 
     571           0 :     if ( nFD == nFDs_ )
     572             :     {
     573           0 :         for ( nFD = nFDs_ - 1;
     574           0 :               nFD >= 0 && !yieldTable[nFD].fd;
     575             :               nFD-- ) ;
     576             : 
     577           0 :         nFDs_ = nFD + 1;
     578             :     }
     579           0 : }
     580             : 
     581           0 : bool SalXLib::CheckTimeout( bool bExecuteTimers )
     582             : {
     583           0 :     bool bRet = false;
     584           0 :     if( m_aTimeout.tv_sec ) // timer is started
     585             :     {
     586             :         timeval aTimeOfDay;
     587           0 :         gettimeofday( &aTimeOfDay, 0 );
     588           0 :         if( aTimeOfDay >= m_aTimeout )
     589             :         {
     590           0 :             bRet = true;
     591           0 :             if( bExecuteTimers )
     592             :             {
     593             :                 // timed out, update timeout
     594           0 :                 m_aTimeout = aTimeOfDay;
     595             :                 /*
     596             :                 *  #107827# autorestart immediately, will be stopped (or set
     597             :                 *  to different value in notify hdl if necessary;
     598             :                 *  CheckTimeout should return false while
     599             :                 *  timers are being dispatched.
     600             :                 */
     601           0 :                 m_aTimeout += m_nTimeoutMS;
     602             :                 // Determine if the app is idle (for idle timers). If there's user input pending,
     603             :                 // if there's IO pending or if we're called inside a temporary yield (=blockIdleTimeout),
     604             :                 // then the app is not idle.
     605           0 :                 bool idle = true;
     606           0 :                 if( blockIdleTimeout || XPending( vcl_sal::getSalDisplay(GetGenericData())->GetDisplay()))
     607           0 :                     idle = false;
     608           0 :                 for ( int nFD = 0; idle && nFD < nFDs_; nFD++ )
     609             :                 {
     610           0 :                     YieldEntry* pEntry = &(yieldTable[nFD]);
     611           0 :                     if ( pEntry->fd && pEntry->HasPendingEvent())
     612           0 :                         idle = false;
     613             :                 }
     614             :                 // notify
     615           0 :                 X11SalData::Timeout( idle );
     616             :             }
     617             :         }
     618             :     }
     619           0 :     return bRet;
     620             : }
     621             : 
     622           0 : void SalXLib::Yield( bool bWait, bool bHandleAllCurrentEvents )
     623             : {
     624           0 :     blockIdleTimeout = !bWait;
     625             :     // check for timeouts here if you want to make screenshots
     626           0 :     static char* p_prioritize_timer = getenv ("SAL_HIGHPRIORITY_REPAINT");
     627           0 :     if (p_prioritize_timer != NULL)
     628           0 :         CheckTimeout();
     629             : 
     630           0 :     const int nMaxEvents = bHandleAllCurrentEvents ? 100 : 1;
     631             : 
     632             :     // first, check for already queued events.
     633           0 :     for ( int nFD = 0; nFD < nFDs_; nFD++ )
     634             :     {
     635           0 :         YieldEntry* pEntry = &(yieldTable[nFD]);
     636           0 :         if ( pEntry->fd )
     637             :         {
     638             :             DBG_ASSERT( nFD == pEntry->fd, "wrong fd in Yield()" );
     639           0 :             for( int i = 0; i < nMaxEvents && pEntry->HasPendingEvent(); i++ )
     640             :             {
     641           0 :                 pEntry->HandleNextEvent();
     642           0 :                 if( ! bHandleAllCurrentEvents )
     643             :                 {
     644           0 :                     blockIdleTimeout = false;
     645           0 :                     return;
     646             :                 }
     647             :             }
     648             :         }
     649             :     }
     650             : 
     651             :     // next, select with or without timeout according to bWait.
     652           0 :     int      nFDs         = nFDs_;
     653           0 :     fd_set   ReadFDS      = aReadFDS_;
     654           0 :     fd_set   ExceptionFDS = aExceptionFDS_;
     655           0 :     int      nFound       = 0;
     656             : 
     657           0 :     timeval  Timeout      = noyield__;
     658           0 :     timeval *pTimeout     = &Timeout;
     659             : 
     660           0 :     if (bWait)
     661             :     {
     662           0 :         pTimeout = 0;
     663           0 :         if (m_aTimeout.tv_sec) // Timer is started.
     664             :         {
     665             :             // determine remaining timeout.
     666           0 :             gettimeofday (&Timeout, 0);
     667           0 :             Timeout = m_aTimeout - Timeout;
     668           0 :             if (yield__ >= Timeout)
     669             :             {
     670             :                 // guard against micro timeout.
     671           0 :                 Timeout = yield__;
     672             :             }
     673           0 :             pTimeout = &Timeout;
     674             :         }
     675             :     }
     676             : 
     677             :     {
     678             :         // release YieldMutex (and re-acquire at block end)
     679           0 :         SalYieldMutexReleaser aReleaser;
     680           0 :         nFound = select( nFDs, &ReadFDS, NULL, &ExceptionFDS, pTimeout );
     681             :     }
     682           0 :     if( nFound < 0 ) // error
     683             :     {
     684             : #ifdef DBG_UTIL
     685             :         std::fprintf( stderr, "SalXLib::Yield e=%d f=%d\n", errno, nFound );
     686             : #endif
     687           0 :         if( EINTR == errno )
     688             :         {
     689           0 :             errno = 0;
     690             :         }
     691             :     }
     692             : 
     693             :     // usually handle timeouts here (as in 5.2)
     694           0 :     if (p_prioritize_timer == NULL)
     695           0 :         CheckTimeout();
     696             : 
     697             :     // handle wakeup events.
     698           0 :     if ((nFound > 0) && (FD_ISSET(m_pTimeoutFDS[0], &ReadFDS)))
     699             :     {
     700             :         int buffer;
     701           0 :         while (read (m_pTimeoutFDS[0], &buffer, sizeof(buffer)) > 0)
     702           0 :             continue;
     703           0 :         nFound -= 1;
     704             :     }
     705             : 
     706             :     // handle other events.
     707           0 :     if( nFound > 0 )
     708             :     {
     709             :         // now we are in the protected section !
     710             :         // recall select if we have acquired fd's, ready for reading,
     711             : 
     712           0 :         struct timeval noTimeout = { 0, 0 };
     713             :         nFound = select( nFDs_, &ReadFDS, NULL,
     714           0 :                          &ExceptionFDS, &noTimeout );
     715             : 
     716             :         // someone-else has done the job for us
     717           0 :         if (nFound == 0)
     718             :         {
     719           0 :             blockIdleTimeout = false;
     720           0 :             return;
     721             :         }
     722             : 
     723           0 :         for ( int nFD = 0; nFD < nFDs_; nFD++ )
     724             :         {
     725           0 :             YieldEntry* pEntry = &(yieldTable[nFD]);
     726           0 :             if ( pEntry->fd )
     727             :             {
     728           0 :                 if ( FD_ISSET( nFD, &ExceptionFDS ) ) {
     729             : #if OSL_DEBUG_LEVEL > 1
     730             :                     std::fprintf( stderr, "SalXLib::Yield exception\n" );
     731             : #endif
     732           0 :                     nFound--;
     733             :                 }
     734           0 :                 if ( FD_ISSET( nFD, &ReadFDS ) )
     735             :                 {
     736           0 :                     for( int i = 0; pEntry->IsEventQueued() && i < nMaxEvents; i++ )
     737             :                     {
     738           0 :                         pEntry->HandleNextEvent();
     739             :                         // if a recursive call has done the job
     740             :                         // so abort here
     741             :                     }
     742           0 :                     nFound--;
     743             :                 }
     744             :             }
     745             :         }
     746             :     }
     747           0 :     blockIdleTimeout = false;
     748             : }
     749             : 
     750           0 : void SalXLib::Wakeup()
     751             : {
     752           0 :     OSL_VERIFY(write (m_pTimeoutFDS[1], "", 1) == 1);
     753           0 : }
     754             : 
     755           0 : void SalXLib::PostUserEvent()
     756             : {
     757           0 :     Wakeup();
     758           9 : }
     759             : 
     760             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.11