epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 1 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 2 | /* |
epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 3 | * Copyright 2010 Google Inc. |
| 4 | * |
| 5 | * Use of this source code is governed by a BSD-style license that can be |
| 6 | * found in the LICENSE file. |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 7 | */ |
| 8 | |
| 9 | |
epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 10 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 11 | #ifndef SkGpuDevice_DEFINED |
| 12 | #define SkGpuDevice_DEFINED |
| 13 | |
| 14 | #include "SkGr.h" |
reed@google.com | af951c9 | 2011-06-16 19:10:39 +0000 | [diff] [blame] | 15 | #include "SkBitmap.h" |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 16 | #include "SkBitmapDevice.h" |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 17 | #include "SkRegion.h" |
bsalomon@google.com | 50398bf | 2011-07-26 20:45:30 +0000 | [diff] [blame] | 18 | #include "GrContext.h" |
commit-bot@chromium.org | cc40f06 | 2014-01-24 14:38:27 +0000 | [diff] [blame^] | 19 | #include "GrTextContext.h" |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 20 | |
| 21 | struct SkDrawProcs; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 22 | struct GrSkDrawProcs; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 23 | |
| 24 | /** |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 25 | * Subclass of SkBitmapDevice, which directs all drawing to the GrGpu owned by the |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 26 | * canvas. |
| 27 | */ |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 28 | class SK_API SkGpuDevice : public SkBitmapDevice { |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 29 | public: |
bsalomon@google.com | 123ac1d | 2013-03-28 19:18:12 +0000 | [diff] [blame] | 30 | |
| 31 | /** |
| 32 | * Creates an SkGpuDevice from a GrSurface. This will fail if the surface is not a render |
| 33 | * target. The caller owns a ref on the returned device. |
| 34 | */ |
| 35 | static SkGpuDevice* Create(GrSurface* surface); |
| 36 | |
bsalomon@google.com | 2e7b43d | 2011-01-18 20:57:22 +0000 | [diff] [blame] | 37 | /** |
reed@google.com | af951c9 | 2011-06-16 19:10:39 +0000 | [diff] [blame] | 38 | * New device that will create an offscreen renderTarget based on the |
skia.committer@gmail.com | f3dc199 | 2012-11-01 02:01:27 +0000 | [diff] [blame] | 39 | * config, width, height, and sampleCount. The device's storage will not |
| 40 | * count against the GrContext's texture cache budget. The device's pixels |
bsalomon@google.com | 123ac1d | 2013-03-28 19:18:12 +0000 | [diff] [blame] | 41 | * will be uninitialized. TODO: This can fail, replace with a factory function. |
bsalomon@google.com | 2e7b43d | 2011-01-18 20:57:22 +0000 | [diff] [blame] | 42 | */ |
robertphillips@google.com | d3eb336 | 2012-10-31 13:56:35 +0000 | [diff] [blame] | 43 | SkGpuDevice(GrContext*, SkBitmap::Config, int width, int height, int sampleCount = 0); |
reed@google.com | af951c9 | 2011-06-16 19:10:39 +0000 | [diff] [blame] | 44 | |
| 45 | /** |
reed@google.com | 4469938 | 2013-10-31 17:28:30 +0000 | [diff] [blame] | 46 | * DEPRECATED -- need to make this private, call Create(surface) |
reed@google.com | af951c9 | 2011-06-16 19:10:39 +0000 | [diff] [blame] | 47 | * New device that will render to the specified renderTarget. |
| 48 | */ |
| 49 | SkGpuDevice(GrContext*, GrRenderTarget*); |
bsalomon@google.com | 2e7b43d | 2011-01-18 20:57:22 +0000 | [diff] [blame] | 50 | |
| 51 | /** |
reed@google.com | 4469938 | 2013-10-31 17:28:30 +0000 | [diff] [blame] | 52 | * DEPRECATED -- need to make this private, call Create(surface) |
bsalomon@google.com | f9046fe | 2011-06-17 15:10:21 +0000 | [diff] [blame] | 53 | * New device that will render to the texture (as a rendertarget). |
| 54 | * The GrTexture's asRenderTarget() must be non-NULL or device will not |
| 55 | * function. |
| 56 | */ |
| 57 | SkGpuDevice(GrContext*, GrTexture*); |
| 58 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 59 | virtual ~SkGpuDevice(); |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 60 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 61 | GrContext* context() const { return fContext; } |
| 62 | |
commit-bot@chromium.org | b8d00db | 2013-06-26 19:18:23 +0000 | [diff] [blame] | 63 | virtual GrRenderTarget* accessRenderTarget() SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 64 | |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 65 | // overrides from SkBaseDevice |
robertphillips@google.com | 07f81a5 | 2013-09-17 12:26:23 +0000 | [diff] [blame] | 66 | virtual uint32_t getDeviceCapabilities() SK_OVERRIDE { |
| 67 | return 0; |
| 68 | } |
| 69 | virtual int width() const SK_OVERRIDE { |
| 70 | return NULL == fRenderTarget ? 0 : fRenderTarget->width(); |
| 71 | } |
| 72 | virtual int height() const SK_OVERRIDE { |
| 73 | return NULL == fRenderTarget ? 0 : fRenderTarget->height(); |
| 74 | } |
skia.committer@gmail.com | 2291e72 | 2013-09-18 07:01:33 +0000 | [diff] [blame] | 75 | virtual bool isOpaque() const SK_OVERRIDE { |
| 76 | return NULL == fRenderTarget ? false |
| 77 | : kRGB_565_GrPixelConfig == fRenderTarget->config(); |
robertphillips@google.com | 07f81a5 | 2013-09-17 12:26:23 +0000 | [diff] [blame] | 78 | } |
| 79 | virtual SkBitmap::Config config() const SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 80 | |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 81 | virtual void clear(SkColor color) SK_OVERRIDE; |
bsalomon@google.com | d58a1cd | 2011-11-10 20:57:43 +0000 | [diff] [blame] | 82 | virtual void writePixels(const SkBitmap& bitmap, int x, int y, |
| 83 | SkCanvas::Config8888 config8888) SK_OVERRIDE; |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 84 | |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 85 | virtual void drawPaint(const SkDraw&, const SkPaint& paint) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 86 | virtual void drawPoints(const SkDraw&, SkCanvas::PointMode mode, size_t count, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 87 | const SkPoint[], const SkPaint& paint) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 88 | virtual void drawRect(const SkDraw&, const SkRect& r, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 89 | const SkPaint& paint) SK_OVERRIDE; |
commit-bot@chromium.org | f2bfd54 | 2013-04-25 15:27:00 +0000 | [diff] [blame] | 90 | virtual void drawRRect(const SkDraw&, const SkRRect& r, |
| 91 | const SkPaint& paint) SK_OVERRIDE; |
jvanverth@google.com | 46d3d39 | 2013-01-22 13:34:01 +0000 | [diff] [blame] | 92 | virtual void drawOval(const SkDraw&, const SkRect& oval, |
| 93 | const SkPaint& paint) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 94 | virtual void drawPath(const SkDraw&, const SkPath& path, |
| 95 | const SkPaint& paint, const SkMatrix* prePathMatrix, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 96 | bool pathIsMutable) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 97 | virtual void drawBitmap(const SkDraw&, const SkBitmap& bitmap, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 98 | const SkMatrix&, const SkPaint&) SK_OVERRIDE; |
reed@google.com | 33535f3 | 2012-09-25 15:37:50 +0000 | [diff] [blame] | 99 | virtual void drawBitmapRect(const SkDraw&, const SkBitmap&, |
| 100 | const SkRect* srcOrNull, const SkRect& dst, |
commit-bot@chromium.org | eed779d | 2013-08-16 10:24:37 +0000 | [diff] [blame] | 101 | const SkPaint& paint, |
| 102 | SkCanvas::DrawBitmapRectFlags flags) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 103 | virtual void drawSprite(const SkDraw&, const SkBitmap& bitmap, |
| 104 | int x, int y, const SkPaint& paint); |
| 105 | virtual void drawText(const SkDraw&, const void* text, size_t len, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 106 | SkScalar x, SkScalar y, const SkPaint&) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 107 | virtual void drawPosText(const SkDraw&, const void* text, size_t len, |
| 108 | const SkScalar pos[], SkScalar constY, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 109 | int scalarsPerPos, const SkPaint&) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 110 | virtual void drawTextOnPath(const SkDraw&, const void* text, size_t len, |
| 111 | const SkPath& path, const SkMatrix* matrix, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 112 | const SkPaint&) SK_OVERRIDE; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 113 | virtual void drawVertices(const SkDraw&, SkCanvas::VertexMode, int vertexCount, |
| 114 | const SkPoint verts[], const SkPoint texs[], |
| 115 | const SkColor colors[], SkXfermode* xmode, |
| 116 | const uint16_t indices[], int indexCount, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 117 | const SkPaint&) SK_OVERRIDE; |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 118 | virtual void drawDevice(const SkDraw&, SkBaseDevice*, int x, int y, |
reed@google.com | 75d939b | 2011-12-07 15:07:23 +0000 | [diff] [blame] | 119 | const SkPaint&) SK_OVERRIDE; |
| 120 | virtual bool filterTextFlags(const SkPaint&, TextFlags*) SK_OVERRIDE; |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 121 | |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 122 | virtual void flush() SK_OVERRIDE; |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 123 | |
robertphillips@google.com | 40a1ae4 | 2012-07-13 15:36:15 +0000 | [diff] [blame] | 124 | virtual void onAttachToCanvas(SkCanvas* canvas) SK_OVERRIDE; |
| 125 | virtual void onDetachFromCanvas() SK_OVERRIDE; |
| 126 | |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 127 | /** |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 128 | * Make's this device's rendertarget current in the underlying 3D API. |
| 129 | * Also implicitly flushes. |
| 130 | */ |
| 131 | virtual void makeRenderTargetCurrent(); |
| 132 | |
reed@google.com | 8926b16 | 2012-03-23 15:36:36 +0000 | [diff] [blame] | 133 | virtual bool canHandleImageFilter(SkImageFilter*) SK_OVERRIDE; |
| 134 | virtual bool filterImage(SkImageFilter*, const SkBitmap&, const SkMatrix&, |
| 135 | SkBitmap*, SkIPoint*) SK_OVERRIDE; |
vandebo@chromium.org | 74b4619 | 2012-01-28 01:45:11 +0000 | [diff] [blame] | 136 | |
bsalomon@google.com | 84405e0 | 2012-03-05 19:57:21 +0000 | [diff] [blame] | 137 | class SkAutoCachedTexture; // used internally |
| 138 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 139 | protected: |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 140 | // overrides from SkBaseDevice |
bsalomon@google.com | 910267d | 2011-11-02 20:06:25 +0000 | [diff] [blame] | 141 | virtual bool onReadPixels(const SkBitmap& bitmap, |
bsalomon@google.com | 6850eab | 2011-11-03 20:29:47 +0000 | [diff] [blame] | 142 | int x, int y, |
| 143 | SkCanvas::Config8888 config8888) SK_OVERRIDE; |
bsalomon@google.com | c698097 | 2011-11-02 19:57:21 +0000 | [diff] [blame] | 144 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 145 | private: |
| 146 | GrContext* fContext; |
bsalomon@google.com | 5782d71 | 2011-01-21 21:03:59 +0000 | [diff] [blame] | 147 | |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 148 | GrSkDrawProcs* fDrawProcs; |
| 149 | |
robertphillips@google.com | 641f8b1 | 2012-07-31 19:15:58 +0000 | [diff] [blame] | 150 | GrClipData fClipData; |
robertphillips@google.com | 40a1ae4 | 2012-07-13 15:36:15 +0000 | [diff] [blame] | 151 | |
commit-bot@chromium.org | cc40f06 | 2014-01-24 14:38:27 +0000 | [diff] [blame^] | 152 | GrTextContextManager* fTextContextManager; |
| 153 | |
bsalomon@google.com | a6926b1 | 2012-10-10 15:25:50 +0000 | [diff] [blame] | 154 | // state for our render-target |
bsalomon@google.com | 50398bf | 2011-07-26 20:45:30 +0000 | [diff] [blame] | 155 | GrRenderTarget* fRenderTarget; |
| 156 | bool fNeedClear; |
reed@google.com | 7b201d2 | 2011-01-11 18:59:23 +0000 | [diff] [blame] | 157 | |
bsalomon@google.com | f9046fe | 2011-06-17 15:10:21 +0000 | [diff] [blame] | 158 | // called from rt and tex cons |
bsalomon@google.com | 8090e65 | 2012-08-28 15:07:11 +0000 | [diff] [blame] | 159 | void initFromRenderTarget(GrContext*, GrRenderTarget*, bool cached); |
bsalomon@google.com | f9046fe | 2011-06-17 15:10:21 +0000 | [diff] [blame] | 160 | |
bsalomon@google.com | 06cd732 | 2012-03-30 18:45:35 +0000 | [diff] [blame] | 161 | // used by createCompatibleDevice |
robertphillips@google.com | 1f47f4f | 2012-08-16 14:49:16 +0000 | [diff] [blame] | 162 | SkGpuDevice(GrContext*, GrTexture* texture, bool needClear); |
bsalomon@google.com | 06cd732 | 2012-03-30 18:45:35 +0000 | [diff] [blame] | 163 | |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 164 | // override from SkBaseDevice |
| 165 | virtual SkBaseDevice* onCreateCompatibleDevice(SkBitmap::Config config, |
| 166 | int width, int height, |
| 167 | bool isOpaque, |
| 168 | Usage usage) SK_OVERRIDE; |
bsalomon@google.com | e97f085 | 2011-06-17 13:10:25 +0000 | [diff] [blame] | 169 | |
bsalomon@google.com | 5782d71 | 2011-01-21 21:03:59 +0000 | [diff] [blame] | 170 | SkDrawProcs* initDrawForText(GrTextContext*); |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 171 | |
bsalomon@google.com | a99226e | 2012-10-12 15:01:38 +0000 | [diff] [blame] | 172 | // sets the render target, clip, and matrix on GrContext. Use forceIdenity to override |
| 173 | // SkDraw's matrix and draw in device coords. |
| 174 | void prepareDraw(const SkDraw&, bool forceIdentity); |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 175 | |
| 176 | /** |
| 177 | * Implementation for both drawBitmap and drawBitmapRect. |
| 178 | */ |
skia.committer@gmail.com | 22b460c | 2012-09-29 02:01:02 +0000 | [diff] [blame] | 179 | void drawBitmapCommon(const SkDraw&, |
robertphillips@google.com | bac6b05 | 2012-09-28 18:06:49 +0000 | [diff] [blame] | 180 | const SkBitmap& bitmap, |
| 181 | const SkRect* srcRectPtr, |
commit-bot@chromium.org | a7d89c8 | 2014-01-13 14:47:00 +0000 | [diff] [blame] | 182 | const SkSize* dstSizePtr, // ignored iff srcRectPtr == NULL |
commit-bot@chromium.org | eed779d | 2013-08-16 10:24:37 +0000 | [diff] [blame] | 183 | const SkPaint&, |
| 184 | SkCanvas::DrawBitmapRectFlags flags); |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 185 | |
bsalomon@google.com | f4a9c82 | 2012-03-16 14:02:46 +0000 | [diff] [blame] | 186 | /** |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 187 | * Helper functions called by drawBitmapCommon. By the time these are called the SkDraw's |
commit-bot@chromium.org | 7edad87 | 2013-10-25 14:58:12 +0000 | [diff] [blame] | 188 | * matrix, clip, and the device's render target has already been set on GrContext. |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 189 | */ |
commit-bot@chromium.org | 7edad87 | 2013-10-25 14:58:12 +0000 | [diff] [blame] | 190 | |
| 191 | // The tileSize and clippedSrcRect will be valid only if true is returned. |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 192 | bool shouldTileBitmap(const SkBitmap& bitmap, |
| 193 | const GrTextureParams& sampler, |
bsalomon@google.com | af562b4 | 2013-10-24 17:52:07 +0000 | [diff] [blame] | 194 | const SkRect* srcRectPtr, |
| 195 | int maxTileSize, |
commit-bot@chromium.org | 7edad87 | 2013-10-25 14:58:12 +0000 | [diff] [blame] | 196 | int* tileSize, |
| 197 | SkIRect* clippedSrcRect) const; |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 198 | void internalDrawBitmap(const SkBitmap&, |
| 199 | const SkRect&, |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 200 | const GrTextureParams& params, |
commit-bot@chromium.org | eed779d | 2013-08-16 10:24:37 +0000 | [diff] [blame] | 201 | const SkPaint& paint, |
commit-bot@chromium.org | dec6150 | 2013-12-02 22:22:35 +0000 | [diff] [blame] | 202 | SkCanvas::DrawBitmapRectFlags flags, |
| 203 | bool bicubic); |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 204 | void drawTiledBitmap(const SkBitmap& bitmap, |
| 205 | const SkRect& srcRect, |
commit-bot@chromium.org | 7edad87 | 2013-10-25 14:58:12 +0000 | [diff] [blame] | 206 | const SkIRect& clippedSrcRect, |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 207 | const GrTextureParams& params, |
commit-bot@chromium.org | eed779d | 2013-08-16 10:24:37 +0000 | [diff] [blame] | 208 | const SkPaint& paint, |
bsalomon@google.com | af562b4 | 2013-10-24 17:52:07 +0000 | [diff] [blame] | 209 | SkCanvas::DrawBitmapRectFlags flags, |
commit-bot@chromium.org | dec6150 | 2013-12-02 22:22:35 +0000 | [diff] [blame] | 210 | int tileSize, |
| 211 | bool bicubic); |
bsalomon@google.com | 3ab43d5 | 2012-10-11 19:39:09 +0000 | [diff] [blame] | 212 | |
| 213 | /** |
bsalomon@google.com | f4a9c82 | 2012-03-16 14:02:46 +0000 | [diff] [blame] | 214 | * Returns non-initialized instance. |
| 215 | */ |
| 216 | GrTextContext* getTextContext(); |
| 217 | |
robertphillips@google.com | 1f2f338 | 2013-08-29 11:54:56 +0000 | [diff] [blame] | 218 | typedef SkBitmapDevice INHERITED; |
reed@google.com | ac10a2d | 2010-12-22 21:39:39 +0000 | [diff] [blame] | 219 | }; |
| 220 | |
| 221 | #endif |