/* Greeter module for xdm Copyright (C) 1997, 1998 Steffen Hansen <hansen@kde.org> Copyright (C) 2000-2003 Oswald Buddenhagen <ossi@kde.org> This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <config.h> #include "tdm_greet.h" #include "tdmshutdown.h" #include "tdmconfig.h" #include "kgapp.h" #include "kgreeter.h" #ifdef XDMCP # include "kchooser.h" #endif #include "sakdlg.h" #include <kprocess.h> #include <tdecmdlineargs.h> #include <kcrash.h> #include <kstandarddirs.h> #include <ksimpleconfig.h> #include <tdelocale.h> #include <kdebug.h> #ifdef WITH_XRANDR #include <libtderandr/libtderandr.h> #endif #include <tqtimer.h> #include <tqstring.h> #include <tqcursor.h> #include <tqpalette.h> #include <stdlib.h> // free(), exit() #include <unistd.h> // alarm() #include <X11/Xlib.h> #include <X11/Xutil.h> #include <X11/keysym.h> #include <X11/cursorfont.h> #ifdef HAVE_XCOMPOSITE #include <X11/extensions/Xrender.h> #include <X11/extensions/Xcomposite.h> #endif #include <pwd.h> #define TSAK_FIFO_FILE "/tmp/tdesocket-global/tsak" bool argb_visual_available = false; bool has_twin = false; bool is_themed = false; bool trinity_desktop_lock_use_sak = TRUE; bool trinity_desktop_synchronize_keyboard_lights = TRUE; TQPoint primaryScreenPosition; static int ignoreXError( Display *dpy ATTR_UNUSED, XErrorEvent *event ATTR_UNUSED ) { return 0; } extern "C" { static void sigAlarm( int ) { exit( EX_RESERVER_DPY ); } } GreeterApp::GreeterApp() { init(); } GreeterApp::GreeterApp(Display *dpy) : TDEApplication(dpy) { init(); } GreeterApp::GreeterApp(Display *dpy, Qt::HANDLE visual, Qt::HANDLE colormap) : TDEApplication(dpy, visual, colormap) { init(); } GreeterApp::~GreeterApp() { // } void GreeterApp::init() { pingInterval = _isLocal ? 0 : _pingInterval; if (pingInterval) { struct sigaction sa; sigemptyset( &sa.sa_mask ); sa.sa_flags = 0; sa.sa_handler = sigAlarm; sigaction( SIGALRM, &sa, 0 ); alarm( pingInterval * 70 ); // sic! give the "proper" pinger enough time startTimer( pingInterval * 60000 ); } TDEHardwareDevices *hwdevices = TDEGlobal::hardwareDevices(); connect(hwdevices, TQT_SIGNAL(hardwareUpdated(TDEGenericDevice*)), this, TQT_SLOT(deviceChanged(TDEGenericDevice*))); } void GreeterApp::deviceChanged(TDEGenericDevice* device) { #ifdef WITH_XRANDR if (device->type() == TDEGenericDeviceType::Monitor) { KRandrSimpleAPI *randrsimple = new KRandrSimpleAPI(); randrsimple->applyHotplugRules(KDE_CONFDIR); delete randrsimple; } #endif // WITH_XRANDR } void GreeterApp::timerEvent( TQTimerEvent * ) { alarm( 0 ); if (!PingServer( tqt_xdisplay() )) ::exit( EX_RESERVER_DPY ); alarm( pingInterval * 70 ); // sic! give the "proper" pinger enough time } bool GreeterApp::x11EventFilter( XEvent * ev ) { KeySym sym; switch (ev->type) { case FocusIn: case FocusOut: // Hack to tell dialogs to take focus when the keyboard is grabbed ev->xfocus.mode = NotifyNormal; break; case KeyPress: sym = XLookupKeysym( &ev->xkey, 0 ); if (sym != XK_Return && !IsModifierKey( sym )) emit activity(); break; case ButtonPress: emit activity(); /* fall through */ case ButtonRelease: // Hack to let the RMB work as LMB if (ev->xbutton.button == 3) ev->xbutton.button = 1; /* fall through */ case MotionNotify: if (ev->xbutton.state & Button3Mask) ev->xbutton.state = (ev->xbutton.state & ~Button3Mask) | Button1Mask; break; } return false; } extern bool kde_have_kipc; extern "C" { static int xIOErr( Display * ) { exit( EX_RESERVER_DPY ); } //KSimpleConfig *iccconfig; void checkSAK(GreeterApp* app) { app->restoreOverrideCursor(); SAKDlg sak(0); sak.exec(); app->setOverrideCursor( Qt::WaitCursor ); } void kg_main( const char *argv0 ) { static char *argv[] = { (char *)"tdmgreet", 0 }; TDECmdLineArgs::init( 1, argv, *argv, 0, 0, 0, true ); kdDebug() << "[tdm-kfrontend] " << timestamp() << "start" << endl; kde_have_kipc = false; TDEApplication::disableAutoDcopRegistration(); TDECrash::setSafer( true ); TDEProcess *tsak = 0; TDEProcess *kbdl = 0; TDEProcess *proc = 0; TDEProcess *comp = 0; TDEProcess *dcop = 0; TDEProcess *twin = 0; #ifdef BUILD_TSAK trinity_desktop_lock_use_sak = _useSAK; #else trinity_desktop_lock_use_sak = false; #endif if (trinity_desktop_lock_use_sak) { if (system(KDE_BINDIR "/tsak checkdeps") != 0) { trinity_desktop_lock_use_sak = false; } } if (trinity_desktop_lock_use_sak) { tsak = new TDEProcess; *tsak << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + "tsak"; tsak->start(TDEProcess::Block, TDEProcess::AllOutput); } else { remove(TSAK_FIFO_FILE); } if (tsak) { tsak->closeStdin(); tsak->closeStdout(); tsak->detach(); delete tsak; } if (trinity_desktop_synchronize_keyboard_lights) { kbdl = new TDEProcess; *kbdl << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + "tdekbdledsync"; kbdl->start(); } XSetErrorHandler( ignoreXError ); argb_visual_available = false; char *display = 0; Display *dpyi = XOpenDisplay( display ); if ( !dpyi ) { kdError() << "cannot connect to X server " << display << endl; exit( 1 ); } #ifdef HAVE_XCOMPOSITE // Begin ARGB initialization int screen = DefaultScreen( dpyi ); Colormap colormap = 0; Visual *visual = 0; int event_base, error_base; if ( XRenderQueryExtension( dpyi, &event_base, &error_base ) ) { int nvi; XVisualInfo templ; templ.screen = screen; templ.depth = 32; templ.c_class = TrueColor; XVisualInfo *xvi = XGetVisualInfo( dpyi, VisualScreenMask | VisualDepthMask | VisualClassMask, &templ, &nvi ); for ( int i = 0; i < nvi; i++ ) { XRenderPictFormat *format = XRenderFindVisualFormat( dpyi, xvi[i].visual ); if ( format->type == PictTypeDirect && format->direct.alphaMask ) { visual = xvi[i].visual; colormap = XCreateColormap( dpyi, RootWindow( dpyi, screen ), visual, AllocNone ); kdDebug() << "[tdm-kfrontend] Found visual with alpha support" << endl; argb_visual_available = true; break; } } } XSync( dpyi, False ); XSetErrorHandler( (XErrorHandler)0 ); GreeterApp *app; if ((!_compositor.isEmpty()) && ( argb_visual_available == true )) { app = new GreeterApp(dpyi, Qt::HANDLE( visual ), Qt::HANDLE( colormap )); } else { argb_visual_available = false; app = new GreeterApp(dpyi); } // End ARGB initialization #else GreeterApp *app = new GreeterApp(dpyi); #endif // Load up systemwide display settings #ifdef WITH_XRANDR KRandrSimpleAPI *randrsimple = new KRandrSimpleAPI(); primaryScreenPosition = randrsimple->applyStartupDisplayConfiguration(KDE_CONFDIR); randrsimple->applyHotplugRules(KDE_CONFDIR); delete randrsimple; #endif // Load up the systemwide ICC profile TQString iccConfigFile = TQString(KDE_CONFDIR); iccConfigFile += "/kicc/kiccconfigrc"; KSimpleConfig iccconfig(iccConfigFile, true); if (iccconfig.readBoolEntry("EnableICC", false) == true) { TQString iccCommand = TQString("/usr/bin/xcalib "); iccCommand += iccconfig.readEntry("ICCFile"); iccCommand += TQString(" &"); if (system(iccCommand.ascii()) < 0) { printf("WARNING: Unable to execute command \"%s\"\n", iccCommand.ascii()); } } // Make sure TQt is aware of the screen geometry changes before any dialogs are created XSync(tqt_xdisplay(), false); app->processEvents(); TQRect screenRect = TQApplication::desktop()->screenGeometry(); TQCursor::setPos(screenRect.center().x(), screenRect.center().y()); XSetIOErrorHandler( xIOErr ); TQString login_user; TQString login_session_wm; Display *dpy = tqt_xdisplay(); if (!_GUIStyle.isEmpty()) { app->setStyle( _GUIStyle ); } _colorScheme = locate( "data", "tdedisplay/color-schemes/" + _colorScheme + ".kcsrc" ); if (!_colorScheme.isEmpty()) { KSimpleConfig config( _colorScheme, true ); config.setGroup( "Color Scheme" ); app->setPalette( app->createApplicationPalette( &config, 7 ) ); } app->setFont( _normalFont ); setup_modifiers( dpy, _numLockStatus ); SecureDisplay( dpy ); if (!_grabServer) { if (_useBackground) { proc = new TDEProcess; *proc << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + "krootimage"; *proc << _backgroundCfg; proc->start(); } GSendInt( G_SetupDpy ); GRecvInt(); } if (!_compositor.isEmpty()) { comp = new TDEProcess; *comp << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + _compositor.ascii(); comp->start(TDEProcess::NotifyOnExit, TDEProcess::Stdin); } if (!_windowManager.isEmpty()) { if (_windowManager == "twin") { // Special case // Start DCOP... dcop = new TDEProcess; *dcop << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + "dcopserver" << TQCString("--suicide"); dcop->start(); } twin = new TDEProcess; *twin << TQCString( argv0, strrchr( argv0, '/' ) - argv0 + 2 ) + _windowManager.ascii(); if (_windowManager == "twin") { // Special case // Do not allow twin to start kompmgr... *twin << "--disablecompositionmanager"; } twin->start(); has_twin = true; } GSendInt( G_Ready ); kdDebug() << "[tdm-kfrontend] " << timestamp() << " main1" << endl; setCursor( dpy, app->desktop()->winId(), XC_left_ptr ); for (;;) { int rslt, cmd = GRecvInt(); if (cmd == G_ConfShutdown) { int how = GRecvInt(), uid = GRecvInt(); char *os = GRecvStr(); TDMSlimShutdown::externShutdown( how, os, uid ); if (os) free( os ); GSendInt( G_Ready ); _autoLoginDelay = 0; continue; } if (cmd == G_ErrorGreet) { if (KGVerify::handleFailVerify( TQT_TQWIDGET(tqApp->desktop()->screen( _greeterScreen )) )) break; _autoLoginDelay = 0; cmd = G_Greet; } TDEProcess *proc2 = 0; app->setOverrideCursor( Qt::WaitCursor ); FDialog *dialog = NULL; #ifdef XDMCP if (cmd == G_Choose) { dialog = new ChooserDlg; GSendInt( G_Ready ); /* tell chooser to go into async mode */ GRecvInt(); /* ack */ } else #endif { if ((cmd != G_GreetTimed && !_autoLoginAgain) || _autoLoginUser.isEmpty()) _autoLoginDelay = 0; if (_useTheme && !_theme.isEmpty() && !trinity_desktop_lock_use_sak) { // Qt4 has a nasty habit of generating BadWindow errors in normal operation, so we simply ignore them // This also prevents the user from being dropped to a console login if Xorg glitches or is buggy XSetErrorHandler( ignoreXError ); KThemedGreeter *tgrt; bool has_twin_bkp = has_twin; is_themed = true; if (has_twin) { has_twin = false; // [FIXME] The themed greeter is built on the assumption that there is no window manager available (i.e. it keeps stealing focus) and needs to be repaired. twin->kill(SIGKILL); } dialog = tgrt = new KThemedGreeter; kdDebug() << "[tdm-kfrontend] " << timestamp() << " themed" << endl; if (!tgrt->isOK()) { is_themed = false; has_twin = has_twin_bkp; delete tgrt; if (trinity_desktop_lock_use_sak) { checkSAK(app); } dialog = new KStdGreeter; #ifdef WITH_XRANDR dialog->move(dialog->x() + primaryScreenPosition.x(), dialog->y() + primaryScreenPosition.y()); #endif } else { #ifdef WITH_XRANDR dialog->move(primaryScreenPosition.x(), primaryScreenPosition.y()); #endif } XSync( tqt_xdisplay(), False ); XSetErrorHandler( (XErrorHandler)0 ); } else { if (trinity_desktop_lock_use_sak) { checkSAK(app); } dialog = new KStdGreeter; #ifdef WITH_XRANDR dialog->move(dialog->x() + primaryScreenPosition.x(), dialog->y() + primaryScreenPosition.y()); #endif } TQPoint oldCursorPos = TQCursor::pos(); #ifdef WITH_XRANDR TQCursor::setPos(oldCursorPos.x() + primaryScreenPosition.x(), oldCursorPos.y() + primaryScreenPosition.y()); #endif if (*_preloader) { proc2 = new TDEProcess; *proc2 << _preloader; proc2->start(); } } app->restoreOverrideCursor(); Debug( "entering event loop\n" ); // Qt4 has a nasty habit of generating BadWindow errors in normal operation, so we simply ignore them // This also prevents the user from being dropped to a console login if Xorg glitches or is buggy XSetErrorHandler( ignoreXError ); rslt = dialog->exec(); XSync( tqt_xdisplay(), False ); XSetErrorHandler( (XErrorHandler)0 ); Debug( "left event loop\n" ); login_user = static_cast<KGreeter*>(dialog)->curUser; login_session_wm = static_cast<KGreeter*>(dialog)->curWMSession; if (rslt != ex_greet) { delete dialog; } delete proc2; #ifdef XDMCP switch (rslt) { case ex_greet: GSendInt( G_DGreet ); continue; case ex_choose: GSendInt( G_DChoose ); continue; default: break; } #endif break; } KGVerify::done(); if (kbdl) { kbdl->closeStdin(); kbdl->detach(); } if (comp) { if (comp->isRunning()) { if (_compositor == TDE_COMPOSITOR_BINARY) { // Change process UID // Get user UID passwd* userinfo = getpwnam(login_user.ascii()); if (userinfo) { TQString newuid = TQString("%1").arg(userinfo->pw_uid); // kompmgr allows us to change its uid in this manner: // 1.) Send SIGUSR1 // 2.) Send the new UID to it on the command line comp->kill(SIGUSR1); comp->writeStdin(newuid.ascii(), newuid.length()); usleep(50000); // Give the above function some time to execute. Note that on REALLY slow systems this could fail, leaving kompmgr running as root. TODO: Look into ways to make this more robust. } } comp->closeStdin(); comp->detach(); } delete comp; } if (twin) { if (twin->isRunning()) { if (login_session_wm.endsWith("/starttde") || (login_session_wm == "failsafe")) { twin->closeStdin(); twin->detach(); dcop->detach(); } else { twin->kill(); dcop->kill(); } } delete twin; delete dcop; } delete proc; UnsecureDisplay( dpy ); restore_modifiers(); // Qt4 has a nasty habit of generating BadWindow errors in normal operation, so we simply ignore them // This also prevents the user from being dropped to a console login if Xorg glitches or is buggy XSetErrorHandler( ignoreXError ); XSetInputFocus( tqt_xdisplay(), PointerRoot, PointerRoot, CurrentTime ); XSync( tqt_xdisplay(), False ); XSetErrorHandler( (XErrorHandler)0 ); delete app; } } // extern "C" #include "kgapp.moc"