2017-06-21 19:55:46 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// This is a GPU-backend specific test. It relies on static intializers to work
|
|
|
|
|
|
|
|
#include "SkTypes.h"
|
|
|
|
|
|
|
|
#include "GrContextFactory.h"
|
|
|
|
#include "GrContextPriv.h"
|
|
|
|
#include "GrGpu.h"
|
|
|
|
#include "GrResourceProvider.h"
|
|
|
|
#include "GrSurfaceProxy.h"
|
|
|
|
#include "GrTexture.h"
|
|
|
|
#include "SkGr.h"
|
|
|
|
#include "SkSurface.h"
|
|
|
|
#include "Test.h"
|
|
|
|
|
|
|
|
using sk_gpu_test::GrContextFactory;
|
|
|
|
|
|
|
|
void fill_transfer_data(int left, int top, int width, int height, int bufferWidth,
|
|
|
|
GrColor* data) {
|
|
|
|
|
|
|
|
// build red-green gradient
|
|
|
|
for (int j = top; j < top + height; ++j) {
|
|
|
|
for (int i = left; i < left + width; ++i) {
|
|
|
|
unsigned int red = (unsigned int)(256.f*((i - left) / (float)width));
|
|
|
|
unsigned int green = (unsigned int)(256.f*((j - top) / (float)height));
|
|
|
|
data[i + j*bufferWidth] = GrColorPackRGBA(red - (red>>8),
|
|
|
|
green - (green>>8), 0xff, 0xff);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-08 15:59:07 +00:00
|
|
|
bool do_buffers_contain_same_values(const GrColor* bufferA,
|
|
|
|
const GrColor* bufferB,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
size_t rowBytesA,
|
2019-04-09 19:00:05 +00:00
|
|
|
size_t rowBytesB,
|
|
|
|
bool swiz) {
|
2017-06-21 19:55:46 +00:00
|
|
|
for (int j = 0; j < height; ++j) {
|
|
|
|
for (int i = 0; i < width; ++i) {
|
2019-04-09 19:00:05 +00:00
|
|
|
auto colorA = bufferA[i];
|
|
|
|
if (swiz) {
|
|
|
|
colorA = GrColorPackRGBA(GrColorUnpackB(colorA), GrColorUnpackG(colorA),
|
|
|
|
GrColorUnpackR(colorA), GrColorUnpackA(colorA));
|
|
|
|
}
|
|
|
|
if (colorA != bufferB[i]) {
|
2017-06-21 19:55:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-04-08 15:59:07 +00:00
|
|
|
bufferA = reinterpret_cast<const GrColor*>(reinterpret_cast<const char*>(bufferA) +
|
|
|
|
rowBytesA);
|
|
|
|
bufferB = reinterpret_cast<const GrColor*>(reinterpret_cast<const char*>(bufferB) +
|
|
|
|
rowBytesB);
|
2017-06-21 19:55:46 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-04-08 15:59:07 +00:00
|
|
|
void basic_transfer_to_test(skiatest::Reporter* reporter, GrContext* context, GrColorType colorType,
|
|
|
|
bool renderTarget) {
|
2019-02-04 18:26:26 +00:00
|
|
|
if (GrCaps::kNone_MapFlags == context->priv().caps()->mapBufferFlags()) {
|
2017-10-09 17:02:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-04 18:26:26 +00:00
|
|
|
auto resourceProvider = context->priv().resourceProvider();
|
|
|
|
GrGpu* gpu = context->priv().getGpu();
|
2018-01-16 20:07:54 +00:00
|
|
|
|
2017-06-21 19:55:46 +00:00
|
|
|
// set up the data
|
|
|
|
const int kTextureWidth = 16;
|
|
|
|
const int kTextureHeight = 16;
|
2018-12-07 15:57:26 +00:00
|
|
|
#ifdef SK_BUILD_FOR_IOS
|
|
|
|
// UNPACK_ROW_LENGTH is broken on iOS so rowBytes needs to match data width
|
2019-01-30 18:08:28 +00:00
|
|
|
const int kBufferWidth = GrBackendApi::kOpenGL == context->backend() ? 16 : 20;
|
2018-12-07 15:57:26 +00:00
|
|
|
#else
|
2017-06-21 19:55:46 +00:00
|
|
|
const int kBufferWidth = 20;
|
2018-12-07 15:57:26 +00:00
|
|
|
#endif
|
2017-06-21 19:55:46 +00:00
|
|
|
const int kBufferHeight = 16;
|
|
|
|
size_t rowBytes = kBufferWidth * sizeof(GrColor);
|
|
|
|
SkAutoTMalloc<GrColor> srcBuffer(kBufferWidth*kBufferHeight);
|
|
|
|
SkAutoTMalloc<GrColor> dstBuffer(kBufferWidth*kBufferHeight);
|
|
|
|
|
|
|
|
fill_transfer_data(0, 0, kTextureWidth, kTextureHeight, kBufferWidth, srcBuffer.get());
|
|
|
|
|
|
|
|
// create and fill transfer buffer
|
|
|
|
size_t size = rowBytes*kBufferHeight;
|
2019-02-07 16:31:24 +00:00
|
|
|
sk_sp<GrGpuBuffer> buffer(resourceProvider->createBuffer(size, GrGpuBufferType::kXferCpuToGpu,
|
|
|
|
kDynamic_GrAccessPattern));
|
2017-06-21 19:55:46 +00:00
|
|
|
if (!buffer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void* data = buffer->map();
|
|
|
|
memcpy(data, srcBuffer.get(), size);
|
|
|
|
buffer->unmap();
|
|
|
|
|
2018-02-20 19:05:36 +00:00
|
|
|
for (auto srgbEncoding : {GrSRGBEncoded::kNo, GrSRGBEncoded::kYes}) {
|
|
|
|
// create texture
|
|
|
|
GrSurfaceDesc desc;
|
|
|
|
desc.fFlags = renderTarget ? kRenderTarget_GrSurfaceFlag : kNone_GrSurfaceFlags;
|
|
|
|
desc.fWidth = kTextureWidth;
|
|
|
|
desc.fHeight = kTextureHeight;
|
|
|
|
desc.fConfig = GrColorTypeToPixelConfig(colorType, srgbEncoding);
|
|
|
|
desc.fSampleCnt = 1;
|
|
|
|
|
|
|
|
if (kUnknown_GrPixelConfig == desc.fConfig) {
|
|
|
|
SkASSERT(GrSRGBEncoded::kYes == srgbEncoding);
|
|
|
|
continue;
|
|
|
|
}
|
2017-06-21 19:55:46 +00:00
|
|
|
|
2019-02-04 18:26:26 +00:00
|
|
|
if (!context->priv().caps()->isConfigTexturable(desc.fConfig) ||
|
|
|
|
(renderTarget && !context->priv().caps()->isConfigRenderable(desc.fConfig))) {
|
2018-02-20 19:05:36 +00:00
|
|
|
continue;
|
|
|
|
}
|
2017-06-21 19:55:46 +00:00
|
|
|
|
2019-04-09 22:41:27 +00:00
|
|
|
sk_sp<GrTexture> tex = resourceProvider->createTexture(
|
|
|
|
desc, SkBudgeted::kNo, GrResourceProvider::Flags::kNoPendingIO);
|
2018-07-27 16:21:37 +00:00
|
|
|
if (!tex) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-02-20 19:05:36 +00:00
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
// transfer full data
|
|
|
|
|
|
|
|
bool result;
|
2019-04-08 15:59:07 +00:00
|
|
|
result = gpu->transferPixelsTo(tex.get(), 0, 0, kTextureWidth, kTextureHeight, colorType,
|
|
|
|
buffer.get(), 0, rowBytes);
|
2018-02-20 19:05:36 +00:00
|
|
|
REPORTER_ASSERT(reporter, result);
|
|
|
|
|
|
|
|
memset(dstBuffer.get(), 0xCDCD, size);
|
2018-06-01 19:33:20 +00:00
|
|
|
result = gpu->readPixels(tex.get(), 0, 0, kTextureWidth, kTextureHeight, colorType,
|
2018-02-20 19:05:36 +00:00
|
|
|
dstBuffer.get(), rowBytes);
|
|
|
|
if (result) {
|
2019-04-08 15:59:07 +00:00
|
|
|
REPORTER_ASSERT(reporter, do_buffers_contain_same_values(srcBuffer,
|
|
|
|
dstBuffer,
|
|
|
|
kTextureWidth,
|
|
|
|
kTextureHeight,
|
|
|
|
rowBytes,
|
2019-04-09 19:00:05 +00:00
|
|
|
rowBytes,
|
|
|
|
false));
|
2018-02-20 19:05:36 +00:00
|
|
|
}
|
2017-06-21 19:55:46 +00:00
|
|
|
|
2018-02-20 19:05:36 +00:00
|
|
|
//////////////////////////
|
|
|
|
// transfer partial data
|
2018-12-07 15:57:26 +00:00
|
|
|
#ifdef SK_BUILD_FOR_IOS
|
|
|
|
// UNPACK_ROW_LENGTH is broken on iOS so we can't do partial transfers
|
2019-01-30 18:08:28 +00:00
|
|
|
if (GrBackendApi::kOpenGL == context->backend()) {
|
2018-12-07 15:57:26 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
#endif
|
2018-02-20 19:05:36 +00:00
|
|
|
const int kLeft = 2;
|
|
|
|
const int kTop = 10;
|
|
|
|
const int kWidth = 10;
|
|
|
|
const int kHeight = 2;
|
|
|
|
|
|
|
|
// change color of subrectangle
|
|
|
|
fill_transfer_data(kLeft, kTop, kWidth, kHeight, kBufferWidth, srcBuffer.get());
|
|
|
|
data = buffer->map();
|
|
|
|
memcpy(data, srcBuffer.get(), size);
|
|
|
|
buffer->unmap();
|
|
|
|
|
|
|
|
size_t offset = sizeof(GrColor) * (kTop * kBufferWidth + kLeft);
|
2019-04-08 15:59:07 +00:00
|
|
|
result = gpu->transferPixelsTo(tex.get(), kLeft, kTop, kWidth, kHeight, colorType,
|
|
|
|
buffer.get(), offset, rowBytes);
|
2018-02-20 19:05:36 +00:00
|
|
|
REPORTER_ASSERT(reporter, result);
|
|
|
|
|
|
|
|
memset(dstBuffer.get(), 0xCDCD, size);
|
2018-06-01 19:33:20 +00:00
|
|
|
result = gpu->readPixels(tex.get(), 0, 0, kTextureWidth, kTextureHeight, colorType,
|
2018-02-20 19:05:36 +00:00
|
|
|
dstBuffer.get(), rowBytes);
|
|
|
|
if (result) {
|
2019-04-08 15:59:07 +00:00
|
|
|
REPORTER_ASSERT(reporter, do_buffers_contain_same_values(srcBuffer,
|
|
|
|
dstBuffer,
|
|
|
|
kTextureWidth,
|
|
|
|
kTextureHeight,
|
|
|
|
rowBytes,
|
2019-04-09 19:00:05 +00:00
|
|
|
rowBytes,
|
|
|
|
false));
|
2018-02-20 19:05:36 +00:00
|
|
|
}
|
2017-06-27 14:36:56 +00:00
|
|
|
}
|
2017-06-21 19:55:46 +00:00
|
|
|
}
|
|
|
|
|
2019-04-09 19:00:05 +00:00
|
|
|
void basic_transfer_from_test(skiatest::Reporter* reporter, const sk_gpu_test::ContextInfo& ctxInfo,
|
2019-04-08 15:59:07 +00:00
|
|
|
GrColorType colorType, bool renderTarget) {
|
2019-04-09 19:00:05 +00:00
|
|
|
auto context = ctxInfo.grContext();
|
2019-04-08 15:59:07 +00:00
|
|
|
if (GrCaps::kNone_MapFlags == context->priv().caps()->mapBufferFlags()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-04-09 19:00:05 +00:00
|
|
|
// On OpenGL ES it may not be possible to read back in to BGRA becagse GL_RGBA/GL_UNSIGNED_BYTE
|
|
|
|
// may be the only allowed format/type params to glReadPixels. So read back into GL_RGBA.
|
|
|
|
// TODO(bsalomon): Make this work in GrGLGpu.
|
|
|
|
auto readColorType = colorType;
|
|
|
|
if (GrColorType::kBGRA_8888 == colorType &&
|
|
|
|
ctxInfo.type() == sk_gpu_test::GrContextFactory::kGLES_ContextType) {
|
|
|
|
readColorType = GrColorType::kRGBA_8888;
|
|
|
|
}
|
|
|
|
|
2019-04-08 15:59:07 +00:00
|
|
|
auto resourceProvider = context->priv().resourceProvider();
|
|
|
|
GrGpu* gpu = context->priv().getGpu();
|
|
|
|
|
|
|
|
const int kTextureWidth = 16;
|
|
|
|
const int kTextureHeight = 16;
|
|
|
|
|
|
|
|
// We'll do a full texture read into the buffer followed by a partial read. These values
|
|
|
|
// describe the partial read subrect.
|
|
|
|
const int kPartialLeft = 2;
|
|
|
|
const int kPartialTop = 10;
|
|
|
|
const int kPartialWidth = 10;
|
|
|
|
const int kPartialHeight = 2;
|
|
|
|
|
2019-04-10 16:14:26 +00:00
|
|
|
size_t bpp = GrColorTypeBytesPerPixel(readColorType);
|
|
|
|
size_t fullBufferRowBytes = kTextureWidth * bpp;
|
|
|
|
size_t partialBufferRowBytes = kPartialWidth * bpp;
|
|
|
|
size_t offsetAlignment = context->priv().caps()->transferFromOffsetAlignment(readColorType);
|
|
|
|
SkASSERT(offsetAlignment);
|
2019-04-08 15:59:07 +00:00
|
|
|
|
|
|
|
size_t bufferSize = fullBufferRowBytes * kTextureHeight;
|
|
|
|
// Arbitrary starting offset for the partial read.
|
2019-04-10 16:14:26 +00:00
|
|
|
size_t partialReadOffset = GrSizeAlignUp(11, offsetAlignment);
|
|
|
|
bufferSize = SkTMax(bufferSize, partialReadOffset + partialBufferRowBytes * kPartialHeight);
|
2019-04-08 15:59:07 +00:00
|
|
|
|
|
|
|
sk_sp<GrGpuBuffer> buffer(resourceProvider->createBuffer(
|
|
|
|
bufferSize, GrGpuBufferType::kXferGpuToCpu, kDynamic_GrAccessPattern));
|
|
|
|
REPORTER_ASSERT(reporter, buffer);
|
|
|
|
if (!buffer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int expectedTransferCnt = 0;
|
|
|
|
gpu->stats()->reset();
|
|
|
|
for (auto srgbEncoding : {GrSRGBEncoded::kNo, GrSRGBEncoded::kYes}) {
|
|
|
|
// create texture
|
|
|
|
GrSurfaceDesc desc;
|
|
|
|
desc.fFlags = renderTarget ? kRenderTarget_GrSurfaceFlag : kNone_GrSurfaceFlags;
|
|
|
|
desc.fWidth = kTextureWidth;
|
|
|
|
desc.fHeight = kTextureHeight;
|
|
|
|
desc.fConfig = GrColorTypeToPixelConfig(colorType, srgbEncoding);
|
|
|
|
desc.fSampleCnt = 1;
|
|
|
|
|
|
|
|
if (kUnknown_GrPixelConfig == desc.fConfig) {
|
|
|
|
SkASSERT(GrSRGBEncoded::kYes == srgbEncoding);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!context->priv().caps()->isConfigTexturable(desc.fConfig) ||
|
|
|
|
(renderTarget && !context->priv().caps()->isConfigRenderable(desc.fConfig))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkAutoTMalloc<GrColor> textureData(kTextureWidth * kTextureHeight);
|
|
|
|
size_t textureDataRowBytes = kTextureWidth * sizeof(GrColor);
|
|
|
|
fill_transfer_data(0, 0, kTextureWidth, kTextureHeight, kTextureWidth, textureData.get());
|
|
|
|
GrMipLevel data;
|
|
|
|
data.fPixels = textureData.get();
|
|
|
|
data.fRowBytes = kTextureWidth * sizeof(GrColor);
|
|
|
|
sk_sp<GrTexture> tex = resourceProvider->createTexture(desc, SkBudgeted::kNo, &data, 1);
|
|
|
|
if (!tex) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
// transfer full data
|
2019-04-10 16:14:26 +00:00
|
|
|
bool result = gpu->transferPixelsFrom(tex.get(), 0, 0, kTextureWidth, kTextureHeight,
|
|
|
|
readColorType, buffer.get(), 0);
|
|
|
|
if (!result) {
|
|
|
|
ERRORF(reporter, "transferPixelsFrom failed.");
|
2019-04-08 15:59:07 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++expectedTransferCnt;
|
|
|
|
|
|
|
|
// TODO(bsalomon): caps to know if the map() is synchronous and skip the flush if so.
|
|
|
|
gpu->finishFlush(nullptr, SkSurface::BackendSurfaceAccess::kNoAccess,
|
2019-04-12 18:24:55 +00:00
|
|
|
kSyncCpu_GrFlushFlag, 0, nullptr, nullptr, nullptr);
|
2019-04-08 15:59:07 +00:00
|
|
|
|
|
|
|
const auto* map = reinterpret_cast<const GrColor*>(buffer->map());
|
|
|
|
REPORTER_ASSERT(reporter, map);
|
|
|
|
if (!map) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(reporter, do_buffers_contain_same_values(textureData.get(),
|
|
|
|
map,
|
|
|
|
kTextureWidth,
|
|
|
|
kTextureHeight,
|
|
|
|
textureDataRowBytes,
|
2019-04-10 16:14:26 +00:00
|
|
|
fullBufferRowBytes,
|
2019-04-09 19:00:05 +00:00
|
|
|
readColorType != colorType));
|
2019-04-08 15:59:07 +00:00
|
|
|
buffer->unmap();
|
|
|
|
|
2019-04-09 19:00:05 +00:00
|
|
|
///////////////////////
|
2019-04-08 15:59:07 +00:00
|
|
|
// Now test a partial read at an offset into the buffer.
|
2019-04-10 16:14:26 +00:00
|
|
|
result = gpu->transferPixelsFrom(tex.get(), kPartialLeft, kPartialTop, kPartialWidth,
|
|
|
|
kPartialHeight, readColorType, buffer.get(),
|
|
|
|
partialReadOffset);
|
|
|
|
if (!result) {
|
|
|
|
ERRORF(reporter, "transferPixelsFrom failed.");
|
2019-04-08 15:59:07 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++expectedTransferCnt;
|
|
|
|
|
|
|
|
// TODO(bsalomon): caps to know if the map() is synchronous and skip the flush if so.
|
|
|
|
gpu->finishFlush(nullptr, SkSurface::BackendSurfaceAccess::kNoAccess,
|
2019-04-12 18:24:55 +00:00
|
|
|
kSyncCpu_GrFlushFlag, 0, nullptr, nullptr, nullptr);
|
2019-04-08 15:59:07 +00:00
|
|
|
|
|
|
|
map = reinterpret_cast<const GrColor*>(buffer->map());
|
|
|
|
REPORTER_ASSERT(reporter, map);
|
|
|
|
if (!map) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const GrColor* textureDataStart = reinterpret_cast<const GrColor*>(
|
|
|
|
reinterpret_cast<const char*>(textureData.get()) +
|
|
|
|
textureDataRowBytes * kPartialTop + sizeof(GrColor) * kPartialLeft);
|
|
|
|
const GrColor* bufferStart = reinterpret_cast<const GrColor*>(
|
|
|
|
reinterpret_cast<const char*>(map) + partialReadOffset);
|
|
|
|
REPORTER_ASSERT(reporter, do_buffers_contain_same_values(textureDataStart,
|
|
|
|
bufferStart,
|
|
|
|
kPartialWidth,
|
|
|
|
kPartialHeight,
|
|
|
|
textureDataRowBytes,
|
2019-04-10 16:14:26 +00:00
|
|
|
partialBufferRowBytes,
|
2019-04-09 19:00:05 +00:00
|
|
|
readColorType != colorType));
|
2019-04-08 15:59:07 +00:00
|
|
|
buffer->unmap();
|
|
|
|
}
|
|
|
|
#if GR_GPU_STATS
|
|
|
|
REPORTER_ASSERT(reporter, gpu->stats()->transfersFromSurface() == expectedTransferCnt);
|
|
|
|
#else
|
|
|
|
(void)expectedTransferCnt;
|
|
|
|
#endif
|
|
|
|
}
|
2019-03-18 19:39:22 +00:00
|
|
|
|
2019-04-08 15:59:07 +00:00
|
|
|
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(TransferPixelsToTest, reporter, ctxInfo) {
|
|
|
|
if (!ctxInfo.grContext()->priv().caps()->transferBufferSupport()) {
|
|
|
|
return;
|
|
|
|
}
|
2017-06-21 19:55:46 +00:00
|
|
|
// RGBA
|
2019-04-08 15:59:07 +00:00
|
|
|
basic_transfer_to_test(reporter, ctxInfo.grContext(), GrColorType::kRGBA_8888, false);
|
|
|
|
basic_transfer_to_test(reporter, ctxInfo.grContext(), GrColorType::kRGBA_8888, true);
|
2017-06-21 19:55:46 +00:00
|
|
|
|
|
|
|
// BGRA
|
2019-04-08 15:59:07 +00:00
|
|
|
basic_transfer_to_test(reporter, ctxInfo.grContext(), GrColorType::kBGRA_8888, false);
|
|
|
|
basic_transfer_to_test(reporter, ctxInfo.grContext(), GrColorType::kBGRA_8888, true);
|
2017-06-21 19:55:46 +00:00
|
|
|
}
|
2019-03-18 19:39:22 +00:00
|
|
|
|
2019-04-09 18:57:00 +00:00
|
|
|
// TODO(bsalomon): Metal
|
|
|
|
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(TransferPixelsFromTest, reporter, ctxInfo) {
|
2019-04-08 15:59:07 +00:00
|
|
|
if (!ctxInfo.grContext()->priv().caps()->transferBufferSupport()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// RGBA
|
2019-04-09 19:00:05 +00:00
|
|
|
basic_transfer_from_test(reporter, ctxInfo, GrColorType::kRGBA_8888, false);
|
|
|
|
basic_transfer_from_test(reporter, ctxInfo, GrColorType::kRGBA_8888, true);
|
2019-04-08 15:59:07 +00:00
|
|
|
|
|
|
|
// BGRA
|
2019-04-09 19:00:05 +00:00
|
|
|
basic_transfer_from_test(reporter, ctxInfo, GrColorType::kBGRA_8888, false);
|
|
|
|
basic_transfer_from_test(reporter, ctxInfo, GrColorType::kBGRA_8888, true);
|
2019-04-08 15:59:07 +00:00
|
|
|
}
|