/* * Copyright (C) 2011 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "TestInvocation.h" #include "PixelDumpSupport.h" #include "PlatformWebView.h" #include "TestController.h" #include #include #include #include #if PLATFORM(MAC) && !PLATFORM(IOS_FAMILY) #include #endif #if PLATFORM(IOS_FAMILY) // FIXME: get kUTTypePNG from MobileCoreServices on iOS static const CFStringRef kUTTypePNG = CFSTR("public.png"); #endif namespace WTR { static RetainPtr createCGContextFromCGImage(CGImageRef image) { size_t pixelsWide = CGImageGetWidth(image); size_t pixelsHigh = CGImageGetHeight(image); size_t rowBytes = (4 * pixelsWide + 63) & ~63; // Creating this bitmap in the device color space should prevent any color conversion when the image of the web view is drawn into it. auto colorSpace = adoptCF(CGColorSpaceCreateDeviceRGB()); auto context = adoptCF(CGBitmapContextCreate(0, pixelsWide, pixelsHigh, 8, rowBytes, colorSpace.get(), kCGImageAlphaPremultipliedFirst | kCGBitmapByteOrder32Host)); if (!context) return nullptr; CGContextDrawImage(context.get(), CGRectMake(0, 0, pixelsWide, pixelsHigh), image); return context; } static RetainPtr createCGContextFromImage(WKImageRef wkImage) { auto image = adoptCF(WKImageCreateCGImage(wkImage)); return createCGContextFromCGImage(image.get()); } void computeSHA1HashStringForContext(CGContextRef bitmapContext, char hashString[33]) { if (!bitmapContext) { WTFLogAlways("computeSHA1HashStringForContext: context is null\n"); return; } ASSERT(CGBitmapContextGetBitsPerPixel(bitmapContext) == 32); // ImageDiff assumes 32 bit RGBA, we must as well. size_t pixelsHigh = CGBitmapContextGetHeight(bitmapContext); size_t pixelsWide = CGBitmapContextGetWidth(bitmapContext); size_t bytesPerRow = CGBitmapContextGetBytesPerRow(bitmapContext); // We need to swap the bytes to ensure consistent hashes independently of endianness SHA1 sha1; unsigned char* bitmapData = static_cast(CGBitmapContextGetData(bitmapContext)); #if PLATFORM(COCOA) if ((CGBitmapContextGetBitmapInfo(bitmapContext) & kCGBitmapByteOrderMask) == kCGBitmapByteOrder32Big) { for (unsigned row = 0; row < pixelsHigh; row++) { Vector buffer(4 * pixelsWide); for (unsigned column = 0; column < pixelsWide; column++) buffer[column] = OSReadLittleInt32(bitmapData, 4 * column); sha1.addBytes(buffer); bitmapData += bytesPerRow; } } else #endif { for (unsigned row = 0; row < pixelsHigh; row++) { sha1.addBytes(bitmapData, 4 * pixelsWide); bitmapData += bytesPerRow; } } SHA1::Digest hash; sha1.computeHash(hash); hashString[0] = '\0'; for (size_t i = 0; i < 16; i++) snprintf(hashString, 33, "%s%02x", hashString, hash[i]); } static void dumpBitmap(CGContextRef bitmapContext, const char* checksum) { auto image = adoptCF(CGBitmapContextCreateImage(bitmapContext)); auto imageData = adoptCF(CFDataCreateMutable(0, 0)); auto imageDest = adoptCF(CGImageDestinationCreateWithData(imageData.get(), kUTTypePNG, 1, 0)); CGImageDestinationAddImage(imageDest.get(), image.get(), 0); CGImageDestinationFinalize(imageDest.get()); const unsigned char* data = CFDataGetBytePtr(imageData.get()); const size_t dataLength = CFDataGetLength(imageData.get()); printPNG(data, dataLength, checksum); } static void paintRepaintRectOverlay(CGContextRef context, WKSize imageSize, WKArrayRef repaintRects) { CGContextSaveGState(context); // Using a transparency layer is easier than futzing with clipping. CGContextBeginTransparencyLayer(context, 0); // Flip the context. CGContextScaleCTM(context, 1, -1); CGContextTranslateCTM(context, 0, -imageSize.height); CGContextSetRGBFillColor(context, 0, 0, 0, static_cast(0.66)); CGContextFillRect(context, CGRectMake(0, 0, imageSize.width, imageSize.height)); // Clear the repaint rects. size_t count = WKArrayGetSize(repaintRects); for (size_t i = 0; i < count; ++i) { WKRect rect = WKRectGetValue(static_cast(WKArrayGetItemAtIndex(repaintRects, i))); CGRect cgRect = CGRectMake(rect.origin.x, rect.origin.y, rect.size.width, rect.size.height); CGContextClearRect(context, cgRect); } CGContextEndTransparencyLayer(context); CGContextRestoreGState(context); } void TestInvocation::dumpPixelsAndCompareWithExpected(SnapshotResultType snapshotType, WKArrayRef repaintRects, WKImageRef wkImage) { RetainPtr context; WKSize imageSize; switch (snapshotType) { case SnapshotResultType::WebContents: if (!wkImage) { WTFLogAlways("dumpPixelsAndCompareWithExpected: image is null\n"); return; } context = createCGContextFromImage(wkImage); imageSize = WKImageGetSize(wkImage); break; case SnapshotResultType::WebView: auto image = TestController::singleton().mainWebView()->windowSnapshotImage(); if (!image) { WTFLogAlways("dumpPixelsAndCompareWithExpected: image is null\n"); return; } context = createCGContextFromCGImage(image.get()); imageSize = WKSizeMake(CGImageGetWidth(image.get()), CGImageGetHeight(image.get())); break; } if (!context) { WTFLogAlways("dumpPixelsAndCompareWithExpected: context is null\n"); return; } // A non-null repaintRects array means we're doing a repaint test. if (repaintRects) paintRepaintRectOverlay(context.get(), imageSize, repaintRects); char actualHash[33]; computeSHA1HashStringForContext(context.get(), actualHash); if (!compareActualHashToExpectedAndDumpResults(actualHash)) dumpBitmap(context.get(), actualHash); } } // namespace WTR