| Index: tests/SurfaceTest.cpp | 
| diff --git a/tests/SurfaceTest.cpp b/tests/SurfaceTest.cpp | 
| index 878adeda3d957a86fe2805b429c2c8e5626c425f..0cf602007ca2ba839ac98f7d2ea0f092355ef6aa 100644 | 
| --- a/tests/SurfaceTest.cpp | 
| +++ b/tests/SurfaceTest.cpp | 
| @@ -20,6 +20,7 @@ | 
| #include "GrContext.h" | 
| #include "GrDrawContext.h" | 
| #include "GrGpu.h" | 
| +#include "GrResourceProvider.h" | 
| #endif | 
|  | 
| #include <initializer_list> | 
| @@ -83,75 +84,6 @@ DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfaceEmpty_Gpu, reporter, ctxInfo) { | 
| } | 
| #endif | 
|  | 
| -#if SK_SUPPORT_GPU | 
| -DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfaceWrappedTexture, reporter, ctxInfo) { | 
| -    GrGpu* gpu = ctxInfo.fGrContext->getGpu(); | 
| -    if (!gpu) { | 
| -        return; | 
| -    } | 
| - | 
| -    // Test the wrapped factory for SkSurface by creating a backend texture and then wrap it in | 
| -    // a SkSurface. | 
| -    static const int kW = 100; | 
| -    static const int kH = 100; | 
| -    static const uint32_t kOrigColor = 0xFFAABBCC; | 
| -    SkAutoTArray<uint32_t> pixels(kW * kH); | 
| -    sk_memset32(pixels.get(), kOrigColor, kW * kH); | 
| -    GrBackendObject texHandle = gpu->createTestingOnlyBackendTexture(pixels.get(), kW, kH, | 
| -                                                                     kRGBA_8888_GrPixelConfig); | 
| - | 
| -    GrBackendTextureDesc wrappedDesc; | 
| -    wrappedDesc.fConfig = kRGBA_8888_GrPixelConfig; | 
| -    wrappedDesc.fWidth = kW; | 
| -    wrappedDesc.fHeight = kH; | 
| -    wrappedDesc.fOrigin = kBottomLeft_GrSurfaceOrigin; | 
| -    wrappedDesc.fSampleCnt = 0; | 
| -    wrappedDesc.fFlags = kRenderTarget_GrBackendTextureFlag; | 
| -    wrappedDesc.fTextureHandle = texHandle; | 
| - | 
| -    auto surface(SkSurface::MakeFromBackendTexture(ctxInfo.fGrContext, wrappedDesc, nullptr)); | 
| -    REPORTER_ASSERT(reporter, surface); | 
| -    if (surface) { | 
| -        // Validate that we can draw to the canvas and that the original texture color is preserved | 
| -        // in pixels that aren't rendered to via the surface. | 
| -        SkPaint paint; | 
| -        static const SkColor kRectColor = ~kOrigColor | 0xFF000000; | 
| -        paint.setColor(kRectColor); | 
| -        surface->getCanvas()->drawRect(SkRect::MakeWH(SkIntToScalar(kW), SkIntToScalar(kH)/2), | 
| -                                       paint); | 
| -        SkImageInfo readInfo = SkImageInfo::MakeN32Premul(kW, kH); | 
| -        surface->readPixels(readInfo, pixels.get(), kW * sizeof(uint32_t), 0, 0); | 
| -        bool stop = false; | 
| -        SkPMColor origColorPM = SkPackARGB32((kOrigColor >> 24 & 0xFF), | 
| -                                             (kOrigColor >>  0 & 0xFF), | 
| -                                             (kOrigColor >>  8 & 0xFF), | 
| -                                             (kOrigColor >> 16 & 0xFF)); | 
| -        SkPMColor rectColorPM = SkPackARGB32((kRectColor >> 24 & 0xFF), | 
| -                                             (kRectColor >> 16 & 0xFF), | 
| -                                             (kRectColor >>  8 & 0xFF), | 
| -                                             (kRectColor >>  0 & 0xFF)); | 
| -        for (int y = 0; y < kH/2 && !stop; ++y) { | 
| -            for (int x = 0; x < kW && !stop; ++x) { | 
| -                REPORTER_ASSERT(reporter, rectColorPM == pixels[x + y * kW]); | 
| -                if (rectColorPM != pixels[x + y * kW]) { | 
| -                    stop = true; | 
| -                } | 
| -            } | 
| -        } | 
| -        stop = false; | 
| -        for (int y = kH/2; y < kH && !stop; ++y) { | 
| -            for (int x = 0; x < kW && !stop; ++x) { | 
| -                REPORTER_ASSERT(reporter, origColorPM == pixels[x + y * kW]); | 
| -                if (origColorPM != pixels[x + y * kW]) { | 
| -                    stop = true; | 
| -                } | 
| -            } | 
| -        } | 
| -    } | 
| -    gpu->deleteTestingOnlyBackendTexture(texHandle); | 
| -} | 
| -#endif | 
| - | 
| static void test_canvas_peek(skiatest::Reporter* reporter, | 
| sk_sp<SkSurface>& surface, | 
| const SkImageInfo& requestInfo, | 
| @@ -820,10 +752,56 @@ DEF_TEST(surface_rowbytes, reporter) { | 
| } | 
|  | 
| #if SK_SUPPORT_GPU | 
| +static sk_sp<SkSurface> create_gpu_surface_backend_texture( | 
| +    GrContext* context, int sampleCnt, uint32_t color, GrBackendObject* outTexture) { | 
| +    const int kWidth = 10; | 
| +    const int kHeight = 10; | 
| +    SkAutoTDeleteArray<uint32_t> pixels(new uint32_t[kWidth * kHeight]); | 
| +    sk_memset32(pixels.get(), color, kWidth * kHeight); | 
| +    GrBackendTextureDesc desc; | 
| +    desc.fConfig = kRGBA_8888_GrPixelConfig; | 
| +    desc.fWidth = kWidth; | 
| +    desc.fHeight = kHeight; | 
| +    desc.fFlags = kRenderTarget_GrBackendTextureFlag; | 
| +    desc.fTextureHandle = context->getGpu()->createTestingOnlyBackendTexture( | 
| +        pixels.get(), kWidth, kHeight, kRGBA_8888_GrPixelConfig); | 
| +    desc.fSampleCnt = sampleCnt; | 
| +    sk_sp<SkSurface> surface = SkSurface::MakeFromBackendTexture(context, desc, nullptr); | 
| +    if (!surface) { | 
| +        context->getGpu()->deleteTestingOnlyBackendTexture(desc.fTextureHandle); | 
| +        return nullptr; | 
| +    } | 
| +    *outTexture = desc.fTextureHandle; | 
| +    return surface; | 
| +} | 
| + | 
| +static sk_sp<SkSurface> create_gpu_surface_backend_texture_as_render_target( | 
| +    GrContext* context, int sampleCnt, uint32_t color, GrBackendObject* outTexture) { | 
| +    const int kWidth = 10; | 
| +    const int kHeight = 10; | 
| +    SkAutoTDeleteArray<uint32_t> pixels(new uint32_t[kWidth * kHeight]); | 
| +    sk_memset32(pixels.get(), color, kWidth * kHeight); | 
| +    GrBackendTextureDesc desc; | 
| +    desc.fConfig = kRGBA_8888_GrPixelConfig; | 
| +    desc.fWidth = kWidth; | 
| +    desc.fHeight = kHeight; | 
| +    desc.fFlags = kRenderTarget_GrBackendTextureFlag; | 
| +    desc.fTextureHandle = context->getGpu()->createTestingOnlyBackendTexture( | 
| +        pixels.get(), kWidth, kHeight, kRGBA_8888_GrPixelConfig); | 
| +    desc.fSampleCnt = sampleCnt; | 
| +    sk_sp<SkSurface> surface = SkSurface::MakeFromBackendTextureAsRenderTarget(context, desc, | 
| +                                                                               nullptr); | 
| +    if (!surface) { | 
| +        context->getGpu()->deleteTestingOnlyBackendTexture(desc.fTextureHandle); | 
| +        return nullptr; | 
| +    } | 
| +    *outTexture = desc.fTextureHandle; | 
| +    return surface; | 
| +} | 
|  | 
| -void test_surface_clear(skiatest::Reporter* reporter, sk_sp<SkSurface> surface, | 
| -                        std::function<GrSurface*(SkSurface*)> grSurfaceGetter, | 
| -                        uint32_t expectedValue) { | 
| +static void test_surface_clear(skiatest::Reporter* reporter, sk_sp<SkSurface> surface, | 
| +                               std::function<GrSurface*(SkSurface*)> grSurfaceGetter, | 
| +                               uint32_t expectedValue) { | 
| if (!surface) { | 
| ERRORF(reporter, "Could not create GPU SkSurface."); | 
| return; | 
| @@ -831,7 +809,7 @@ void test_surface_clear(skiatest::Reporter* reporter, sk_sp<SkSurface> surface, | 
| int w = surface->width(); | 
| int h = surface->height(); | 
| SkAutoTDeleteArray<uint32_t> pixels(new uint32_t[w * h]); | 
| -    memset(pixels.get(), ~expectedValue, sizeof(uint32_t) * w * h); | 
| +    sk_memset32(pixels.get(), ~expectedValue, w * h); | 
|  | 
| SkAutoTUnref<GrSurface> grSurface(SkSafeRef(grSurfaceGetter(surface.get()))); | 
| if (!grSurface) { | 
| @@ -861,6 +839,7 @@ void test_surface_clear(skiatest::Reporter* reporter, sk_sp<SkSurface> surface, | 
|  | 
| DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfaceClear_Gpu, reporter, ctxInfo) { | 
| GrContext* context = ctxInfo.fGrContext; | 
| + | 
| std::function<GrSurface*(SkSurface*)> grSurfaceGetters[] = { | 
| [] (SkSurface* s){ | 
| GrDrawContext* dc = s->getCanvas()->internal_private_accessTopLayerDrawContext(); | 
| @@ -870,33 +849,119 @@ DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfaceClear_Gpu, reporter, ctxInfo) { | 
| s->getCanvas()->getDevice_just_for_deprecated_compatibility_testing(); | 
| return d->accessRenderTarget(); }, | 
| [] (SkSurface* s){ sk_sp<SkImage> i(s->makeImageSnapshot()); | 
| -                           return as_IB(i)->peekTexture(); }, | 
| +                           return as_IB(i)->peekTexture(); } | 
| }; | 
| + | 
| for (auto grSurfaceGetter : grSurfaceGetters) { | 
| +        // Test that non-wrapped RTs are created clear. | 
| for (auto& surface_func : {&create_gpu_surface, &create_gpu_scratch_surface}) { | 
| auto surface = surface_func(context, kPremul_SkAlphaType, nullptr); | 
| test_surface_clear(reporter, surface, grSurfaceGetter, 0x0); | 
| } | 
| // Wrapped RTs are *not* supposed to clear (to allow client to partially update a surface). | 
| -        static const int kWidth = 10; | 
| -        static const int kHeight = 10; | 
| -        SkAutoTDeleteArray<uint32_t> pixels(new uint32_t[kWidth * kHeight]); | 
| -        memset(pixels.get(), 0xAB, sizeof(uint32_t) * kWidth * kHeight); | 
| +        const uint32_t kOrigColor = 0xABABABAB; | 
| +        for (auto& surfaceFunc : {&create_gpu_surface_backend_texture, | 
| +                                  &create_gpu_surface_backend_texture_as_render_target}) { | 
| +            GrBackendObject textureObject; | 
| +            auto surface = surfaceFunc(context, 0, kOrigColor, &textureObject); | 
| +            test_surface_clear(reporter, surface, grSurfaceGetter, kOrigColor); | 
| +            surface.reset(); | 
| +            context->getGpu()->deleteTestingOnlyBackendTexture(textureObject); | 
| +        } | 
| +    } | 
| +} | 
| + | 
| +static void test_surface_draw_partially( | 
| +    skiatest::Reporter* reporter, sk_sp<SkSurface> surface, uint32_t origColor) { | 
| +    const int kW = surface->width(); | 
| +    const int kH = surface->height(); | 
| +    SkPaint paint; | 
| +    const SkColor kRectColor = ~origColor | 0xFF000000; | 
| +    paint.setColor(kRectColor); | 
| +    surface->getCanvas()->drawRect(SkRect::MakeWH(SkIntToScalar(kW), SkIntToScalar(kH)/2), | 
| +                                   paint); | 
| +    SkAutoTDeleteArray<uint32_t> pixels(new uint32_t[kW * kH]); | 
| +    sk_memset32(pixels.get(), ~origColor, kW * kH); | 
| +    // Read back RGBA to avoid format conversions that may not be supported on all platforms. | 
| +    SkImageInfo readInfo = SkImageInfo::Make(kW, kH, kRGBA_8888_SkColorType, kPremul_SkAlphaType); | 
| +    SkAssertResult(surface->readPixels(readInfo, pixels.get(), kW * sizeof(uint32_t), 0, 0)); | 
| +    bool stop = false; | 
| +    SkPMColor origColorPM = SkPackARGB_as_RGBA((origColor >> 24 & 0xFF), | 
| +                                               (origColor >>  0 & 0xFF), | 
| +                                               (origColor >>  8 & 0xFF), | 
| +                                               (origColor >> 16 & 0xFF)); | 
| +    SkPMColor rectColorPM = SkPackARGB_as_RGBA((kRectColor >> 24 & 0xFF), | 
| +                                               (kRectColor >> 16 & 0xFF), | 
| +                                               (kRectColor >>  8 & 0xFF), | 
| +                                               (kRectColor >>  0 & 0xFF)); | 
| +    for (int y = 0; y < kH/2 && !stop; ++y) { | 
| +       for (int x = 0; x < kW && !stop; ++x) { | 
| +            REPORTER_ASSERT(reporter, rectColorPM == pixels[x + y * kW]); | 
| +            if (rectColorPM != pixels[x + y * kW]) { | 
| +                stop = true; | 
| +            } | 
| +        } | 
| +    } | 
| +    stop = false; | 
| +    for (int y = kH/2; y < kH && !stop; ++y) { | 
| +        for (int x = 0; x < kW && !stop; ++x) { | 
| +            REPORTER_ASSERT(reporter, origColorPM == pixels[x + y * kW]); | 
| +            if (origColorPM != pixels[x + y * kW]) { | 
| +                stop = true; | 
| +            } | 
| +        } | 
| +    } | 
| +} | 
|  | 
| -        GrBackendObject textureObject = | 
| -                context->getGpu()->createTestingOnlyBackendTexture(pixels.get(), kWidth, kHeight, | 
| -                                                                   kRGBA_8888_GrPixelConfig); | 
| +DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfacePartialDraw_Gpu, reporter, ctxInfo) { | 
| +    GrGpu* gpu = ctxInfo.fGrContext->getGpu(); | 
| +    if (!gpu) { | 
| +        return; | 
| +    } | 
| +    static const uint32_t kOrigColor = 0xFFAABBCC; | 
|  | 
| -        GrBackendTextureDesc desc; | 
| -        desc.fConfig = kRGBA_8888_GrPixelConfig; | 
| -        desc.fWidth = kWidth; | 
| -        desc.fHeight = kHeight; | 
| -        desc.fFlags = kRenderTarget_GrBackendTextureFlag; | 
| -        desc.fTextureHandle = textureObject; | 
| +    for (auto& surfaceFunc : {&create_gpu_surface_backend_texture, | 
| +                              &create_gpu_surface_backend_texture_as_render_target}) { | 
| +        // Validate that we can draw to the canvas and that the original texture color is | 
| +        // preserved in pixels that aren't rendered to via the surface. | 
| +        // This works only for non-multisampled case. | 
| +        GrBackendObject textureObject; | 
| +        auto surface = surfaceFunc(ctxInfo.fGrContext, 0, kOrigColor, &textureObject); | 
| +        if (surface) { | 
| +            test_surface_draw_partially(reporter, surface, kOrigColor); | 
| +            surface.reset(); | 
| +            gpu->deleteTestingOnlyBackendTexture(textureObject); | 
| +        } | 
| +    } | 
| +} | 
|  | 
| -        auto surface = SkSurface::MakeFromBackendTexture(context, desc, nullptr); | 
| -        test_surface_clear(reporter, surface, grSurfaceGetter, 0xABABABAB); | 
| -        context->getGpu()->deleteTestingOnlyBackendTexture(textureObject); | 
| + | 
| +DEF_GPUTEST_FOR_GL_RENDERING_CONTEXTS(SurfaceAttachStencil_Gpu, reporter, ctxInfo) { | 
| +    GrGpu* gpu = ctxInfo.fGrContext->getGpu(); | 
| +    if (!gpu) { | 
| +        return; | 
| +    } | 
| +    static const uint32_t kOrigColor = 0xFFAABBCC; | 
| + | 
| +    for (auto& surfaceFunc : {&create_gpu_surface_backend_texture, | 
| +                              &create_gpu_surface_backend_texture_as_render_target}) { | 
| +        for (int sampleCnt : {0, 4, 8}) { | 
| +            GrBackendObject textureObject; | 
| +            auto surface = surfaceFunc(ctxInfo.fGrContext, sampleCnt, kOrigColor, &textureObject); | 
| + | 
| +            if (!surface && sampleCnt > 0) { | 
| +              // Certain platforms don't support MSAA, skip these. | 
| +              continue; | 
| +            } | 
| + | 
| +            // Validate that we can attach a stencil buffer to an SkSurface created by either of | 
| +            // our surface functions. | 
| +            GrRenderTarget* rt = surface->getCanvas()->internal_private_accessTopLayerDrawContext() | 
| +                ->accessRenderTarget(); | 
| +            REPORTER_ASSERT(reporter, | 
| +                            ctxInfo.fGrContext->resourceProvider()->attachStencilAttachment(rt)); | 
| +            gpu->deleteTestingOnlyBackendTexture(textureObject); | 
| +        } | 
| } | 
| } | 
| #endif | 
|  |