Update contrib.
1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
26 #include "test_step_logger.h"
27 #include "comparison.h"
32 #include "coordinatewin.h"
33 #include "backedupwin.h"
34 #include "enormouswin.h"
35 #include "animatedwin.h"
36 #include "spritewin.h"
40 const TInt KSteps = 10;
42 CComparison::COperationTimer::COperationTimer(CComparison* aComp):CTimer(0), iComp(aComp)
45 void CComparison::COperationTimer::ConstructL()
50 void CComparison::COperationTimer::DoCancel()
53 void CComparison::COperationTimer::RunL()
59 /*-------------------------------------------------------------------------*/
61 CComparison::COneShotCompare* CComparison::COneShotCompare::NewL(TInt aPriority, CComparison& aComparison)
63 return new(ELeave)CComparison::COneShotCompare(aPriority, aComparison);
66 CComparison::COneShotCompare::COneShotCompare(TInt aPriority, CComparison& aComparison)
67 :CAsyncOneShot(aPriority), iComparison(aComparison)
71 void CComparison::COneShotCompare::RunL()
73 iComparison.iScreen->CopyScreenToBitmap(iComparison.iScreenBitmap);
74 iComparison.SetVerifyTick(User::NTickCount());
75 iComparison.Verify(iComparison.iScreenBitmap);
77 /*-------------------------------------------------------------------------*/
79 _LIT(KConfigurationWindowSuffix, "-Window");
81 /*******************************************************************************
82 This is the stresslet itself
83 *******************************************************************************/
84 CComparison * CComparison::NewLC(MTestStepReporter& aReporter)
86 CComparison * self = new (ELeave) CComparison(aReporter);
87 CleanupStack::PushL(self);
92 void CComparison::ConstructL()
94 iWatchCat = CTestExecWatchCat::NewL( CExecutionContext::ECtxRandomAndRecord );
97 CComparison::CComparison (MTestStepReporter& aReporter):
98 CStresslet(aReporter), iDifferenceBitmap(NULL), iDifferenceDevice(NULL), iOneShotCompare(NULL)
103 Sets stress-test configuration (read from ini file).
104 @param aRunData configuration data
105 @param aConfFactory a factory interface for creating "section" readers. These
106 are used to supply configuration data for each window type.
108 void CComparison::SetRunDataL(const TRunData& aRunData, MTestStepConfigurationContextFactory* aConfFactory)
111 User::LeaveIfError(iFs.Connect());
112 iFs.MkDirAll(iData.loggingPath);
113 iWatchCat->SetLoggingPathL(iData.loggingPath);
115 //each window type can have its own configuration
116 MTestStepConfigurationContext* context; //context created by the factory argument
117 RBuf section; //section name (window-type + "-window"
119 CleanupStack::PushL(§ion);
121 struct STestStepConfEntry
123 const TPtrC8 iType; //section name
124 void (*iFunc)(const MTestStepConfigurationContext*); //configuration loader
127 {KCommonWindowType(), CCompWin::LoadConfiguration},
128 {KAnimatedWindowType(), CAnimatedWin::LoadConfiguration},
129 {KBackedupWindowType(), CBackedupWin::LoadConfiguration},
130 {KCoordinateWindowType(), CCoordinateWin::LoadConfiguration},
131 {KCrpWindowType(), CCrpWin::LoadConfiguration},
132 {KEdgedWindowType(), CEdgedWin::LoadConfiguration},
133 {KEnormousWindowType(), CEnormousWin::LoadConfiguration},
134 {KSpritedWindowType(), CSpritedWin::LoadConfiguration}
137 for (int i=0; i<sizeof(confLoaders)/sizeof(STestStepConfEntry); i++)
139 section.Copy(confLoaders[i].iType);
140 section.Append(KConfigurationWindowSuffix);
141 context = aConfFactory->GetConfigurationContextLC(section);
142 confLoaders[i].iFunc(context);
143 CleanupStack::PopAndDestroy(context);
146 CleanupStack::Pop(§ion);
149 void CComparison::StartL()
151 iStartTime.UniversalTime();
152 if (iData.randomSeed > -1)
154 //use seed from ini file
155 iFirstSeed = iData.randomSeed;
159 // randomize seed by time
160 iFirstSeed = iStartTime.Int64();
162 TRnd::SetSeed(iFirstSeed);
167 iWinGroup=new(ELeave) RWindowGroup(Session());
168 User::LeaveIfError(iWinGroup->Construct( (TUint32)(iWinGroup) ));
169 iWinGroup->SetOrdinalPosition(0,0);
171 //make sure at least minNumWindows are created, or if it's zero no limit
172 iNumWindowsLeft = iData.minNumWindows;
174 iScreen = new (ELeave) CWsScreenDevice(Session());
175 User::LeaveIfError(iScreen->Construct(0));
176 TDisplayMode screenMode = iScreen->DisplayMode();
178 iScreenBitmap = new (ELeave) CFbsBitmap;
179 iScreenBitmap->Create(TSize(iData.windowWidth, iData.windowHeight), screenMode);
181 iBackground = new(ELeave) RBlankWindow(Session());
182 iBackground->Construct(*iWinGroup,reinterpret_cast<TUint32>(iBackground));
183 iBackground->SetColor(KRgbBlack);
184 iBackground->SetExtent(TPoint(0,0), TSize(iData.windowWidth, iData.windowHeight));
185 iBackground->SetOrdinalPosition(0);
186 iBackground->Activate();
190 iBmpGc = CFbsBitGc::NewL();
191 for (TInt bmp = 0; bmp < 2; ++bmp)
193 iBitmap[bmp] = new (ELeave) CFbsBitmap;
194 iBitmap[bmp]->Create(TSize(iData.windowWidth, iData.windowHeight), screenMode); //EColor16MU); //
195 iDevice[bmp] = CFbsBitmapDevice::NewL(iBitmap[bmp]);
196 //clear bitmap background
197 iBmpGc->Activate(iDevice[bmp]);
199 iBmpGc->SetPenStyle(CGraphicsContext::ENullPen);
200 iBmpGc->SetBrushColor(KRgbBlack);
201 iBmpGc->SetBrushStyle(CGraphicsContext::ESolidBrush);
203 TRect rect(origin, TSize(iData.windowWidth, iData.windowHeight));
204 iBmpGc->DrawRect(rect);
207 iDifferenceBitmap = new (ELeave) CFbsBitmap;
208 iDifferenceBitmap->Create(TSize(iData.windowWidth, iData.windowHeight), screenMode); //EColor16MU); //
209 iDifferenceDevice = CFbsBitmapDevice::NewL(iDifferenceBitmap);
211 iTimer = new(ELeave) COperationTimer(this);
212 iTimer->ConstructL();
213 CActiveScheduler::Add (iTimer);
214 iTimer->After(iData.initPeriod);
218 Simulate a Left key stroke
220 void CComparison::Touch()
222 User::ResetInactivityTime();
225 keyDown.Set(TRawEvent::EKeyDown,EStdKeyLeftArrow);
226 Session().SimulateRawEvent(keyDown);
229 keyUp.Set(TRawEvent::EKeyUp,EStdKeyLeftArrow);
230 Session().SimulateRawEvent(keyUp);
234 Non leaving version of TickL that reports failures
236 TInt CComparison::Tick()
241 REPORT_EVENT( EFalse );
249 void CComparison::TickL()
257 if (0 == iTestNum % 100)
269 Dispatch work to selected operation by mode
271 void CComparison::DoStuffL()
286 if (iWasOk && iData.compareBitmaps && iStuffDone)
289 // schedule a delayed compare, will take place after all updates to windows
290 if (iOneShotCompare == NULL)
292 iOneShotCompare = CComparison::COneShotCompare::NewL(EPriorityMuchLess, *this);
294 iOneShotCompare->Call();
298 if (iTestNum > iData.maxRunCycles && iNumWindowsLeft == 0)
300 iMustConclude = ETrue;
302 iTimer->After(iData.period);
310 Performs a random operation on a random window
311 Operation can be one of create, destroy, move, bring to front/back,
312 resize, tick, toggle-visibility
314 void CComparison::ActL()
316 TInt action = TRnd::rnd(EACount);
317 if (iWindows.Count() == 0)
322 iBehaviour.iWin = RandomWindow();
323 iAct = static_cast<TAct>(action);
336 BringWindowToFrontL();
347 case EAToggleVisible:
350 } //lint !e787 enum constant TAct::EACount not used within switch
353 void CComparison::MoveL()
355 if (iBehaviour.iCount < 1)
361 TPoint pos = iBehaviour.iWin->Pos() + iBehaviour.iPos;
362 iBehaviour.iWin->SetPos(pos);
368 void CComparison::ResizeL()
370 if (iBehaviour.iCount < 1)
376 TSize size = iBehaviour.iWin->Size();
377 size.iWidth += iBehaviour.iPos.iX;
378 size.iHeight += iBehaviour.iPos.iY;
379 iBehaviour.iWin->SetSize(size);
386 Sets the position for a future window-move command, the actual move will be
387 done in the next step
389 void CComparison::MoveWindow()
394 TPoint pos = TPoint(TRnd::rnd(KPosLimit), TRnd::rnd(KPosLimit));
395 pos.iX -= iBehaviour.iWin->Size().iWidth / 2;
396 pos.iY -= iBehaviour.iWin->Size().iHeight / 2;
397 iBehaviour.iPos = pos - iBehaviour.iWin->Pos();
398 iBehaviour.iCount = KSteps;
399 iBehaviour.iPos.iX /= iBehaviour.iCount;
400 iBehaviour.iPos.iY /= iBehaviour.iCount;
407 Sets the size for a future window-resize command, the actual resize will be
408 done in the next step
410 void CComparison::ResizeWindow()
415 TPoint newsize = TPoint(TRnd::rnd(KPosLimit), TRnd::rnd(KPosLimit));
417 oldsize.iX = iBehaviour.iWin->Size().iWidth;
418 oldsize.iY = iBehaviour.iWin->Size().iHeight;
419 iBehaviour.iPos = newsize - oldsize;
420 iBehaviour.iCount = KSteps;
421 iBehaviour.iPos.iX /= iBehaviour.iCount;
422 iBehaviour.iPos.iY /= iBehaviour.iCount;
428 void CComparison::BringWindowToFrontL()
431 CCompWin* win = iBehaviour.iWin;
434 win->BringToFrontL();
435 TInt pos = FindTopWindow(win);
438 iWindows.Remove(pos);
439 iWindows.Append(win);
446 void CComparison::SendWindowToBackL()
449 TInt ord = iBackground->OrdinalPosition();
452 CCompWin* win = iBehaviour.iWin;
455 TInt pos = FindTopWindow(win);
458 iWindows.Remove(pos);
459 iWindows.Insert(win, 0);
460 win->Window()->SetOrdinalPosition(ord - 1);
472 TInt CComparison::FindTopWindow(CCompWin* win)
474 return iWindows.Find(win);
478 Returns a random window (which may be a NULL window)
480 CCompWin * CComparison::RandomWindow()
482 if (iWindows.Count() == 0)
487 TInt num = TRnd::rnd(iWindows.Count() + 1);
489 if (num == iWindows.Count())
495 return iWindows[num]->RandomWindow();
499 void CComparison::CreateWindowL()
501 CCompWin* parent = iBehaviour.iWin;
502 CCompWin* win = CCompWin::NewLC(Session(), iWinGroup, parent, WindowGc());
503 iBehaviour.iWin = win;
506 iWindows.AppendL(win);
508 CleanupStack::Pop(win);
509 if (iNumWindowsLeft > 0) //decrement window count
516 void CComparison::DestroyWindow()
518 CCompWin* win = iBehaviour.iWin;
521 TInt num = iWindows.Find(win);
522 if (num != KErrNotFound)
524 iWindows.Remove(num);
527 iBehaviour.iWin = 0; // rby added
532 void CComparison::TickWindowL()
534 CCompWin* win = iBehaviour.iWin;
544 void CComparison::ToggleVisible()
547 CCompWin* win = iBehaviour.iWin;
550 win->ToggleVisible();
556 void CComparison::HandleRedraw(TWsRedrawEvent &aEvent)
559 if (aEvent.Handle() != reinterpret_cast<TUint32>(iBackground))
561 reinterpret_cast<CCompWin*>(aEvent.Handle())->HandleRedraw(aEvent);
567 void CComparison::HandleEvent(TWsEvent &/*aEvent*/)
572 Compares two bitmaps. Comparison is done on a pixel-by-pixel basis.
573 If the bitmaps are of different sizes, the smaller axis of each is used for
574 comparison (top-left subregions).
575 First different pixel is stored in internally.
576 @return ETrue if bitmaps match, EFalse otherwise.
578 TBool CComparison::BitmapsMatch(const CFbsBitmap * aBitmap1, const CFbsBitmap * aBitmap2)
580 TDisplayMode mode1 = aBitmap1->DisplayMode();
581 TDisplayMode mode2 = aBitmap2->DisplayMode();
583 TSize bmpSize = aBitmap1->SizeInPixels();
584 TSize bmpSize2 = aBitmap2->SizeInPixels();
585 if (bmpSize2.iWidth < bmpSize.iWidth)
587 bmpSize.iWidth = bmpSize2.iWidth;
589 if (bmpSize2.iHeight < bmpSize.iHeight)
591 bmpSize.iHeight = bmpSize2.iHeight;
596 for (TInt y=0; y < bmpSize.iHeight; y++)
598 for (TInt x=0; x < bmpSize.iWidth; x++)
601 aBitmap1->GetPixel(c1, point);
602 aBitmap2->GetPixel(c2, point);
605 iPixel1 = c1.Value();
606 iPixel2 = c2.Value();
614 void CComparison::Verify(CFbsBitmap * aServerBmp)
616 //Copy the currentBmp to lastBmp
617 TInt dataStride = iBitmap[iCurrentBmp]->DataStride();
618 TInt sizeInByte = dataStride * iBitmap[iCurrentBmp]->SizeInPixels().iHeight;
619 TUint32* bitmapAddressSource = iBitmap[iCurrentBmp]->DataAddress();
620 TUint32* bitmapAddressTarget = iBitmap[iLastBmp]->DataAddress();
621 memcpy(bitmapAddressTarget, bitmapAddressSource, sizeInByte);
622 // This method is only used by the animation and sprite windows; ignores
623 // comparisons for times that are too close to the sprite change. But as the
624 // difference error drifts, there is no point relying on it.
625 if (!WindowsReadyForVerification())
635 const TBool bmpMatch = BitmapsMatch(aServerBmp, iBitmap[iCurrentBmp]);
636 REPORT_EVENT( bmpMatch );
645 if (!iData.saveOnlyDifferent || !bmpMatch)
648 fileName.Append(iData.loggingPath);
649 fileName.Append(_L("Stresslet_Comparison_"));
650 if (iData.saveOnlyDifferent)
652 fileName.AppendNumFixedWidthUC((TUint)iErrorNum, EDecimal, 3);
656 fileName.AppendNumFixedWidthUC((TUint)iTestNum, EDecimal, 3);
658 TInt baseLength = fileName.Length();
660 //previous is saved only when saving different bitmaps
661 //otherwise it is always the same as the previous expected
662 if (iData.saveOnlyDifferent)
664 fileName.Append(_L("_Previous.mbm"));
665 __ASSERT_ALWAYS(KErrNone == iBitmap[iLastBmp]->Save(fileName), Panic(EPanic6));
668 fileName.SetLength(baseLength);
669 fileName.Append(_L("_Expected.mbm"));
670 __ASSERT_ALWAYS(KErrNone == iBitmap[iCurrentBmp]->Save(fileName), Panic(EPanic7));
672 fileName.SetLength(baseLength);
673 fileName.Append(_L("_Screen.mbm"));
674 __ASSERT_ALWAYS(KErrNone == aServerBmp->Save(fileName), Panic(EPanic8));
676 //store difference between expected and screen bitmaps (XOR image)
677 if (iData.saveDifferenceBitmap)
679 iBmpGc->Activate(iDifferenceDevice);
681 iBmpGc->SetDrawMode(CGraphicsContext::EDrawModePEN);
682 iBmpGc->BitBlt(TPoint(0,0), iBitmap[iCurrentBmp]);
683 iBmpGc->SetDrawMode(CGraphicsContext::EDrawModeXOR);
684 iBmpGc->BitBlt(TPoint(0,0), aServerBmp);
686 fileName.SetLength(baseLength);
687 fileName.Append(_L("_Difference.mbm"));
688 __ASSERT_ALWAYS(KErrNone == iDifferenceBitmap->Save(fileName), Panic(EPanic19));
691 fileName.SetLength(baseLength);
692 fileName.Append(_L("_info.txt"));
694 TInt err = file.Create(iFs, fileName, EFileWrite);
696 err = file.Replace(iFs, fileName, EFileWrite);
700 info.Append(_L8("Seed = "));
701 info.AppendNum(iFirstSeed);
702 info.Append(_L8("\n"));
707 info.Append(_L8(" Runtime = "));
708 info.AppendNum(now.MicroSecondsFrom(iStartTime).Int64() / 1000);
709 info.Append(_L8(" ms\r\n"));
712 info.Append(_L8("Action = ["));
713 info.AppendNum((TInt64)iAct);
714 info.Append(_L8("] Mode = ["));
715 info.AppendNum((TInt64)iMode);
716 info.Append(_L8("] Test = ["));
717 info.AppendNum((TInt64)iTestNum);
718 info.Append(_L8("]\r\n"));
721 info.Append(_L8("Pixel at ["));
722 info.AppendNum((TInt64)iPixelPos.iX);
723 info.Append(_L8(","));
724 info.AppendNum((TInt64)iPixelPos.iY);
725 info.Append(_L8("] mismatch screen 0x"));
726 info.AppendNum((TInt64)iPixel1, EHex);
727 info.Append(_L8(" != bitmap 0x"));
728 info.AppendNum((TInt64)iPixel2, EHex);
729 info.Append(_L8("\r\n\r\n"));
733 for (TInt num = 0; num < iWindows.Count(); ++num)
735 iWindows[num]->Dump(file, zero, 0, iBehaviour.iWin);
737 info.Append(_L8("\r\n"));
744 //failed create logfile
745 __ASSERT_ALWAYS(EFalse, Panic(EPanic9));
754 if ((iTestNum > iData.maxRunCycles && iNumWindowsLeft == 0) || !iWasOk)
756 iMustConclude = ETrue;
758 iTimer->After(iData.period);
762 Write information(eg. seed, starttime, endtime and time elapse) to Stresslet_Log.txt file
764 void CComparison::WriteLog()
767 fileName.Append(iData.loggingPath);
768 fileName.Append(_L("Stresslet_Log.txt"));
770 TInt err = file.Create(iFs, fileName, EFileWrite);
773 err = file.Replace(iFs, fileName, EFileWrite);
778 info.Append(_L8("Seed = "));
779 info.AppendNum(iFirstSeed);
780 info.Append(_L8("\r\n"));
783 TBuf<40> dateTimeString;
784 _LIT(KFormat2,"%D%M%Y%/0%1%/1%2%/2%3%/3 %:0%H%:1%T%:2%S.%C%:3");
785 iStartTime.FormatL(dateTimeString,KFormat2);
786 info.Append(_L8("StartTime = "));
787 info.Append(dateTimeString);
788 info.Append(_L8("\r\n"));
791 now.FormatL(dateTimeString,KFormat2);
792 info.Append(_L8("Endtime = "));
793 info.Append(dateTimeString);
794 info.Append(_L8("\r\n"));
795 info.Append(_L8("Elapse = "));
796 info.AppendNum(now.MicroSecondsFrom(iStartTime).Int64() / 1000);
797 info.Append(_L8(" ms\r\n"));
803 Returns true if all visible windows are ready for verification
805 TBool CComparison::WindowsReadyForVerification() const
809 while ( idx < iWindows.Count() && res )
811 if ( iWindows[ idx ]->IsVisible() )
813 res = iWindows[ idx ]->QueryReadyForVerification();
821 Calls all windows to draw themselves on a bitmap
823 void CComparison::DrawBitmap()
825 iBmpGc->Activate(iDevice[iCurrentBmp]);
828 // clear background area
829 iBmpGc->SetPenStyle(CGraphicsContext::ENullPen);
830 iBmpGc->SetBrushColor(KRgbBlack);
831 iBmpGc->SetBrushStyle(CGraphicsContext::ESolidBrush);
834 TRect rect(origin, TSize(iData.windowWidth, iData.windowHeight));
836 //create a region containing all screen, subtract all visible windows
837 RRegion backClipping(rect);
838 if (!iData.clearAllBackground)
840 for (TInt num = 0; num < iWindows.Count(); ++num)
842 if (iWindows[num]->IsVisible())
843 iWindows[num]->SubSelfFromRegion(backClipping, rect, origin);
847 //clip drawing to background only & clear
848 iBmpGc->SetClippingRegion(backClipping);
849 iBmpGc->DrawRect(rect);
850 //go back to no clipping
851 iBmpGc->CancelClipping();
852 backClipping.Close();
854 for (TInt num = 0; num < iWindows.Count(); ++num)
856 if (iWindows[num]->IsVisible())
858 iWindows[num]->ClearBitmapBackground(iBmpGc, rect, origin);
859 iWindows[num]->DrawBitmap(iBmpGc, rect, origin);
864 void CComparison::SetVerifyTick(TUint32 aTick)
866 for (TInt num = 0; num < iWindows.Count(); ++num)
868 if (iWindows[num]->IsVisible())
869 iWindows[num]->SetVerifyTick(aTick);
873 CComparison::~CComparison()
879 iBackground->Close();
883 iWindows.ResetAndDestroy();
890 delete iOneShotCompare; //delayed compare
892 delete iDifferenceDevice;
893 delete iDifferenceBitmap;
895 for (TInt bmp = 0; bmp < 2; ++bmp)
904 delete iScreenBitmap;