/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; namespace { class GrindApp : public Application { public: virtual int Main() override; virtual void Exception( ExceptionCategory nCategory ) override; }; class TestWindow : public Dialog { public: TestWindow() : Dialog( nullptr ) { SetText( "OutDev grinding" ); SetSizePixel( Size( 1024, 1024 ) ); EnablePaint( true ); Show(); } virtual void Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle& rRect) override; }; typedef std::function functor_type; typedef std::vector< std::pair > functor_vector_type; template< typename Functor > void add( functor_vector_type& res, const char* pStr, const Functor& func ) { res.push_back( std::make_pair(pStr,functor_type(func)) ); } void setupMethodStubs( functor_vector_type& res ) { const Color aWhiteColor( COL_WHITE ); const Color aBlackColor( COL_BLACK ); const Point aPt1(10,10); const Point aPt2(500,500); const Point aPt3(0,0); const Point aPt4(450,450); const tools::Rectangle aRect(aPt1,aPt2); const tools::Rectangle aRect2(aPt3,aPt4); const tools::Polygon aPoly(aRect); const tools::Polygon aPoly2(aRect2); tools::PolyPolygon aPolyPoly(aPoly); aPolyPoly.Insert( aPoly2 ); tools::Polygon aPoly3(aPoly2); aPoly3.Rotate( aPoly3.GetBoundRect().Center(), 900 ); const LineInfo aLineInfo(LineStyle::Solid,5); #ifdef FIXME_VDEV const OUString aString("This is a test"); // unfortunately, VDevs have inaccessible copy constructors static VirtualDevice aVDev; static VirtualDevice aVDevBW(1); const Size aVDevSize; aVDev.SetOutputSizePixel(aVDevSize); aVDevBW.SetOutputSizePixel(aVDevSize); const Bitmap aBitmap( aVDev.GetBitmap(aPt1,aVDevSize) ); const Bitmap aBitmapBW( aVDevBW.GetBitmap(aPt1,aVDevSize) ); const Bitmap aBitmapAlien( aVDevSize, 8 ); #else BitmapEx aIntro; rtl::Bootstrap::set("BRAND_BASE_DIR", "."); if (Application::LoadBrandBitmap ("intro", aIntro)) Application::Abort( "Failed to load intro image, run inside program/" ); const Bitmap aBitmap( aIntro.GetBitmap() ); Bitmap aBitmapBW( aBitmap ); aBitmapBW.Filter( BmpFilter::EmbossGrey ); Bitmap aBitmapAlien( Size( 100, 100 ), 8 ); aBitmapAlien.Erase( COL_RED ); #endif const BitmapEx aBitmapEx( aBitmap, aBitmapBW ); const BitmapEx aBitmapExBW( aBitmapBW, aBitmapBW ); const BitmapEx aBitmapExAlien( aBitmapAlien, aBitmapBW ); const BitmapEx aBitmapExAlpha( aBitmap, aBitmapAlien ); const BitmapEx aBitmapExAlphaAlien( aBitmapAlien, aBitmapAlien ); #ifdef NEEDS_QUALITY_PARAMETER const Image aImage( aBitmapEx ); #endif const Gradient aGradient(GradientStyle::Elliptical,aBlackColor,aWhiteColor); const Hatch aHatch(HatchStyle::Triple,aBlackColor,4,450); const Wallpaper aWallpaper( aWhiteColor ); GDIMetaFile aMtf; aMtf.AddAction( new MetaFillColorAction(Color(COL_RED),true) ); aMtf.AddAction( new MetaRectAction(aRect) ); #ifdef FIXME_NEEDS_LOVE add(res, "DrawTextArray", [&] (OutputDevice * pDev) { return pDev->DrawTextArray(aPt1, aString, (const sal_Int32*)0, (sal_uInt16)0, aString.getLength()); }); #endif /* void DrawPixel( const Point& rPt, const Color& rColor ); */ add(res, "DrawPixel", [aPt1, aBlackColor](OutputDevice *pOutDev) { return pOutDev->DrawPixel(aPt1, aBlackColor); }); /* void DrawLine( const Point& rStartPt, const Point& rEndPt ); */ add(res, "DrawLine", [aPt1, aPt2] (OutputDevice *pOutDev) { return pOutDev->DrawLine(aPt1, aPt2); }); /* void DrawLine( const Point& rStartPt, const Point& rEndPt, const LineInfo& rLineInfo ); */ add(res, "DrawLine(LineInfo)", [aPt1, aPt2, aLineInfo] (OutputDevice *pOutDev) { return pOutDev->DrawLine(aPt1, aPt2, aLineInfo); }); /* void DrawPolyLine( const Polygon& rPoly ); */ add(res, "DrawPolyLine", [aPoly] (OutputDevice *pOutDev) {return pOutDev->DrawPolyLine(aPoly); }); /* void DrawPolyLine( const Polygon& rPoly, const LineInfo& rLineInfo ); */ add(res, "DrawPolyLine(LineInfo)", [aPoly, aLineInfo] (OutputDevice *pOutDev) { return pOutDev->DrawPolyLine(aPoly, aLineInfo); }); /* void DrawPolygon( const Polygon& rPoly ); */ add(res, "DrawPolygon", [aPoly] (OutputDevice *pOutDev) { return pOutDev->DrawPolygon(aPoly); }); /* void DrawPolyPolygon( const tools::PolyPolygon& rPolyPoly ); */ add(res, "DrawPolyPolygon", [aPolyPoly] (OutputDevice *pOutDev) { return pOutDev->DrawPolyPolygon(aPolyPoly); }); /* void DrawRect( const Rectangle& rRect ); */ add(res, "DrawRect", [aRect] (OutputDevice *pOutDev) { return pOutDev->DrawRect(aRect); }); /* void DrawRect( const Rectangle& rRect, sal_uLong nHorzRount, sal_uLong nVertRound ); */ add(res, "DrawRect(round corners)", [aRect2] (OutputDevice *pOutDev) { return pOutDev->DrawRect(aRect2,4,4); }); /* void DrawEllipse( const Rectangle& rRect ); */ add(res, "DrawEllipse", [aRect] (OutputDevice *pOutDev) { return pOutDev->DrawEllipse(aRect); }); /* void DrawArc( const Rectangle& rRect, const Point& rStartPt, const Point& rEndPt ); */ add(res, "DrawArc", [aRect,aPt1,aPt2] (OutputDevice *pOutDev) { return pOutDev->DrawArc(aRect,aPt1,aPt2); }); /* void DrawPie( const Rectangle& rRect, const Point& rStartPt, const Point& rEndPt ); */ add(res, "DrawPie", [aRect2,aPt3,aPt4] (OutputDevice *pOutDev) { return pOutDev->DrawPie(aRect2,aPt3,aPt4); }); /* void DrawChord( const Rectangle& rRect, const Point& rStartPt, const Point& rEndPt ); */ add(res, "DrawChord", [aRect2,aPt3,aPt4] (OutputDevice *pOutDev) { return pOutDev->DrawChord(aRect2,aPt3,aPt4); }); /* void DrawOutDev( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPt, const Size& rSrcSize ); */ add(res, "DrawOutDev", [aRect,aRect2] (OutputDevice *pOutDev) { return pOutDev->DrawOutDev(aRect2.TopLeft(), aRect2.GetSize(), aRect.TopLeft(), aRect.GetSize()); }); #ifdef FIXME_VDEV /* void DrawOutDev( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPt, const Size& rSrcSize, const OutputDevice& rOutDev ); */ add(res, "DrawOutDev(foreign source)", [&] (OutputDevice * pDev) { return pDev->DrawOutDev(aRect2.TopLeft(), aRect2.GetSize(), aRect.TopLeft(), aRect.GetSize(), aVDevBW); }); /* void DrawOutDev( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPt, const Size& rSrcSize, const OutputDevice& rOutDev ); */ add(res, "DrawOutDev(foreign source, scaled)", [&] (OutputDevice * pDev) { return pDev->DrawOutDev(aRect2.TopLeft(), aRect2.GetSize(), aRect.TopLeft(), aRect.GetSize(), aVDev); }); #endif /* void CopyArea( const Point& rDestPt, const Point& rSrcPt, const Size& rSrcSize, sal_uInt16 nFlags = 0 ); */ add(res, "CopyArea", [&] (OutputDevice * pDev) { return pDev->CopyArea(aPt1, aPt3, aRect2.GetSize()); } ); #ifdef NEEDS_QUALITY_PARAMETER /* void DrawBitmap( const Point& rDestPt, const Bitmap& rBitmap ); */ add(res, "DrawBitmap(alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aBitmapAlien); }); /* void DrawBitmap( const Point& rDestPt, const Bitmap& rBitmap ); */ add(res, "DrawBitmap", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aBitmap); }); /* void DrawBitmap( const Point& rDestPt, const Size& rDestSize, const Bitmap& rBitmap ); */ add(res, "DrawBitmap(scaled,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aRect.GetSize(), aBitmapAlien); }); /* void DrawBitmap( const Point& rDestPt, const Size& rDestSize, const Bitmap& rBitmap ); */ add(res, "DrawBitmap(scaled)", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aRect.GetSize(), aBitmap); }); #if 0 /* void DrawBitmap( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const Bitmap& rBitmap ); */ add(res, "DrawBitmap(scaled subset,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapAlien); }); #endif /* void DrawBitmap( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const Bitmap& rBitmap ); */ add(res, "DrawBitmap(scaled subset)", [&] (OutputDevice * pDev) { return pDev->DrawBitmap(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmap); }); /* void DrawBitmapEx( const Point& rDestPt, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aBitmapExAlien); }); /* void DrawBitmapEx( const Point& rDestPt, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aBitmapEx); }); /* void DrawBitmapEx( const Point& rDestPt, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(alpha)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aBitmapExAlpha); }); /* void DrawBitmapEx( const Point& rDestPt, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(alpha, alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aBitmapExAlphaAlien); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aBitmapExAlien); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aBitmapEx); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled alpha)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aBitmapExAlpha); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled alpha, alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aBitmapExAlphaAlien); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled subset,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapExAlien); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled subset)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapEx); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled subset, alpha)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapExAlpha); }); /* void DrawBitmapEx( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const BitmapEx& rBitmapEx ); */ add(res, "DrawBitmapEx(scaled subset, alpha alien source)", [&] (OutputDevice * pDev) { return pDev->DrawBitmapEx(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapExAlphaAlien); }); /* void DrawMask( const Point& rDestPt, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask(alien source)", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aBitmapAlien, aBlackColor); }); /* void DrawMask( const Point& rDestPt, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aBitmap, aBlackColor); }); /* void DrawMask( const Point& rDestPt, const Size& rDestSize, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask(scaled,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aRect.GetSize(), aBitmapAlien, aBlackColor); }); /* void DrawMask( const Point& rDestPt, const Size& rDestSize, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask(scaled)", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aRect.GetSize(), aBitmap, aBlackColor); }); /* void DrawMask( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask(scaled subset,alien source)", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmapAlien, aBlackColor); }); /* void DrawMask( const Point& rDestPt, const Size& rDestSize, const Point& rSrcPtPixel, const Size& rSrcSizePixel, const Bitmap& rBitmap, const Color& rMaskColor ); */ add(res, "DrawMask(scaled subset)", [&] (OutputDevice * pDev) { return pDev->DrawMask(aPt1, aRect.GetSize(), aPt3, aRect2.GetSize(), aBitmap, aBlackColor); }); /* void DrawImage( const Point& rPos, const Image& rImage, sal_uInt16 nStyle = 0 ); */ add(res, "DrawImage", [&] (OutputDevice * pDev) { return pDev->DrawImage(aPt1, aImage, static_cast(0)); }); /* void DrawImage( const Point& rPos, const Size& rSize, const Image& rImage, sal_uInt16 nStyle = 0 ); */ add(res, "DrawImage(scaled)", [&] (OutputDevice * pDev) { return pDev->DrawImage(aPt1, aRect.GetSize(), aImage, static_cast(0))); }); #endif // NEEDS_QUALITY_PARAMETER /* void DrawGradient( const Rectangle& rRect, const Gradient& rGradient ); */ add(res, "DrawGradient", [&] (OutputDevice * pDev) { return pDev->DrawGradient(aRect, aGradient); }); /* void DrawGradient( const tools::PolyPolygon& rPolyPoly, const Gradient& rGradient ); */ add(res, "DrawGradient(polygon)", [&] (OutputDevice * pDev) { return pDev->DrawGradient(aPoly3, aGradient); }); /* void DrawHatch( const tools::PolyPolygon& rPolyPoly, const Hatch& rHatch ); */ add(res, "DrawHatch", [&] (OutputDevice * pDev) { return pDev->DrawHatch(aPoly3, aHatch); }); /* void DrawWallpaper( const Rectangle& rRect, const Wallpaper& rWallpaper ); */ add(res, "DrawWallpaper", [&] (OutputDevice * pDev) { return pDev->DrawWallpaper(aRect2, aWallpaper); }); #ifdef FIXME_HAVE_WAVE_NORMAL /* void DrawWaveLine( const Point& rStartPos, const Point& rEndPos, sal_uInt16 nStyle ); */ add(res, "DrawWaveLine", [&] (OutputDevice * pDev) { return pDev->DrawWaveLine(aPt1, aPt2, (sal_uInt16)WAVE_NORMAL); }); #endif /* void DrawGrid( const Rectangle& rRect, const Size& rDist, sal_uLong nFlags ); */ add(res, "DrawGrid", [&] (OutputDevice * pDev) { return pDev->DrawGrid(aRect, Size(10,20), DrawGridFlags::HorzLines|DrawGridFlags::VertLines); }); /* void DrawTransparent( const tools::PolyPolygon& rPolyPoly, sal_uInt16 nTransparencePercent ); */ add(res, "DrawTransparent", [&] (OutputDevice * pDev) { return pDev->DrawTransparent(aPoly3, (sal_uInt16)50); }); /* void DrawTransparent( const GDIMetaFile& rMtf, const Point& rPos, const Size& rSize, const Gradient& rTransparenceGradient ); */ add(res, "DrawTransparent(metafile)", [&] (OutputDevice * pDev) { return pDev->DrawTransparent(aMtf, aPt1, aRect.GetSize(), aGradient); }); /* void Erase(); */ add(res, "Erase", [] (OutputDevice * pDev) { return pDev->Erase(); } ); } void grindFunc( OutputDevice& rTarget, functor_vector_type::const_iterator const & iter, sal_Int32 nTurns, const char* pMsg ) { const sal_uInt32 nStartTime( osl_getGlobalTimer() ); for( sal_Int32 i=0; isecond(&rTarget); if( rTarget.GetOutDevType() == OUTDEV_WINDOW ) static_cast< vcl::Window & >( rTarget ).Flush(); fprintf( stdout, "Duration: %d ms (%d repetitions)\tOperation: %s\tSetup: %s\n", (int)(osl_getGlobalTimer() - nStartTime), (int)(nTurns), iter->first, pMsg ); } /** Call OutputDevice render methods repeatedly, and output elapsed time to stdout */ void outDevGrind(vcl::RenderContext& rTarget) { sal_Int32 nTurns = 100; // TODO(F1): also profile pure complex clip setup times! // State: fill/line color, draw mode, w/o clip, rect clip, complex clip functor_vector_type aMethods; setupMethodStubs( aMethods ); const tools::Rectangle aClipRect(10,10,1000,1000); const tools::Polygon aPoly1( aClipRect ); tools::Polygon aPoly2( aClipRect ); aPoly2.Rotate(aClipRect.Center(),450); tools::PolyPolygon aClipPoly(aPoly1); aClipPoly.Insert(aPoly2); functor_vector_type::const_iterator iter = aMethods.begin(); const functor_vector_type::const_iterator end = aMethods.end(); while( iter != end ) { rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::OverPaint ); rTarget.SetClipRegion(); grindFunc( rTarget, iter, nTurns, "w/o clip, w/o xor" ); rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::OverPaint ); rTarget.SetClipRegion( vcl::Region( aClipRect ) ); grindFunc( rTarget, iter, nTurns, "with rect clip, w/o xor" ); rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::OverPaint ); rTarget.SetClipRegion( vcl::Region( aClipPoly ) ); grindFunc( rTarget, iter, nTurns, "with complex clip, w/o xor" ); rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::Xor ); rTarget.SetClipRegion(); grindFunc( rTarget, iter, nTurns, "w/o clip, with xor" ); rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::Xor ); rTarget.SetClipRegion( vcl::Region( aClipRect ) ); grindFunc( rTarget, iter, nTurns, "with rect clip, with xor" ); rTarget.SetLineColor( Color(COL_BLACK) ); rTarget.SetFillColor( Color(COL_GREEN) ); rTarget.SetRasterOp( RasterOp::Xor ); rTarget.SetClipRegion( vcl::Region( aClipPoly ) ); grindFunc( rTarget, iter, nTurns, "with complex clip, with xor" ); ++iter; } } void TestWindow::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle&) { outDevGrind(rRenderContext); fflush(stdout); } void GrindApp::Exception( ExceptionCategory nCategory ) { switch( nCategory ) { case ExceptionCategory::ResourceNotLoaded: Abort( "Error: could not load language resources.\nPlease check your installation.\n" ); break; default: break; } } int GrindApp::Main() { ScopedVclPtrInstance aWindow; aWindow->Execute(); return 0; } } // namespace SAL_IMPLEMENT_MAIN() { try { bool bHelp = false; for( sal_uInt16 i = 0; i < Application::GetCommandLineParamCount(); i++ ) { OUString aParam = Application::GetCommandLineParam( i ); if( aParam == "--help" || aParam == "-h" ) bHelp = true; } if( bHelp ) { printf( "outdevgrind - Profile OutputDevice\n" ); return EXIT_SUCCESS; } tools::extendApplicationEnvironment(); uno::Reference< uno::XComponentContext > xContext = cppu::defaultBootstrap_InitialComponentContext(); uno::Reference< lang::XMultiServiceFactory > xServiceManager( xContext->getServiceManager(), uno::UNO_QUERY ); if( !xServiceManager.is() ) Application::Abort( "Failed to bootstrap" ); comphelper::setProcessServiceFactory( xServiceManager ); InitVCL(); GrindApp aGrindApp; aGrindApp.Main(); DeInitVCL(); } catch (const css::uno::Exception& e) { SAL_WARN("vcl.app", "Fatal exception: " << e.Message); return EXIT_FAILURE; } catch (const std::exception& e) { SAL_WARN("vcl.app", "Fatal exception: " << e.what()); return EXIT_FAILURE; } return EXIT_SUCCESS; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */