chromium/components/viz/service/display_embedder/software_output_device_mac_unittest.mm

// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "components/viz/service/display_embedder/software_output_device_mac.h"

#include "base/task/sequenced_task_runner.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/geometry/skia_conversions.h"

namespace viz {

namespace {

TEST(SoftwareOutputDeviceMacTest, Basics) {
  auto device = std::make_unique<SoftwareOutputDeviceMac>(
      base::SequencedTaskRunner::GetCurrentDefault());
  gfx::Size pixel_size(512, 512);
  float scale_factor = 1;

  EXPECT_EQ(device->BufferQueueSizeForTesting(), 0u);
  device->Resize(pixel_size, scale_factor);

  // Frame 0.
  gfx::Rect damage0(pixel_size);
  device->BeginPaint(damage0);
  IOSurfaceRef io_surface0 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();

  // Frame 1.
  // We didn't set the IOSurface in use, so it should be re-used, and we should
  // have no copy.
  gfx::Rect damage1(10, 10, 10, 10);
  device->BeginPaint(damage1);
  IOSurfaceRef io_surface1 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();
  EXPECT_EQ(io_surface0, io_surface1);
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 1u);
  EXPECT_TRUE(device->LastCopyRegionForTesting().isEmpty());

  // Frame 2.
  // The IOSurface is in use, so we should allocate a new one. We'll do a full
  // copy because it's a new buffer.
  IOSurfaceIncrementUseCount(io_surface1);
  gfx::Rect damage2(20, 20, 10, 10);
  device->BeginPaint(damage2);
  IOSurfaceRef io_surface2 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();
  EXPECT_NE(io_surface1, io_surface2);
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 2u);
  SkRegion copy_region2(gfx::RectToSkIRect(gfx::Rect(pixel_size)));
  copy_region2.op(gfx::RectToSkIRect(damage2), SkRegion::kDifference_Op);
  EXPECT_EQ(device->LastCopyRegionForTesting(), copy_region2);

  // Frame 3.
  // Both IOSurfaces are in use, so we'll allocate yet a new one and do another
  // full copy.
  IOSurfaceIncrementUseCount(io_surface2);
  gfx::Rect damage3(30, 30, 10, 10);
  device->BeginPaint(damage3);
  IOSurfaceRef io_surface3 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();
  EXPECT_NE(io_surface1, io_surface3);
  EXPECT_NE(io_surface2, io_surface3);
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 3u);
  SkRegion copy_region3(gfx::RectToSkIRect(gfx::Rect(pixel_size)));
  copy_region3.op(gfx::RectToSkIRect(damage3), SkRegion::kDifference_Op);
  EXPECT_EQ(device->LastCopyRegionForTesting(), copy_region3);

  // Frame 4.
  // The IOSurface from frame1 is free, so we should re-use it. We should be
  // copying the damage from frame2 and frame3.
  IOSurfaceIncrementUseCount(io_surface3);
  IOSurfaceDecrementUseCount(io_surface1);
  gfx::Rect damage4(35, 35, 15, 15);
  device->BeginPaint(damage4);
  IOSurfaceRef io_surface4 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();
  EXPECT_EQ(io_surface1, io_surface4);
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 3u);
  SkRegion copy_region4;
  copy_region4.op(gfx::RectToSkIRect(damage2), SkRegion::kUnion_Op);
  copy_region4.op(gfx::RectToSkIRect(damage3), SkRegion::kUnion_Op);
  copy_region4.op(gfx::RectToSkIRect(damage4), SkRegion::kDifference_Op);
  EXPECT_EQ(device->LastCopyRegionForTesting(), copy_region4);

  // Frame 5.
  // All IOSurfaces are allocated, allocate another.
  IOSurfaceIncrementUseCount(io_surface4);
  gfx::Rect damage5(50, 50, 10, 10);
  device->BeginPaint(damage5);
  IOSurfaceRef io_surface5 = device->CurrentPaintIOSurfaceForTesting();
  EXPECT_NE(io_surface5, io_surface2);
  EXPECT_NE(io_surface5, io_surface3);
  EXPECT_NE(io_surface5, io_surface4);
  device->EndPaint();
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 4u);

  // Frame 6.
  // All IOSurfaces are in use, allocate another, but free the one from frame2
  // (add an extra retain and check to retain count to verify that it steps
  // down).
  IOSurfaceIncrementUseCount(io_surface5);
  CFRetain(io_surface2);
  EXPECT_EQ(CFGetRetainCount(io_surface2), 2u);
  gfx::Rect damage6(60, 60, 10, 10);
  device->BeginPaint(damage6);
  IOSurfaceRef io_surface6 = device->CurrentPaintIOSurfaceForTesting();
  device->EndPaint();
  EXPECT_EQ(device->BufferQueueSizeForTesting(), 4u);
  EXPECT_NE(io_surface6, io_surface2);
  EXPECT_NE(io_surface6, io_surface3);
  EXPECT_NE(io_surface6, io_surface4);
  EXPECT_NE(io_surface6, io_surface5);
  EXPECT_EQ(CFGetRetainCount(io_surface2), 1u);
  CFRelease(io_surface2);
}

}  // namespace

}  // namespace viz