src/hbservers/hbdevicedialogappserver/main.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Mon, 04 Oct 2010 00:38:12 +0300
changeset 30 80e4d18b72f5
parent 28 b7da29130b0e
permissions -rw-r--r--
Revision: 201037 Kit: 201039

/****************************************************************************
**
** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (developer.feedback@nokia.com)
**
** This file is part of the HbServers module of the UI Extensions for Mobile.
**
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this file.
** Please review the following information to ensure the GNU Lesser General
** Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights.  These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at developer.feedback@nokia.com.
**
****************************************************************************/

#include "hbdevicedialogserver_p.h"

#include "hbdevicedialogtrace_p.h"
#include <hbapplication.h>
#include <hbmainwindow.h>
#include <hbview.h>
#include <hbtransparentwindow.h>
#include <hbstackedlayout.h>
#if defined (Q_OS_SYMBIAN)
#include <QFile>
#include <QDir>
#include <aknappui.h>
#include <eikenv.h>
#include <apgwgnam.h>
#include <hbdevicedialogserverdefs_p.h>
#include "hbddappfactorysymbian_p.h"
#endif // Q_OS_SYMBIAN

#if defined (Q_OS_SYMBIAN)
class Lock
{
public:
    enum State {
        Reserved,
        Acquired,
        Error
    };
    Lock();
    ~Lock(){close();}
    void close(){mFile.Close(); mFs.Close();}
    State acquire();
    static bool serverExists();

private:
    RFs mFs;
    RFile mFile;
};

Lock::Lock()
{
    // Using a file for interprocess lock
    const int NumMessageSlots = 1;
    if (mFs.Connect(NumMessageSlots) == KErrNone) {
        mFs.CreatePrivatePath(EDriveC);
        if (mFs.SetSessionToPrivate(EDriveC) == KErrNone) {
            _LIT(KFileName, "lockFile");
            const TUint mode = EFileShareReadersOrWriters;
            if (mFile.Create(mFs, KFileName, mode) == KErrAlreadyExists) {
                mFile.Open(mFs, KFileName, mode);
            }
        }
    }
}

// Try to acquire lock
Lock::State Lock::acquire()
{
    State state = Error;
    // If process holding the lock crashes, file server releases the lock
    if (mFile.SubSessionHandle()) {
        TInt error = mFile.Lock(0, 1);
        if (error == KErrNone) {
            state = Acquired;
        } else if (error == KErrLocked) {
            state = Reserved;
        }
    }
    return state;
}

// Check if Symbian server exists
bool Lock::serverExists()
{
    TFindServer findHbServer(KHbServerName);
    TFullName name;
    return findHbServer.Next(name) == KErrNone;
}
#endif // Q_OS_SYMBIAN

int main(int arg, char *args[])
{
    INSTALL_MESSAGE_HANDLER

#if defined (Q_OS_SYMBIAN)
    // Guard against starting multiple copies of the server
    Lock lock;
    Lock::State lockState;
    for(;;) {
        lockState = lock.acquire();
        if (lockState == Lock::Acquired) {
            break;
        } else if (lockState == Lock::Reserved) {
            // Process may be starting, wait for server object to be created
            if (Lock::serverExists()) {
                break;
            } else {
                const TInt KTimeout = 100000; // 100 ms
                User::After(KTimeout);
            }
        } else {
            break;
        }
    }
    if (lockState != Lock::Acquired) {
        // With KErrAlreadyExists client should try to connect, otherwise bail out.
        RProcess::Rendezvous(lockState == Lock::Reserved ? KErrAlreadyExists:KErrGeneral);
        return KErrNone;
    }
    _LIT(KThreadName, "hbdevdlgsrvapp");
    RThread().RenameMe(KThreadName); // nicer panic info
    RThread().SetProcessPriority(EPriorityHigh);
    
    HbDeviceDialogServerApp app(deviceDialogAppFactory, arg, args, Hb::NoSplash);
#else // Q_OS_SYMBIAN
    HbApplication app(arg, args);
#endif // Q_OS_SYMBIAN

    HbView* view = new HbView;
    view->hideItems(Hb::AllItems);
    view->setContentFullScreen();

    HbMainWindow mainWindow(0, Hb::WindowFlagTransparent|Hb::WindowFlagNoBackground);
        
#if defined (Q_OS_SYMBIAN)
    CCoeControl *c = mainWindow.effectiveWinId();
    RWindow *rw = static_cast<RWindow*>(c->DrawableWindow());
    rw->SetRequiredDisplayMode( EColor16MA );
    TInt err = rw->SetTransparencyAlphaChannel();
    if ( err == KErrNone ) {
        rw->SetBackgroundColor(~0);
    }
#endif // Q_OS_SYMBIAN

    HbTransparentWindow *transparentWindow = new HbTransparentWindow;
    HbStackedLayout *stackedLayout = new HbStackedLayout;
    stackedLayout->addItem(transparentWindow);
    view->setLayout(stackedLayout);

    mainWindow.addView(view);

#if defined (Q_OS_SYMBIAN)
    _LIT(KCaption, "HbDeviceDialogAppServer");
    CEikonEnv* env = CEikonEnv::Static();

    if (env) {
        env->SetSystem(true);
        CApaWindowGroupName* wgName = CApaWindowGroupName::NewLC(env->WsSession());
        wgName->SetHidden(ETrue); // hides us from FSW and protects us from OOM FW etc.
        wgName->SetSystem(ETrue); // Allow only application with PowerManagement cap to shut us down
        wgName->SetCaptionL(KCaption);
        wgName->SetAppUid(KNullUid);
        wgName->SetWindowGroupName(env->RootWin());
        CleanupStack::PopAndDestroy(wgName);
    }
#endif // Q_OS_SYMBIAN

    // HbApplication must be created before the server to have environment ready
    HbDeviceDialogServer server;
    server.setMainWindow(&mainWindow);

#if defined (Q_OS_SYMBIAN)
    // Server is created, client may connect
    RProcess::Rendezvous(KErrNone);
#endif // Q_OS_SYMBIAN

    // Unhandled exceptions mostly do not end up here. Instead they
    // go to Symbian exception handler which crashes the server.
    int returnValue = 0;
    bool hadException = true;
    try {
        returnValue = app.exec();
        hadException = false;
    } catch (const std::bad_alloc &) {
        returnValue = 1;
    } catch (...) {
        returnValue = 2;
    }

    TRACE_EXIT_ARGS("excep =" << hadException << "ret =" << returnValue);

#if defined (Q_OS_SYMBIAN)
    // Server exit is abnormal. Log event of it.
    QString logPath("c:/logs");
    QDir logDir;
    if (!logDir.exists(logPath)) {
        logDir.mkpath(logPath);
    }
    QFile logFile(logPath + "/hbdevdlgsrv.txt");
    if (logFile.open(QIODevice::WriteOnly | QIODevice::Text)) {
        logFile.write(QString().setNum(returnValue).insert(0, hadException ? 'E':'R').toAscii());
        logFile.close();
    }
#endif // Q_OS_SYMBIAN

    UNINSTALL_MESSAGE_HANDLER

    return returnValue;
}