diff -r 730c025d4b77 -r f378acbc9cfb src/hbcore/gui/hbsplash.cpp --- a/src/hbcore/gui/hbsplash.cpp Thu Jul 15 14:03:49 2010 +0100 +++ b/src/hbcore/gui/hbsplash.cpp Thu Jul 22 16:36:53 2010 +0100 @@ -24,20 +24,23 @@ ****************************************************************************/ #include "hbsplash_p.h" +#include "hbsplash_direct_symbian_p.h" #include #include #include +#include #ifdef Q_OS_SYMBIAN #include #include +#include #include "hborientationstatus_p.h" #include "hbsplashdefs_p.h" #endif /*! \class HbSplash - + \brief Class with utility functions for accessing splash screens. \internal @@ -61,8 +64,7 @@ return QString("prt"); } -struct Params -{ +struct Params { int *w; int *h; int *bpl; @@ -72,6 +74,8 @@ QString screenId; HbSplash::AllocFunc allocFunc; void *allocFuncParam; + quint32 extra; + bool forceFile; }; struct File { @@ -94,18 +98,30 @@ #endif } -static uchar *readSpl(File &f, const Params ¶ms) +static uchar *readSpl(File &f, Params ¶ms) { - int w = 0, h = 0, bpl = 0; + quint32 w = 0, h = 0, bpl = 0; QImage::Format fmt = QImage::Format_Invalid; - f.read((char *) &w, sizeof(int)); - f.read((char *) &h, sizeof(int)); - f.read((char *) &bpl, sizeof(int)); - f.read((char *) &fmt, sizeof(QImage::Format)); + // Have to read the header in one piece in order to minimize the + // number of read() calls. + const int headerLength = sizeof(quint32) * 5; + char headerBuf[headerLength]; + qMemSet(headerBuf, 0, headerLength); + f.read(headerBuf, headerLength); + quint32 *headerPtr = reinterpret_cast(headerBuf); + w = *headerPtr++; + h = *headerPtr++; + bpl = *headerPtr++; + fmt = (QImage::Format) * headerPtr++; + params.extra = *headerPtr; if (fmt != QImage::Format_ARGB32_Premultiplied) { - qWarning("HbSplash: image format for %s is not ARGB32_PRE (is %d instead)", + qWarning("[hbsplash] Image format for %s is not ARGB32_PRE (is %d instead)", qPrintable(f.mFullName), fmt); } + if (fmt < 0 || fmt >= QImage::NImageFormats) { + qWarning("[hbsplash] Image format is invalid"); + return 0; + } qint64 sz = h * bpl; uchar *data = 0; if (w > 0 && h > 0 && bpl > 0 && sz > 0 && sz <= image_bytes_limit) { @@ -116,20 +132,20 @@ data = new uchar[sz]; } if (data) { - qint64 bytesRead = f.read((char *) data, sz); + qint64 bytesRead = f.read(reinterpret_cast(data), sz); if (bytesRead != sz) { - qWarning("HbSplash: file %s is invalid", qPrintable(f.mFullName)); + qWarning("[hbsplash] File %s is invalid", qPrintable(f.mFullName)); if (!params.allocFunc) { - delete data; + delete[] data; } data = 0; } } } catch (const std::bad_alloc &) { - qWarning("HbSplash: failed to allocate image buffer"); + qWarning("[hbsplash] Failed to allocate image buffer"); } } else { - qWarning("HbSplash: image in file %s is too big", qPrintable(f.mFullName)); + qWarning("[hbsplash] Image in file %s is too big", qPrintable(f.mFullName)); } *params.w = w; *params.h = h; @@ -140,13 +156,27 @@ #ifdef Q_OS_SYMBIAN +// Symbian-specific implementation to get splash screens either by +// reading from a file opened on the server side or by using bitmaps +// shared via fbserv. + class HbSplashSrvClient : public RSessionBase { public: HbSplashSrvClient(); ~HbSplashSrvClient(); bool Connect(); - bool getSplash(RFile &f, const QString &ori, const QString &appId, const QString &screenId); + + bool getSplashFileHandle(RFile &f, + const QString &ori, + const QString &appId, + const QString &screenId); + + uchar *getSplashFromBitmap(const QString &ori, + const QString &appId, + const QString &screenId, + Params ¶ms); + private: RMutex mMutex; bool mMutexOk; @@ -185,9 +215,8 @@ if (err == KErrNone) { ok = true; break; -/* } else if (err == KErrNotFound || err == KErrServerTerminated) { - qDebug("[hbsplash] Server not running"); + qDebug("[hbsplash] server not running"); TFindServer findServer(hbsplash_server_name); TFullName name; if (findServer.Next(name) != KErrNone) { @@ -211,15 +240,14 @@ qWarning("[hbsplash] Rendezvous failed (%d)", status.Int()); break; } - qDebug("[hbsplash] Server started"); + qDebug("[hbsplash] server started"); } -*/ } else { break; } } if (!ok) { - qWarning("[hbsplash] cannot connect to splashgen server"); + qWarning("[hbsplash] Cannot connect to splashgen server"); } if (mMutexOk) { mMutex.Signal(); @@ -227,8 +255,10 @@ return ok; } -bool HbSplashSrvClient::getSplash(RFile &f, const QString &ori, - const QString &appId, const QString &screenId) +bool HbSplashSrvClient::getSplashFileHandle(RFile &f, + const QString &ori, + const QString &appId, + const QString &screenId) { TPtrC oriDes(static_cast(ori.utf16()), ori.length()); TPtrC appIdDes(static_cast(appId.utf16()), appId.length()); @@ -236,11 +266,62 @@ TInt fileHandle; TPckg fileHandlePckg(fileHandle); TIpcArgs args(&oriDes, &appIdDes, &screenIdDes, &fileHandlePckg); - TInt fsHandle = SendReceive(HbSplashSrvGetSplash, args); + TInt fsHandle = SendReceive(HbSplashSrvGetSplashFile, args); return f.AdoptFromServer(fsHandle, fileHandle) == KErrNone; } -static uchar *load_symbian(const Params ¶ms) +uchar *HbSplashSrvClient::getSplashFromBitmap(const QString &ori, + const QString &appId, + const QString &screenId, + Params ¶ms) +{ + TPtrC oriDes(static_cast(ori.utf16()), ori.length()); + TPtrC appIdDes(static_cast(appId.utf16()), appId.length()); + TPtrC screenIdDes(static_cast(screenId.utf16()), screenId.length()); + TInt bitmapHandle; + TPckg bitmapHandlePckg(bitmapHandle); + TIpcArgs args(&oriDes, &appIdDes, &screenIdDes, &bitmapHandlePckg); + if (SendReceive(HbSplashSrvGetSplashData, args) == KErrNone) { + QScopedPointer bmp(new CFbsBitmap); + if (bmp->Duplicate(bitmapHandle) == KErrNone) { + TSize size = bmp->SizeInPixels(); + TDisplayMode mode = bmp->DisplayMode(); + if (size.iWidth > 0 && size.iHeight > 0 && mode == EColor16MAP) { + int bpl = CFbsBitmap::ScanLineLength(size.iWidth, mode); + const QImage::Format fmt = QImage::Format_ARGB32_Premultiplied; + int len = bpl * size.iHeight; + uchar *data = 0; + try { + if (params.allocFunc) { + data = params.allocFunc(size.iWidth, size.iHeight, + bpl, fmt, params.allocFuncParam); + } else { + data = new uchar[len]; + } + if (data) { + memcpy(data, bmp->DataAddress(), len); + *params.w = size.iWidth; + *params.h = size.iHeight; + *params.bpl = bpl; + *params.fmt = fmt; + qDebug("[hbsplash] bitmap data received"); + return data; + } + } catch (const std::bad_alloc &) { + qWarning("[hbsplash] Failed to allocate image buffer"); + } + } else { + qWarning("[hbsplash] Invalid bitmap (%d %d %d)", + size.iWidth, size.iHeight, mode); + } + } else { + qWarning("[hbsplash] Cannot duplicate bitmap"); + } + } + return 0; +} + +static uchar *load_symbian(Params ¶ms) { HbSplashSrvClient client; if (!client.Connect()) { @@ -252,15 +333,20 @@ if (appIdStr.isEmpty()) { RProcess process; appIdStr = QString::number(process.SecureId().iId, 16); + process.Close(); } uchar *data = 0; - File f; - f.mFullName = "[unavailable]"; - if (client.getSplash(f.mFile, oriStr, appIdStr, params.screenId)) { - qDebug("[hbsplash] got handle from server"); - data = readSpl(f, params); - f.mFile.Close(); + if (!params.forceFile) { + data = client.getSplashFromBitmap(oriStr, appIdStr, params.screenId, params); + } else { + File f; + f.mFullName = "[unavailable]"; + if (client.getSplashFileHandle(f.mFile, oriStr, appIdStr, params.screenId)) { + qDebug("[hbsplash] got handle from server"); + data = readSpl(f, params); + f.mFile.Close(); + } } client.Close(); @@ -269,7 +355,11 @@ #else -static uchar *read_file_generic(const QString &name, const Params ¶ms) +// Generic cross-platform implementation, reads the pixel data +// directly from files. Not suitable for Symbian due to platsec and +// performance reasons. + +static uchar *read_file_generic(const QString &name, Params ¶ms) { uchar *data = 0; File f; @@ -282,7 +372,7 @@ return data; } -static uchar *load_generic(const Params ¶ms) +static uchar *load_generic(Params ¶ms) { QString appSpecificName("splash_%1_%2.spl"); QString appAndScreenSpecificName("splash_%1_%2_%3.spl"); @@ -361,9 +451,65 @@ params.screenId = screenId; params.allocFunc = allocFunc; params.allocFuncParam = allocFuncParam; + params.forceFile = false; // use CFbsBitmap-based sharing on Symbian #ifdef Q_OS_SYMBIAN return load_symbian(params); #else return load_generic(params); #endif } + +#ifdef Q_OS_SYMBIAN +static uchar *fbsBitmapAllocFunc(int w, int h, int bpl, QImage::Format fmt, void *param) +{ + if (fmt != QImage::Format_ARGB32_Premultiplied) { + qWarning("[hbsplash] fbsBitmapAllocFunc: unsupported format %d", fmt); + return 0; + } + TDisplayMode mode = EColor16MAP; + CFbsBitmap *bmp = static_cast(param); + if (bmp->Create(TSize(w, h), mode) == KErrNone) { + int bmpBpl = CFbsBitmap::ScanLineLength(w, mode); + if (bpl == bmpBpl) { + return reinterpret_cast(bmp->DataAddress()); + } else { + qWarning("[hbsplash] fbsBitmapAllocFunc: bpl mismatch (%d - %d)", bpl, bmpBpl); + } + } else { + qWarning("[hbsplash] fbsBitmapAllocFunc: bitmap Create() failed"); + } + return 0; +} +#endif + +void *HbSplashDirectSymbian::load(void *file, int *extra) +{ +#ifdef Q_OS_SYMBIAN + // Read the data directly into a CFbsBitmap. `file' is assumed to + // be a ptr to an already Open()'ed RFile. + QScopedPointer bmp(new CFbsBitmap); + int w, h, bpl; + QImage::Format fmt; + Params params; + // Everything is ignored except the alloc-func and its param. + params.w = &w; + params.h = &h; + params.bpl = &bpl; + params.fmt = &fmt; + params.flags = HbSplashScreen::Default; + params.allocFunc = fbsBitmapAllocFunc; + params.allocFuncParam = bmp.data(); + File f; + f.mFile = *static_cast(file); + if (readSpl(f, params)) { + if (extra) { + *extra = params.extra; + } + return bmp.take(); + } +#else + Q_UNUSED(file); + Q_UNUSED(extra); +#endif + return 0; +}