2020-09-23 16:00:53 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2020 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2020-12-23 15:11:33 +00:00
|
|
|
#include "include/core/SkBitmap.h"
|
2020-09-23 16:00:53 +00:00
|
|
|
#include "include/core/SkCanvas.h"
|
2022-05-16 15:59:11 +00:00
|
|
|
#include "include/core/SkColorSpace.h"
|
2020-09-23 16:00:53 +00:00
|
|
|
#include "include/core/SkSurface.h"
|
|
|
|
#include "include/core/SkTextBlob.h"
|
2022-05-17 15:14:00 +00:00
|
|
|
#include "src/core/SkDevice.h"
|
2022-06-15 21:58:25 +00:00
|
|
|
#include "src/core/SkGlyphRun.h"
|
2020-09-23 16:00:53 +00:00
|
|
|
#include "src/core/SkSurfacePriv.h"
|
2022-05-16 15:59:11 +00:00
|
|
|
#include "src/gpu/ganesh/GrColorInfo.h"
|
2022-05-23 15:34:17 +00:00
|
|
|
#include "src/text/gpu/TextBlob.h"
|
2020-09-23 16:00:53 +00:00
|
|
|
#include "tests/Test.h"
|
|
|
|
#include "tools/ToolUtils.h"
|
|
|
|
|
2022-05-05 15:08:03 +00:00
|
|
|
using BagOfBytes = sktext::gpu::BagOfBytes;
|
|
|
|
using SubRunAllocator = sktext::gpu::SubRunAllocator;
|
|
|
|
|
2020-09-23 16:00:53 +00:00
|
|
|
SkBitmap rasterize_blob(SkTextBlob* blob,
|
|
|
|
const SkPaint& paint,
|
|
|
|
GrRecordingContext* rContext,
|
|
|
|
const SkMatrix& matrix) {
|
|
|
|
const SkImageInfo info =
|
|
|
|
SkImageInfo::Make(500, 500, kN32_SkColorType, kPremul_SkAlphaType);
|
|
|
|
auto surface = SkSurface::MakeRenderTarget(rContext, SkBudgeted::kNo, info);
|
|
|
|
auto canvas = surface->getCanvas();
|
|
|
|
canvas->drawColor(SK_ColorWHITE);
|
|
|
|
canvas->concat(matrix);
|
|
|
|
canvas->drawTextBlob(blob, 10, 250, paint);
|
|
|
|
SkBitmap bitmap;
|
|
|
|
bitmap.allocN32Pixels(500, 500);
|
|
|
|
surface->readPixels(bitmap, 0, 0);
|
|
|
|
return bitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool check_for_black(const SkBitmap& bm) {
|
|
|
|
for (int y = 0; y < bm.height(); y++) {
|
|
|
|
for (int x = 0; x < bm.width(); x++) {
|
|
|
|
if (bm.getColor(x, y) == SK_ColorBLACK) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(GrTextBlobScaleAnimation, reporter, ctxInfo) {
|
|
|
|
auto tf = ToolUtils::create_portable_typeface("Mono", SkFontStyle());
|
|
|
|
SkFont font{tf};
|
|
|
|
font.setHinting(SkFontHinting::kNormal);
|
|
|
|
font.setSize(12);
|
|
|
|
font.setEdging(SkFont::Edging::kAntiAlias);
|
|
|
|
font.setSubpixel(true);
|
|
|
|
|
|
|
|
SkTextBlobBuilder builder;
|
|
|
|
const auto& runBuffer = builder.allocRunPosH(font, 30, 0, nullptr);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; i++) {
|
|
|
|
runBuffer.glyphs[i] = static_cast<SkGlyphID>(i);
|
|
|
|
runBuffer.pos[i] = SkIntToScalar(i);
|
|
|
|
}
|
|
|
|
auto blob = builder.make();
|
|
|
|
|
|
|
|
auto dContext = ctxInfo.directContext();
|
|
|
|
bool anyBlack = false;
|
|
|
|
for (int n = -13; n < 5; n++) {
|
|
|
|
SkMatrix m = SkMatrix::Scale(std::exp2(n), std::exp2(n));
|
|
|
|
auto bm = rasterize_blob(blob.get(), SkPaint(), dContext, m);
|
|
|
|
anyBlack |= check_for_black(bm);
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(reporter, anyBlack);
|
|
|
|
}
|
2020-11-06 15:21:59 +00:00
|
|
|
|
|
|
|
// Test extreme positions for all combinations of positions, origins, and translation matrices.
|
|
|
|
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(GrTextBlobMoveAround, reporter, ctxInfo) {
|
|
|
|
auto tf = ToolUtils::create_portable_typeface("Mono", SkFontStyle());
|
|
|
|
SkFont font{tf};
|
|
|
|
font.setHinting(SkFontHinting::kNormal);
|
|
|
|
font.setSize(12);
|
|
|
|
font.setEdging(SkFont::Edging::kAntiAlias);
|
|
|
|
font.setSubpixel(true);
|
|
|
|
|
|
|
|
auto makeBlob = [&](SkPoint delta) {
|
|
|
|
SkTextBlobBuilder builder;
|
|
|
|
const auto& runBuffer = builder.allocRunPos(font, 30, nullptr);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; i++) {
|
|
|
|
runBuffer.glyphs[i] = static_cast<SkGlyphID>(i);
|
|
|
|
runBuffer.points()[i] = SkPoint::Make(SkIntToScalar(i*10) + delta.x(), 50 + delta.y());
|
|
|
|
}
|
|
|
|
return builder.make();
|
|
|
|
};
|
|
|
|
|
|
|
|
auto dContext = ctxInfo.directContext();
|
|
|
|
auto rasterizeBlob = [&](SkTextBlob* blob, SkPoint origin, const SkMatrix& matrix) {
|
|
|
|
SkPaint paint;
|
|
|
|
const SkImageInfo info =
|
|
|
|
SkImageInfo::Make(350, 80, kN32_SkColorType, kPremul_SkAlphaType);
|
|
|
|
auto surface = SkSurface::MakeRenderTarget(dContext, SkBudgeted::kNo, info);
|
|
|
|
auto canvas = surface->getCanvas();
|
|
|
|
canvas->drawColor(SK_ColorWHITE);
|
|
|
|
canvas->concat(matrix);
|
|
|
|
canvas->drawTextBlob(blob, 10 + origin.x(), 40 + origin.y(), paint);
|
|
|
|
SkBitmap bitmap;
|
|
|
|
bitmap.allocN32Pixels(350, 80);
|
|
|
|
surface->readPixels(bitmap, 0, 0);
|
|
|
|
return bitmap;
|
|
|
|
};
|
|
|
|
|
|
|
|
SkBitmap benchMark;
|
|
|
|
{
|
|
|
|
auto blob = makeBlob({0, 0});
|
|
|
|
benchMark = rasterizeBlob(blob.get(), {0,0}, SkMatrix::I());
|
|
|
|
}
|
|
|
|
|
|
|
|
auto checkBitmap = [&](const SkBitmap& bitmap) {
|
|
|
|
REPORTER_ASSERT(reporter, benchMark.width() == bitmap.width());
|
|
|
|
REPORTER_ASSERT(reporter, benchMark.width() == bitmap.width());
|
|
|
|
|
|
|
|
for (int y = 0; y < benchMark.height(); y++) {
|
|
|
|
for (int x = 0; x < benchMark.width(); x++) {
|
|
|
|
if (benchMark.getColor(x, y) != bitmap.getColor(x, y)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
SkScalar interestingNumbers[] = {-10'000'000, -1'000'000, -1, 0, +1, +1'000'000, +10'000'000};
|
|
|
|
for (auto originX : interestingNumbers) {
|
|
|
|
for (auto originY : interestingNumbers) {
|
|
|
|
for (auto translateX : interestingNumbers) {
|
|
|
|
for (auto translateY : interestingNumbers) {
|
|
|
|
// Make sure everything adds to zero.
|
|
|
|
SkScalar deltaPosX = -(originX + translateX);
|
|
|
|
SkScalar deltaPosY = -(originY + translateY);
|
|
|
|
auto blob = makeBlob({deltaPosX, deltaPosY});
|
|
|
|
SkMatrix t = SkMatrix::Translate(translateX, translateY);
|
|
|
|
auto bitmap = rasterizeBlob(blob.get(), {originX, originY}, t);
|
|
|
|
REPORTER_ASSERT(reporter, checkBitmap(bitmap));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-08 22:57:03 +00:00
|
|
|
|
2022-05-05 15:08:03 +00:00
|
|
|
DEF_TEST(BagOfBytesBasic, r) {
|
2021-02-08 22:57:03 +00:00
|
|
|
const int k4K = 1 << 12;
|
|
|
|
{
|
|
|
|
// GrBagOfBytes::MinimumSizeWithOverhead(-1); // This should fail
|
2022-05-05 15:08:03 +00:00
|
|
|
BagOfBytes::PlatformMinimumSizeWithOverhead(0, 16);
|
|
|
|
BagOfBytes::PlatformMinimumSizeWithOverhead(
|
2021-02-08 22:57:03 +00:00
|
|
|
std::numeric_limits<int>::max() - k4K - 1, 16);
|
|
|
|
// GrBagOfBytes::MinimumSizeWithOverhead(std::numeric_limits<int>::max() - k4K); // Fail
|
2022-05-05 15:08:03 +00:00
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(0, 1, 16, 16) == 31);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(1, 1, 16, 16) == 32);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(63, 1, 16, 16) == 94);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(0, 8, 16, 16) == 24);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(1, 8, 16, 16) == 32);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(63, 8, 16, 16) == 88);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(0, 16, 16, 16) == 16);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(1, 16, 16, 16) == 32);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(63, 16, 16, 16) == 80);
|
|
|
|
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(0, 1, 8, 16) == 23);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(1, 1, 8, 16) == 24);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(63, 1, 8, 16) == 86);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(0, 8, 8, 16) == 16);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(1, 8, 8, 16) == 24);
|
|
|
|
REPORTER_ASSERT(r, BagOfBytes::MinimumSizeWithOverhead(63, 8, 8, 16) == 80);
|
2021-02-08 22:57:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2022-05-05 15:08:03 +00:00
|
|
|
BagOfBytes bob;
|
2021-02-08 22:57:03 +00:00
|
|
|
// bob.alignedBytes(0, 1); // This should fail
|
|
|
|
// bob.alignedBytes(1, 0); // This should fail
|
|
|
|
// bob.alignedBytes(1, 3); // This should fail
|
|
|
|
|
|
|
|
struct Big {
|
|
|
|
char stuff[std::numeric_limits<int>::max()];
|
|
|
|
};
|
|
|
|
// bob.alignedBytes(sizeof(Big), 1); // this should fail
|
|
|
|
// bob.allocateBytesFor<Big>(); // this should not compile
|
|
|
|
// The following should run, but should not be regularly tested.
|
|
|
|
// bob.allocateBytesFor<int>((std::numeric_limits<int>::max() - (1<<12)) / sizeof(int) - 1);
|
|
|
|
// The following should fail
|
|
|
|
// bob.allocateBytesFor<int>((std::numeric_limits<int>::max() - (1<<12)) / sizeof(int));
|
|
|
|
bob.alignedBytes(1, 1); // To avoid unused variable problems.
|
|
|
|
}
|
|
|
|
|
|
|
|
// Force multiple block allocation
|
|
|
|
{
|
2022-05-05 15:08:03 +00:00
|
|
|
BagOfBytes bob;
|
2021-02-08 22:57:03 +00:00
|
|
|
const int k64K = 1 << 16;
|
|
|
|
// By default allocation block sizes start at 1K and go up with fib. This should allocate
|
|
|
|
// 10 individual blocks.
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
bob.alignedBytes(k64K, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-12-16 15:45:50 +00:00
|
|
|
|
|
|
|
// Helper for defining allocators with inline/reserved storage.
|
2022-05-05 15:08:03 +00:00
|
|
|
// For argument declarations, stick to the base type (SubRunAllocator).
|
2020-12-16 15:45:50 +00:00
|
|
|
// Note: Inheriting from the storage first means the storage will outlive the
|
2022-05-05 15:08:03 +00:00
|
|
|
// SubRunAllocator, letting ~SubRunAllocator read it as it calls destructors.
|
2020-12-16 15:45:50 +00:00
|
|
|
// (This is mostly only relevant for strict tools like MSAN.)
|
|
|
|
|
|
|
|
template <size_t inlineSize>
|
2022-05-05 15:08:03 +00:00
|
|
|
class GrSTSubRunAllocator : private BagOfBytes::Storage<inlineSize>, public SubRunAllocator {
|
2020-12-16 15:45:50 +00:00
|
|
|
public:
|
|
|
|
explicit GrSTSubRunAllocator(int firstHeapAllocation =
|
2022-05-05 15:08:03 +00:00
|
|
|
BagOfBytes::PlatformMinimumSizeWithOverhead(inlineSize, 1))
|
|
|
|
: SubRunAllocator{this->data(), SkTo<int>(this->size()), firstHeapAllocation} {}
|
2020-12-16 15:45:50 +00:00
|
|
|
};
|
|
|
|
|
2022-05-05 15:08:03 +00:00
|
|
|
DEF_TEST(SubRunAllocator, r) {
|
2020-12-16 15:45:50 +00:00
|
|
|
static int created = 0;
|
|
|
|
static int destroyed = 0;
|
|
|
|
struct Foo {
|
|
|
|
Foo() : fI{-2}, fX{-3} { created++; }
|
|
|
|
Foo(int i, float x) : fI{i}, fX{x} { created++; }
|
|
|
|
~Foo() { destroyed++; }
|
|
|
|
int fI;
|
|
|
|
float fX;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct alignas(8) OddAlignment {
|
|
|
|
char buf[10];
|
|
|
|
};
|
|
|
|
|
2022-05-05 15:08:03 +00:00
|
|
|
auto exercise = [&](SubRunAllocator* alloc) {
|
2020-12-16 15:45:50 +00:00
|
|
|
created = 0;
|
|
|
|
destroyed = 0;
|
|
|
|
{
|
|
|
|
int* p = alloc->makePOD<int>(3);
|
|
|
|
REPORTER_ASSERT(r, *p == 3);
|
|
|
|
int* q = alloc->makePOD<int>(7);
|
|
|
|
REPORTER_ASSERT(r, *q == 7);
|
|
|
|
|
|
|
|
REPORTER_ASSERT(r, *alloc->makePOD<int>(3) == 3);
|
|
|
|
auto foo = alloc->makeUnique<Foo>(3, 4.0f);
|
|
|
|
REPORTER_ASSERT(r, foo->fI == 3);
|
|
|
|
REPORTER_ASSERT(r, foo->fX == 4.0f);
|
|
|
|
REPORTER_ASSERT(r, created == 1);
|
|
|
|
REPORTER_ASSERT(r, destroyed == 0);
|
|
|
|
|
|
|
|
alloc->makePODArray<int>(10);
|
|
|
|
|
|
|
|
auto fooArray = alloc->makeUniqueArray<Foo>(10);
|
|
|
|
REPORTER_ASSERT(r, fooArray[3].fI == -2);
|
|
|
|
REPORTER_ASSERT(r, fooArray[4].fX == -3.0f);
|
|
|
|
REPORTER_ASSERT(r, created == 11);
|
|
|
|
REPORTER_ASSERT(r, destroyed == 0);
|
|
|
|
alloc->makePOD<OddAlignment>();
|
|
|
|
}
|
|
|
|
|
|
|
|
REPORTER_ASSERT(r, created == 11);
|
|
|
|
REPORTER_ASSERT(r, destroyed == 11);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Exercise default arena
|
|
|
|
{
|
2022-05-05 15:08:03 +00:00
|
|
|
SubRunAllocator arena{0};
|
2020-12-16 15:45:50 +00:00
|
|
|
exercise(&arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exercise on stack arena
|
|
|
|
{
|
|
|
|
GrSTSubRunAllocator<64> arena;
|
|
|
|
exercise(&arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exercise arena with a heap allocated starting block
|
|
|
|
{
|
|
|
|
std::unique_ptr<char[]> block{new char[1024]};
|
2022-05-05 15:08:03 +00:00
|
|
|
SubRunAllocator arena{block.get(), 1024, 0};
|
2020-12-16 15:45:50 +00:00
|
|
|
exercise(&arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exercise the singly-link list of unique_ptrs use case
|
|
|
|
{
|
|
|
|
created = 0;
|
|
|
|
destroyed = 0;
|
2022-05-05 15:08:03 +00:00
|
|
|
SubRunAllocator arena;
|
2020-12-16 15:45:50 +00:00
|
|
|
|
|
|
|
struct Node {
|
2022-05-05 15:08:03 +00:00
|
|
|
Node(std::unique_ptr<Node, SubRunAllocator::Destroyer> next)
|
2020-12-16 15:45:50 +00:00
|
|
|
: fNext{std::move(next)} { created++; }
|
|
|
|
~Node() { destroyed++; }
|
2022-05-05 15:08:03 +00:00
|
|
|
std::unique_ptr<Node, SubRunAllocator::Destroyer> fNext;
|
2020-12-16 15:45:50 +00:00
|
|
|
};
|
|
|
|
|
2022-05-05 15:08:03 +00:00
|
|
|
std::unique_ptr<Node, SubRunAllocator::Destroyer> current = nullptr;
|
2020-12-16 15:45:50 +00:00
|
|
|
for (int i = 0; i < 128; i++) {
|
|
|
|
current = arena.makeUnique<Node>(std::move(current));
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(r, created == 128);
|
|
|
|
REPORTER_ASSERT(r, destroyed == 0);
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(r, created == 128);
|
|
|
|
REPORTER_ASSERT(r, destroyed == 128);
|
|
|
|
|
|
|
|
// Exercise the array ctor w/ a mapping function
|
|
|
|
{
|
|
|
|
struct I {
|
|
|
|
I(int v) : i{v} {}
|
|
|
|
~I() {}
|
|
|
|
int i;
|
|
|
|
};
|
|
|
|
GrSTSubRunAllocator<64> arena;
|
|
|
|
auto a = arena.makeUniqueArray<I>(8, [](size_t i) { return i; });
|
|
|
|
for (size_t i = 0; i < 8; i++) {
|
|
|
|
REPORTER_ASSERT(r, a[i].i == (int)i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2022-05-05 15:08:03 +00:00
|
|
|
SubRunAllocator arena(4096);
|
2021-02-24 16:20:40 +00:00
|
|
|
void* ptr = arena.alignedBytes(4081, 8);
|
2020-12-16 15:45:50 +00:00
|
|
|
REPORTER_ASSERT(r, ((intptr_t)ptr & 7) == 0);
|
|
|
|
}
|
|
|
|
}
|
2022-05-16 15:59:11 +00:00
|
|
|
|
2022-05-23 15:34:17 +00:00
|
|
|
using TextBlob = sktext::gpu::TextBlob;
|
|
|
|
|
2022-05-16 15:59:11 +00:00
|
|
|
DEF_TEST(KeyEqualityOnPerspective, r) {
|
|
|
|
SkTextBlobBuilder builder;
|
|
|
|
SkFont font(SkTypeface::MakeDefault(), 16);
|
2022-05-16 18:09:00 +00:00
|
|
|
auto runBuffer = builder.allocRun(font, 1, 0.0f, 0.0f);
|
|
|
|
runBuffer.glyphs[0] = 3;
|
2022-05-16 15:59:11 +00:00
|
|
|
auto blob = builder.make();
|
2022-06-15 21:58:25 +00:00
|
|
|
SkGlyphRunBuilder grBuilder;
|
2022-05-16 15:59:11 +00:00
|
|
|
auto glyphRunList = grBuilder.blobToGlyphRunList(*blob, {100, 100});
|
|
|
|
SkPaint paint;
|
2022-05-17 15:14:00 +00:00
|
|
|
|
|
|
|
// Build the strike device.
|
2022-05-16 15:59:11 +00:00
|
|
|
SkSurfaceProps props;
|
2022-05-19 16:41:15 +00:00
|
|
|
sktext::gpu::SDFTControl control(false, false, 1, 100);
|
2022-05-17 15:14:00 +00:00
|
|
|
SkStrikeDeviceInfo strikeDevice{props, SkScalerContextFlags::kBoostContrast, &control};
|
2022-05-16 15:59:11 +00:00
|
|
|
SkMatrix matrix1;
|
|
|
|
matrix1.setAll(1, 0, 0, 0, 1, 0, 1, 1, 1);
|
|
|
|
SkMatrix matrix2;
|
|
|
|
matrix2.setAll(1, 0, 0, 0, 1, 0, 2, 2, 1);
|
|
|
|
auto key1 = std::get<1>(
|
2022-05-23 15:34:17 +00:00
|
|
|
TextBlob::Key::Make(glyphRunList, paint, matrix1, strikeDevice));
|
2022-05-16 15:59:11 +00:00
|
|
|
auto key2 = std::get<1>(
|
2022-05-23 15:34:17 +00:00
|
|
|
TextBlob::Key::Make(glyphRunList, paint, matrix1, strikeDevice));
|
2022-05-16 15:59:11 +00:00
|
|
|
auto key3 = std::get<1>(
|
2022-05-23 15:34:17 +00:00
|
|
|
TextBlob::Key::Make(glyphRunList, paint, matrix2, strikeDevice));
|
2022-05-16 15:59:11 +00:00
|
|
|
REPORTER_ASSERT(r, key1 == key2);
|
|
|
|
REPORTER_ASSERT(r, !(key1 == key3));
|
|
|
|
}
|