robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2012 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
| 8 | #include "gm.h" |
Mike Klein | 33d2055 | 2017-03-22 13:47:51 -0400 | [diff] [blame] | 9 | #include "sk_tool_utils.h" |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 10 | #if SK_SUPPORT_GPU |
joshualitt | 04194f3 | 2016-01-13 10:08:27 -0800 | [diff] [blame] | 11 | #include "GrContext.h" |
Brian Osman | 1105224 | 2016-10-27 14:47:55 -0400 | [diff] [blame] | 12 | #include "GrRenderTargetContextPriv.h" |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 13 | #include "effects/GrRRectEffect.h" |
Brian Salomon | 8952743 | 2016-12-16 09:52:16 -0500 | [diff] [blame] | 14 | #include "ops/GrDrawOp.h" |
Brian Salomon | baaf439 | 2017-06-15 09:59:23 -0400 | [diff] [blame] | 15 | #include "ops/GrRectOpFactory.h" |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 16 | #endif |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 17 | #include "SkRRect.h" |
| 18 | |
| 19 | namespace skiagm { |
| 20 | |
| 21 | /////////////////////////////////////////////////////////////////////////////// |
| 22 | |
| 23 | class RRectGM : public GM { |
| 24 | public: |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 25 | enum Type { |
| 26 | kBW_Draw_Type, |
| 27 | kAA_Draw_Type, |
| 28 | kBW_Clip_Type, |
| 29 | kAA_Clip_Type, |
| 30 | kEffect_Type, |
| 31 | }; |
herb | b10fe49 | 2016-01-08 13:48:43 -0800 | [diff] [blame] | 32 | RRectGM(Type type) : fType(type) { } |
| 33 | |
| 34 | protected: |
| 35 | |
| 36 | void onOnceBeforeDraw() override { |
caryclark | 65cdba6 | 2015-06-15 06:51:08 -0700 | [diff] [blame] | 37 | this->setBGColor(sk_tool_utils::color_to_565(0xFFDDDDDD)); |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 38 | this->setUpRRects(); |
| 39 | } |
| 40 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 41 | SkString onShortName() override { |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 42 | SkString name("rrect"); |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 43 | switch (fType) { |
| 44 | case kBW_Draw_Type: |
| 45 | name.append("_draw_bw"); |
| 46 | break; |
| 47 | case kAA_Draw_Type: |
| 48 | name.append("_draw_aa"); |
| 49 | break; |
| 50 | case kBW_Clip_Type: |
| 51 | name.append("_clip_bw"); |
| 52 | break; |
| 53 | case kAA_Clip_Type: |
| 54 | name.append("_clip_aa"); |
| 55 | break; |
| 56 | case kEffect_Type: |
| 57 | name.append("_effect"); |
| 58 | break; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 59 | } |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 60 | return name; |
| 61 | } |
| 62 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 63 | SkISize onISize() override { return SkISize::Make(kImageWidth, kImageHeight); } |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 64 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 65 | void onDraw(SkCanvas* canvas) override { |
Brian Osman | 1105224 | 2016-10-27 14:47:55 -0400 | [diff] [blame] | 66 | GrRenderTargetContext* renderTargetContext = |
| 67 | canvas->internal_private_accessTopLayerRenderTargetContext(); |
| 68 | if (kEffect_Type == fType && !renderTargetContext) { |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 69 | skiagm::GM::DrawGpuOnlyMessage(canvas); |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 70 | return; |
| 71 | } |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 72 | |
| 73 | SkPaint paint; |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 74 | if (kAA_Draw_Type == fType) { |
| 75 | paint.setAntiAlias(true); |
| 76 | } |
skia.committer@gmail.com | f1f66c0 | 2014-03-05 03:02:06 +0000 | [diff] [blame] | 77 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 78 | const SkRect kMaxTileBound = SkRect::MakeWH(SkIntToScalar(kTileX), |
| 79 | SkIntToScalar(kTileY)); |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 80 | #ifdef SK_DEBUG |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 81 | const SkRect kMaxImageBound = SkRect::MakeWH(SkIntToScalar(kImageWidth), |
| 82 | SkIntToScalar(kImageHeight)); |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 83 | #endif |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 84 | |
bsalomon@google.com | b0ba39d | 2014-03-10 19:51:46 +0000 | [diff] [blame] | 85 | #if SK_SUPPORT_GPU |
Ethan Nicholas | 1706f84 | 2017-11-10 11:58:19 -0500 | [diff] [blame] | 86 | int lastEdgeType = (kEffect_Type == fType) ? (int) GrClipEdgeType::kLast: 0; |
bsalomon@google.com | b0ba39d | 2014-03-10 19:51:46 +0000 | [diff] [blame] | 87 | #else |
| 88 | int lastEdgeType = 0; |
| 89 | #endif |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 90 | |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 91 | int y = 1; |
bsalomon@google.com | b0ba39d | 2014-03-10 19:51:46 +0000 | [diff] [blame] | 92 | for (int et = 0; et <= lastEdgeType; ++et) { |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 93 | int x = 1; |
| 94 | for (int curRRect = 0; curRRect < kNumRRects; ++curRRect) { |
| 95 | bool drew = true; |
| 96 | #ifdef SK_DEBUG |
| 97 | SkASSERT(kMaxTileBound.contains(fRRects[curRRect].getBounds())); |
| 98 | SkRect imageSpaceBounds = fRRects[curRRect].getBounds(); |
| 99 | imageSpaceBounds.offset(SkIntToScalar(x), SkIntToScalar(y)); |
| 100 | SkASSERT(kMaxImageBound.contains(imageSpaceBounds)); |
| 101 | #endif |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 102 | canvas->save(); |
| 103 | canvas->translate(SkIntToScalar(x), SkIntToScalar(y)); |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 104 | if (kEffect_Type == fType) { |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 105 | #if SK_SUPPORT_GPU |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 106 | SkRRect rrect = fRRects[curRRect]; |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 107 | rrect.offset(SkIntToScalar(x), SkIntToScalar(y)); |
Ethan Nicholas | 0f3c732 | 2017-11-09 14:51:17 -0500 | [diff] [blame] | 108 | GrClipEdgeType edgeType = (GrClipEdgeType) et; |
Brian Salomon | 1447177 | 2017-12-05 10:35:15 -0500 | [diff] [blame] | 109 | const auto& caps = *renderTargetContext->caps()->shaderCaps(); |
| 110 | auto fp = GrRRectEffect::Make(edgeType, rrect, caps); |
joshualitt | b0a8a37 | 2014-09-23 09:50:21 -0700 | [diff] [blame] | 111 | if (fp) { |
Brian Salomon | 82f4431 | 2017-01-11 13:42:54 -0500 | [diff] [blame] | 112 | GrPaint grPaint; |
| 113 | grPaint.setXPFactory(GrPorterDuffXPFactory::Get(SkBlendMode::kSrc)); |
robertphillips | 28a838e | 2016-06-23 14:07:00 -0700 | [diff] [blame] | 114 | grPaint.addCoverageFragmentProcessor(std::move(fp)); |
Brian Salomon | ac70f84 | 2017-05-08 10:43:33 -0400 | [diff] [blame] | 115 | grPaint.setColor4f(GrColor4f(0, 0, 0, 1.f)); |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 116 | |
| 117 | SkRect bounds = rrect.getBounds(); |
| 118 | bounds.outset(2.f, 2.f); |
skia.committer@gmail.com | f1f66c0 | 2014-03-05 03:02:06 +0000 | [diff] [blame] | 119 | |
Brian Salomon | ac70f84 | 2017-05-08 10:43:33 -0400 | [diff] [blame] | 120 | renderTargetContext->priv().testingOnly_addDrawOp( |
Brian Salomon | baaf439 | 2017-06-15 09:59:23 -0400 | [diff] [blame] | 121 | GrRectOpFactory::MakeNonAAFill(std::move(grPaint), |
| 122 | SkMatrix::I(), bounds, |
| 123 | GrAAType::kNone)); |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 124 | } else { |
| 125 | drew = false; |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 126 | } |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 127 | #endif |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 128 | } else if (kBW_Clip_Type == fType || kAA_Clip_Type == fType) { |
| 129 | bool aaClip = (kAA_Clip_Type == fType); |
reed | 6699838 | 2016-09-21 11:15:07 -0700 | [diff] [blame] | 130 | canvas->clipRRect(fRRects[curRRect], aaClip); |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 131 | canvas->drawRect(kMaxTileBound, paint); |
| 132 | } else { |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 133 | canvas->drawRRect(fRRects[curRRect], paint); |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 134 | } |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 135 | canvas->restore(); |
commit-bot@chromium.org | bfce48e | 2014-03-10 19:33:16 +0000 | [diff] [blame] | 136 | if (drew) { |
| 137 | x = x + kTileX; |
| 138 | if (x > kImageWidth) { |
| 139 | x = 1; |
| 140 | y += kTileY; |
| 141 | } |
| 142 | } |
| 143 | } |
| 144 | if (x != 1) { |
| 145 | y += kTileY; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 146 | } |
| 147 | } |
| 148 | } |
| 149 | |
| 150 | void setUpRRects() { |
skia.committer@gmail.com | 7a03d86 | 2012-12-18 02:03:03 +0000 | [diff] [blame] | 151 | // each RRect must fit in a 0x0 -> (kTileX-2)x(kTileY-2) block. These will be tiled across |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 152 | // the screen in kTileX x kTileY tiles. The extra empty pixels on each side are for AA. |
| 153 | |
| 154 | // simple cases |
| 155 | fRRects[0].setRect(SkRect::MakeWH(kTileX-2, kTileY-2)); |
| 156 | fRRects[1].setOval(SkRect::MakeWH(kTileX-2, kTileY-2)); |
| 157 | fRRects[2].setRectXY(SkRect::MakeWH(kTileX-2, kTileY-2), 10, 10); |
commit-bot@chromium.org | c2f7824 | 2014-02-19 15:18:05 +0000 | [diff] [blame] | 158 | fRRects[3].setRectXY(SkRect::MakeWH(kTileX-2, kTileY-2), 10, 5); |
| 159 | // small circular corners are an interesting test case for gpu clipping |
| 160 | fRRects[4].setRectXY(SkRect::MakeWH(kTileX-2, kTileY-2), 1, 1); |
| 161 | fRRects[5].setRectXY(SkRect::MakeWH(kTileX-2, kTileY-2), 0.5f, 0.5f); |
| 162 | fRRects[6].setRectXY(SkRect::MakeWH(kTileX-2, kTileY-2), 0.2f, 0.2f); |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 163 | |
| 164 | // The first complex case needs special handling since it is a square |
| 165 | fRRects[kNumSimpleCases].setRectRadii(SkRect::MakeWH(kTileY-2, kTileY-2), gRadii[0]); |
mike@reedtribe.org | f6100c8 | 2012-12-24 13:56:17 +0000 | [diff] [blame] | 166 | for (size_t i = 1; i < SK_ARRAY_COUNT(gRadii); ++i) { |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 167 | fRRects[kNumSimpleCases+i].setRectRadii(SkRect::MakeWH(kTileX-2, kTileY-2), gRadii[i]); |
| 168 | } |
| 169 | } |
| 170 | |
| 171 | private: |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 172 | Type fType; |
skia.committer@gmail.com | f1f66c0 | 2014-03-05 03:02:06 +0000 | [diff] [blame] | 173 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 174 | static constexpr int kImageWidth = 640; |
| 175 | static constexpr int kImageHeight = 480; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 176 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 177 | static constexpr int kTileX = 80; |
| 178 | static constexpr int kTileY = 40; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 179 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 180 | static constexpr int kNumSimpleCases = 7; |
| 181 | static constexpr int kNumComplexCases = 35; |
robertphillips@google.com | 5683d42 | 2012-12-17 21:58:02 +0000 | [diff] [blame] | 182 | static const SkVector gRadii[kNumComplexCases][4]; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 183 | |
mtklein | dbfd7ab | 2016-09-01 11:24:54 -0700 | [diff] [blame] | 184 | static constexpr int kNumRRects = kNumSimpleCases + kNumComplexCases; |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 185 | SkRRect fRRects[kNumRRects]; |
| 186 | |
| 187 | typedef GM INHERITED; |
| 188 | }; |
| 189 | |
| 190 | // Radii for the various test cases. Order is UL, UR, LR, LL |
| 191 | const SkVector RRectGM::gRadii[kNumComplexCases][4] = { |
| 192 | // a circle |
| 193 | { { kTileY, kTileY }, { kTileY, kTileY }, { kTileY, kTileY }, { kTileY, kTileY } }, |
| 194 | |
| 195 | // odd ball cases |
| 196 | { { 8, 8 }, { 32, 32 }, { 8, 8 }, { 32, 32 } }, |
| 197 | { { 16, 8 }, { 8, 16 }, { 16, 8 }, { 8, 16 } }, |
| 198 | { { 0, 0 }, { 16, 16 }, { 8, 8 }, { 32, 32 } }, |
| 199 | |
| 200 | // UL |
| 201 | { { 30, 30 }, { 0, 0 }, { 0, 0 }, { 0, 0 } }, |
| 202 | { { 30, 15 }, { 0, 0 }, { 0, 0 }, { 0, 0 } }, |
| 203 | { { 15, 30 }, { 0, 0 }, { 0, 0 }, { 0, 0 } }, |
| 204 | |
| 205 | // UR |
| 206 | { { 0, 0 }, { 30, 30 }, { 0, 0 }, { 0, 0 } }, |
| 207 | { { 0, 0 }, { 30, 15 }, { 0, 0 }, { 0, 0 } }, |
| 208 | { { 0, 0 }, { 15, 30 }, { 0, 0 }, { 0, 0 } }, |
| 209 | |
| 210 | // LR |
| 211 | { { 0, 0 }, { 0, 0 }, { 30, 30 }, { 0, 0 } }, |
| 212 | { { 0, 0 }, { 0, 0 }, { 30, 15 }, { 0, 0 } }, |
| 213 | { { 0, 0 }, { 0, 0 }, { 15, 30 }, { 0, 0 } }, |
| 214 | |
| 215 | // LL |
| 216 | { { 0, 0 }, { 0, 0 }, { 0, 0 }, { 30, 30 } }, |
| 217 | { { 0, 0 }, { 0, 0 }, { 0, 0 }, { 30, 15 } }, |
| 218 | { { 0, 0 }, { 0, 0 }, { 0, 0 }, { 15, 30 } }, |
| 219 | |
| 220 | // over-sized radii |
| 221 | { { 0, 0 }, { 100, 400 }, { 0, 0 }, { 0, 0 } }, |
| 222 | { { 0, 0 }, { 400, 400 }, { 0, 0 }, { 0, 0 } }, |
| 223 | { { 400, 400 }, { 400, 400 }, { 400, 400 }, { 400, 400 } }, |
commit-bot@chromium.org | cb3672e | 2014-02-21 22:41:56 +0000 | [diff] [blame] | 224 | |
| 225 | // circular corner tabs |
| 226 | { { 0, 0 }, { 20, 20 }, { 20, 20 }, { 0, 0 } }, |
| 227 | { { 20, 20 }, { 20, 20 }, { 0, 0 }, { 0, 0 } }, |
| 228 | { { 0, 0 }, { 0, 0 }, { 20, 20 }, { 20, 20 } }, |
| 229 | { { 20, 20 }, { 0, 0 }, { 0, 0 }, { 20, 20 } }, |
commit-bot@chromium.org | c5c748c | 2014-03-11 15:54:51 +0000 | [diff] [blame] | 230 | |
commit-bot@chromium.org | 04eff72 | 2014-03-24 14:53:09 +0000 | [diff] [blame] | 231 | // small radius circular corner tabs |
| 232 | { { 0, 0 }, { 0.2f, 0.2f }, { 0.2f, 0.2f }, { 0, 0 } }, |
| 233 | { { 0.3f, 0.3f }, { 0.3f, .3f }, { 0, 0 }, { 0, 0 } }, |
| 234 | |
commit-bot@chromium.org | c5c748c | 2014-03-11 15:54:51 +0000 | [diff] [blame] | 235 | // single circular corner cases |
| 236 | { { 0, 0 }, { 0, 0 }, { 0, 0 }, { 15, 15 } }, |
| 237 | { { 0, 0 }, { 0, 0 }, { 15, 15 }, { 0, 0 } }, |
| 238 | { { 0, 0 }, { 15, 15 }, { 0, 0 }, { 0, 0 } }, |
| 239 | { { 15, 15 }, { 0, 0 }, { 0, 0 }, { 0, 0 } }, |
commit-bot@chromium.org | fa5edbe | 2014-03-13 18:01:05 +0000 | [diff] [blame] | 240 | |
| 241 | // nine patch elliptical |
| 242 | { { 5, 7 }, { 8, 7 }, { 8, 12 }, { 5, 12 } }, |
| 243 | { { 0, 7 }, { 8, 7 }, { 8, 12 }, { 0, 12 } }, |
commit-bot@chromium.org | 04eff72 | 2014-03-24 14:53:09 +0000 | [diff] [blame] | 244 | |
| 245 | // nine patch elliptical, small radii |
| 246 | { { 0.4f, 7 }, { 8, 7 }, { 8, 12 }, { 0.4f, 12 } }, |
| 247 | { { 0.4f, 0.4f }, { 8, 0.4f }, { 8, 12 }, { 0.4f, 12 } }, |
| 248 | { { 20, 0.4f }, { 18, 0.4f }, { 18, 0.4f }, { 20, 0.4f } }, |
| 249 | { { 0.3f, 0.4f }, { 0.3f, 0.4f }, { 0.3f, 0.4f }, { 0.3f, 0.4f } }, |
| 250 | |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 251 | }; |
| 252 | |
| 253 | /////////////////////////////////////////////////////////////////////////////// |
| 254 | |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 255 | DEF_GM( return new RRectGM(RRectGM::kAA_Draw_Type); ) |
| 256 | DEF_GM( return new RRectGM(RRectGM::kBW_Draw_Type); ) |
| 257 | DEF_GM( return new RRectGM(RRectGM::kAA_Clip_Type); ) |
| 258 | DEF_GM( return new RRectGM(RRectGM::kBW_Clip_Type); ) |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 259 | #if SK_SUPPORT_GPU |
commit-bot@chromium.org | fbde87f | 2014-03-04 16:25:34 +0000 | [diff] [blame] | 260 | DEF_GM( return new RRectGM(RRectGM::kEffect_Type); ) |
bsalomon@google.com | 707bd60 | 2014-03-04 16:52:20 +0000 | [diff] [blame] | 261 | #endif |
robertphillips@google.com | 4e18c7a | 2012-12-17 21:48:19 +0000 | [diff] [blame] | 262 | |
| 263 | } |