blob: 4268a3a40b8fc10392ffb06ab46b53af526affac [file] [log] [blame]
Nicolas Capens0bac2852016-05-07 06:09:58 -04001// Copyright 2016 The SwiftShader Authors. All Rights Reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15#ifndef sw_Surface_hpp
16#define sw_Surface_hpp
17
18#include "Color.hpp"
19#include "Main/Config.hpp"
20#include "Common/Resource.hpp"
21
22namespace sw
23{
24 class Resource;
25
26 struct Rect
27 {
28 Rect() {}
29 Rect(int x0i, int y0i, int x1i, int y1i) : x0(x0i), y0(y0i), x1(x1i), y1(y1i) {}
30
31 void clip(int minX, int minY, int maxX, int maxY);
32
33 int width() const { return x1 - x0; }
34 int height() const { return y1 - y0; }
35
36 int x0; // Inclusive
37 int y0; // Inclusive
38 int x1; // Exclusive
39 int y1; // Exclusive
40 };
41
42 struct SliceRect : public Rect
43 {
44 SliceRect() : slice(0) {}
45 SliceRect(const Rect& rect) : Rect(rect), slice(0) {}
46 SliceRect(const Rect& rect, int s) : Rect(rect), slice(s) {}
47 SliceRect(int x0, int y0, int x1, int y1, int s) : Rect(x0, y0, x1, y1), slice(s) {}
48 int slice;
49 };
50
51 enum Format : unsigned char
52 {
53 FORMAT_NULL,
54
55 FORMAT_A8,
56 FORMAT_R8I,
57 FORMAT_R8UI,
58 FORMAT_R8I_SNORM,
59 FORMAT_R8, // UI_SNORM
60 FORMAT_R16I,
61 FORMAT_R16UI,
62 FORMAT_R32I,
63 FORMAT_R32UI,
64 FORMAT_R3G3B2,
65 FORMAT_A8R3G3B2,
66 FORMAT_X4R4G4B4,
67 FORMAT_A4R4G4B4,
68 FORMAT_R4G4B4A4,
69 FORMAT_R5G6B5,
70 FORMAT_R8G8B8,
71 FORMAT_B8G8R8,
72 FORMAT_X8R8G8B8,
73 FORMAT_A8R8G8B8,
74 FORMAT_X8B8G8R8I,
75 FORMAT_X8B8G8R8UI,
76 FORMAT_X8B8G8R8I_SNORM,
77 FORMAT_X8B8G8R8, // UI_SNORM
78 FORMAT_A8B8G8R8I,
79 FORMAT_A8B8G8R8UI,
80 FORMAT_A8B8G8R8I_SNORM,
81 FORMAT_A8B8G8R8, // UI_SNORM
82 FORMAT_SRGB8_X8,
83 FORMAT_SRGB8_A8,
84 FORMAT_X1R5G5B5,
85 FORMAT_A1R5G5B5,
86 FORMAT_R5G5B5A1,
87 FORMAT_G8R8I,
88 FORMAT_G8R8UI,
89 FORMAT_G8R8I_SNORM,
90 FORMAT_G8R8, // UI_SNORM
91 FORMAT_G16R16, // D3D format
92 FORMAT_G16R16I,
93 FORMAT_G16R16UI,
94 FORMAT_G32R32I,
95 FORMAT_G32R32UI,
96 FORMAT_A2R10G10B10,
97 FORMAT_A2B10G10R10,
98 FORMAT_A16B16G16R16, // D3D format
99 FORMAT_X16B16G16R16I,
100 FORMAT_X16B16G16R16UI,
101 FORMAT_A16B16G16R16I,
102 FORMAT_A16B16G16R16UI,
103 FORMAT_X32B32G32R32I,
104 FORMAT_X32B32G32R32UI,
105 FORMAT_A32B32G32R32I,
106 FORMAT_A32B32G32R32UI,
107 // Paletted formats
108 FORMAT_P8,
109 FORMAT_A8P8,
110 // Compressed formats
111 FORMAT_DXT1,
112 FORMAT_DXT3,
113 FORMAT_DXT5,
114 FORMAT_ATI1,
115 FORMAT_ATI2,
116 FORMAT_ETC1,
117 FORMAT_R11_EAC,
118 FORMAT_SIGNED_R11_EAC,
119 FORMAT_RG11_EAC,
120 FORMAT_SIGNED_RG11_EAC,
121 FORMAT_RGB8_ETC2,
122 FORMAT_SRGB8_ETC2,
123 FORMAT_RGB8_PUNCHTHROUGH_ALPHA1_ETC2,
124 FORMAT_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2,
125 FORMAT_RGBA8_ETC2_EAC,
126 FORMAT_SRGB8_ALPHA8_ETC2_EAC,
127 FORMAT_RGBA_ASTC_4x4_KHR,
128 FORMAT_RGBA_ASTC_5x4_KHR,
129 FORMAT_RGBA_ASTC_5x5_KHR,
130 FORMAT_RGBA_ASTC_6x5_KHR,
131 FORMAT_RGBA_ASTC_6x6_KHR,
132 FORMAT_RGBA_ASTC_8x5_KHR,
133 FORMAT_RGBA_ASTC_8x6_KHR,
134 FORMAT_RGBA_ASTC_8x8_KHR,
135 FORMAT_RGBA_ASTC_10x5_KHR,
136 FORMAT_RGBA_ASTC_10x6_KHR,
137 FORMAT_RGBA_ASTC_10x8_KHR,
138 FORMAT_RGBA_ASTC_10x10_KHR,
139 FORMAT_RGBA_ASTC_12x10_KHR,
140 FORMAT_RGBA_ASTC_12x12_KHR,
141 FORMAT_SRGB8_ALPHA8_ASTC_4x4_KHR,
142 FORMAT_SRGB8_ALPHA8_ASTC_5x4_KHR,
143 FORMAT_SRGB8_ALPHA8_ASTC_5x5_KHR,
144 FORMAT_SRGB8_ALPHA8_ASTC_6x5_KHR,
145 FORMAT_SRGB8_ALPHA8_ASTC_6x6_KHR,
146 FORMAT_SRGB8_ALPHA8_ASTC_8x5_KHR,
147 FORMAT_SRGB8_ALPHA8_ASTC_8x6_KHR,
148 FORMAT_SRGB8_ALPHA8_ASTC_8x8_KHR,
149 FORMAT_SRGB8_ALPHA8_ASTC_10x5_KHR,
150 FORMAT_SRGB8_ALPHA8_ASTC_10x6_KHR,
151 FORMAT_SRGB8_ALPHA8_ASTC_10x8_KHR,
152 FORMAT_SRGB8_ALPHA8_ASTC_10x10_KHR,
153 FORMAT_SRGB8_ALPHA8_ASTC_12x10_KHR,
154 FORMAT_SRGB8_ALPHA8_ASTC_12x12_KHR,
155 // Floating-point formats
156 FORMAT_A16F,
157 FORMAT_R16F,
158 FORMAT_G16R16F,
159 FORMAT_B16G16R16F,
160 FORMAT_A16B16G16R16F,
161 FORMAT_A32F,
162 FORMAT_R32F,
163 FORMAT_G32R32F,
164 FORMAT_B32G32R32F,
165 FORMAT_X32B32G32R32F,
166 FORMAT_A32B32G32R32F,
167 // Bump map formats
168 FORMAT_V8U8,
169 FORMAT_L6V5U5,
170 FORMAT_Q8W8V8U8,
171 FORMAT_X8L8V8U8,
172 FORMAT_A2W10V10U10,
173 FORMAT_V16U16,
174 FORMAT_A16W16V16U16,
175 FORMAT_Q16W16V16U16,
176 // Luminance formats
177 FORMAT_L8,
178 FORMAT_A4L4,
179 FORMAT_L16,
180 FORMAT_A8L8,
181 FORMAT_L16F,
182 FORMAT_A16L16F,
183 FORMAT_L32F,
184 FORMAT_A32L32F,
185 // Depth/stencil formats
186 FORMAT_D16,
187 FORMAT_D32,
188 FORMAT_D24X8,
189 FORMAT_D24S8,
190 FORMAT_D24FS8,
191 FORMAT_D32F, // Quad layout
192 FORMAT_D32F_COMPLEMENTARY, // Quad layout, 1 - z
193 FORMAT_D32F_LOCKABLE, // Linear layout
194 FORMAT_D32FS8_TEXTURE, // Linear layout, no PCF
195 FORMAT_D32FS8_SHADOW, // Linear layout, PCF
196 FORMAT_DF24S8,
197 FORMAT_DF16S8,
198 FORMAT_INTZ,
199 FORMAT_S8,
200 // Quad layout framebuffer
201 FORMAT_X8G8R8B8Q,
202 FORMAT_A8G8R8B8Q,
203 // YUV formats
204 FORMAT_YV12_BT601,
205 FORMAT_YV12_BT709,
206 FORMAT_YV12_JFIF, // Full-swing BT.601
207
208 FORMAT_LAST = FORMAT_YV12_JFIF
209 };
210
211 enum Lock
212 {
213 LOCK_UNLOCKED,
214 LOCK_READONLY,
215 LOCK_WRITEONLY,
216 LOCK_READWRITE,
217 LOCK_DISCARD
218 };
219
220 class Surface
221 {
222 private:
223 struct Buffer
224 {
225 public:
226 void write(int x, int y, int z, const Color<float> &color);
227 void write(int x, int y, const Color<float> &color);
228 void write(void *element, const Color<float> &color);
229 Color<float> read(int x, int y, int z) const;
230 Color<float> read(int x, int y) const;
231 Color<float> read(void *element) const;
232 Color<float> sample(float x, float y, float z) const;
233 Color<float> sample(float x, float y) const;
234
235 void *lockRect(int x, int y, int z, Lock lock);
236 void unlockRect();
237
238 void *buffer;
239 int width;
240 int height;
241 int depth;
242 int bytes;
243 int pitchB;
244 int pitchP;
245 int sliceB;
246 int sliceP;
247 Format format;
248 Lock lock;
249
250 bool dirty;
251 };
252
Nicolas Capens8560e682017-02-07 13:44:24 -0500253 virtual void typeinfo(); // Dummy key method (https://gcc.gnu.org/onlinedocs/gcc/Vague-Linkage.html)
254
Nicolas Capens0bac2852016-05-07 06:09:58 -0400255 public:
256 Surface(int width, int height, int depth, Format format, void *pixels, int pitch, int slice);
257 Surface(Resource *texture, int width, int height, int depth, Format format, bool lockable, bool renderTarget, int pitchP = 0);
258
259 virtual ~Surface();
260
261 inline void *lock(int x, int y, int z, Lock lock, Accessor client, bool internal = false);
262 inline void unlock(bool internal = false);
263 inline int getWidth() const;
264 inline int getHeight() const;
265 inline int getDepth() const;
266 inline Format getFormat(bool internal = false) const;
267 inline int getPitchB(bool internal = false) const;
268 inline int getPitchP(bool internal = false) const;
269 inline int getSliceB(bool internal = false) const;
270 inline int getSliceP(bool internal = false) const;
271
272 void *lockExternal(int x, int y, int z, Lock lock, Accessor client);
273 void unlockExternal();
274 inline Format getExternalFormat() const;
275 inline int getExternalPitchB() const;
276 inline int getExternalPitchP() const;
277 inline int getExternalSliceB() const;
278 inline int getExternalSliceP() const;
279
280 virtual void *lockInternal(int x, int y, int z, Lock lock, Accessor client);
281 virtual void unlockInternal();
282 inline Format getInternalFormat() const;
283 inline int getInternalPitchB() const;
284 inline int getInternalPitchP() const;
285 inline int getInternalSliceB() const;
286 inline int getInternalSliceP() const;
287
Alexis Hetua52dfbd2016-10-05 17:03:30 -0400288 void *lockStencil(int x, int y, int front, Accessor client);
Nicolas Capens0bac2852016-05-07 06:09:58 -0400289 void unlockStencil();
Alexis Hetua52dfbd2016-10-05 17:03:30 -0400290 inline Format getStencilFormat() const;
Nicolas Capens0bac2852016-05-07 06:09:58 -0400291 inline int getStencilPitchB() const;
292 inline int getStencilSliceB() const;
293
294 inline int getMultiSampleCount() const;
295 inline int getSuperSampleCount() const;
296
297 bool isEntire(const SliceRect& rect) const;
298 SliceRect getRect() const;
299 void clearDepth(float depth, int x0, int y0, int width, int height);
300 void clearStencil(unsigned char stencil, unsigned char mask, int x0, int y0, int width, int height);
301 void fill(const Color<float> &color, int x0, int y0, int width, int height);
302
303 Color<float> readExternal(int x, int y, int z) const;
304 Color<float> readExternal(int x, int y) const;
305 Color<float> sampleExternal(float x, float y, float z) const;
306 Color<float> sampleExternal(float x, float y) const;
307 void writeExternal(int x, int y, int z, const Color<float> &color);
308 void writeExternal(int x, int y, const Color<float> &color);
309
310 void copyInternal(const Surface* src, int x, int y, float srcX, float srcY, bool filter);
311 void copyInternal(const Surface* src, int x, int y, int z, float srcX, float srcY, float srcZ, bool filter);
312
313 bool hasStencil() const;
314 bool hasDepth() const;
315 bool hasPalette() const;
316 bool isRenderTarget() const;
317
318 bool hasDirtyMipmaps() const;
319 void cleanMipmaps();
320 inline bool isExternalDirty() const;
321 Resource *getResource();
322
323 static int bytes(Format format);
324 static int pitchB(int width, Format format, bool target);
325 static int pitchP(int width, Format format, bool target);
326 static int sliceB(int width, int height, Format format, bool target);
327 static int sliceP(int width, int height, Format format, bool target);
328 static unsigned int size(int width, int height, int depth, Format format); // FIXME: slice * depth
329
330 static bool isStencil(Format format);
331 static bool isDepth(Format format);
Alexis Hetub9dda642016-10-06 11:25:32 -0400332 static bool hasQuadLayout(Format format);
Nicolas Capens0bac2852016-05-07 06:09:58 -0400333 static bool isPalette(Format format);
334
335 static bool isFloatFormat(Format format);
336 static bool isUnsignedComponent(Format format, int component);
337 static bool isSRGBreadable(Format format);
338 static bool isSRGBwritable(Format format);
339 static bool isCompressed(Format format);
Nicolas Capens492887a2017-03-27 14:50:51 -0400340 static bool isSignedNonNormalizedInteger(Format format);
341 static bool isUnsignedNonNormalizedInteger(Format format);
Nicolas Capens0bac2852016-05-07 06:09:58 -0400342 static bool isNonNormalizedInteger(Format format);
Nicolas Capens492887a2017-03-27 14:50:51 -0400343 static bool isNormalizedInteger(Format format);
Nicolas Capens0bac2852016-05-07 06:09:58 -0400344 static int componentCount(Format format);
345
346 static void setTexturePalette(unsigned int *palette);
347
348 protected:
349 sw::Resource *resource;
350
351 private:
352 typedef unsigned char byte;
353 typedef unsigned short word;
354 typedef unsigned int dword;
355 typedef uint64_t qword;
356
357 #if S3TC_SUPPORT
358 struct DXT1
359 {
360 word c0;
361 word c1;
362 dword lut;
363 };
364
365 struct DXT3
366 {
367 qword a;
368
369 word c0;
370 word c1;
371 dword lut;
372 };
373
374 struct DXT5
375 {
376 union
377 {
378 struct
379 {
380 byte a0;
381 byte a1;
382 };
383
384 qword alut; // Skip first 16 bit
385 };
386
387 word c0;
388 word c1;
389 dword clut;
390 };
391 #endif
392
393 struct ATI2
394 {
395 union
396 {
397 struct
398 {
399 byte y0;
400 byte y1;
401 };
402
403 qword ylut; // Skip first 16 bit
404 };
405
406 union
407 {
408 struct
409 {
410 byte x0;
411 byte x1;
412 };
413
414 qword xlut; // Skip first 16 bit
415 };
416 };
417
418 struct ATI1
419 {
420 union
421 {
422 struct
423 {
424 byte r0;
425 byte r1;
426 };
427
428 qword rlut; // Skip first 16 bit
429 };
430 };
431
432 static void decodeR8G8B8(Buffer &destination, const Buffer &source);
433 static void decodeX1R5G5B5(Buffer &destination, const Buffer &source);
434 static void decodeA1R5G5B5(Buffer &destination, const Buffer &source);
435 static void decodeX4R4G4B4(Buffer &destination, const Buffer &source);
436 static void decodeA4R4G4B4(Buffer &destination, const Buffer &source);
437 static void decodeP8(Buffer &destination, const Buffer &source);
438
439 #if S3TC_SUPPORT
440 static void decodeDXT1(Buffer &internal, const Buffer &external);
441 static void decodeDXT3(Buffer &internal, const Buffer &external);
442 static void decodeDXT5(Buffer &internal, const Buffer &external);
443 #endif
444 static void decodeATI1(Buffer &internal, const Buffer &external);
445 static void decodeATI2(Buffer &internal, const Buffer &external);
446 static void decodeEAC(Buffer &internal, const Buffer &external, int nbChannels, bool isSigned);
447 static void decodeETC2(Buffer &internal, const Buffer &external, int nbAlphaBits, bool isSRGB);
448 static void decodeASTC(Buffer &internal, const Buffer &external, int xSize, int ySize, int zSize, bool isSRGB);
449
450 static void update(Buffer &destination, Buffer &source);
451 static void genericUpdate(Buffer &destination, Buffer &source);
452 static void *allocateBuffer(int width, int height, int depth, Format format);
453 static void memfill4(void *buffer, int pattern, int bytes);
454
455 bool identicalFormats() const;
456 Format selectInternalFormat(Format format) const;
457
458 void resolve();
459
460 Buffer external;
461 Buffer internal;
462 Buffer stencil;
463
464 const bool lockable;
465 const bool renderTarget;
466
467 bool dirtyMipmaps;
468 unsigned int paletteUsed;
469
470 static unsigned int *palette; // FIXME: Not multi-device safe
471 static unsigned int paletteID;
472
473 bool hasParent;
474 bool ownExternal;
475 };
476}
477
478#undef min
479#undef max
480
481namespace sw
482{
483 void *Surface::lock(int x, int y, int z, Lock lock, Accessor client, bool internal)
484 {
485 return internal ? lockInternal(x, y, z, lock, client) : lockExternal(x, y, z, lock, client);
486 }
487
488 void Surface::unlock(bool internal)
489 {
490 return internal ? unlockInternal() : unlockExternal();
491 }
492
493 int Surface::getWidth() const
494 {
495 return external.width;
496 }
497
498 int Surface::getHeight() const
499 {
500 return external.height;
501 }
502
503 int Surface::getDepth() const
504 {
505 return external.depth;
506 }
507
508 Format Surface::getFormat(bool internal) const
509 {
510 return internal ? getInternalFormat() : getExternalFormat();
511 }
512
513 int Surface::getPitchB(bool internal) const
514 {
515 return internal ? getInternalPitchB() : getExternalPitchB();
516 }
517
518 int Surface::getPitchP(bool internal) const
519 {
Nicolas Capens26b41162017-02-14 22:42:58 -0500520 return internal ? getInternalPitchP() : getExternalPitchP();
Nicolas Capens0bac2852016-05-07 06:09:58 -0400521 }
522
523 int Surface::getSliceB(bool internal) const
524 {
525 return internal ? getInternalSliceB() : getExternalSliceB();
526 }
527
528 int Surface::getSliceP(bool internal) const
529 {
Nicolas Capens26b41162017-02-14 22:42:58 -0500530 return internal ? getInternalSliceP() : getExternalSliceP();
Nicolas Capens0bac2852016-05-07 06:09:58 -0400531 }
532
533 Format Surface::getExternalFormat() const
534 {
535 return external.format;
536 }
537
538 int Surface::getExternalPitchB() const
539 {
540 return external.pitchB;
541 }
542
543 int Surface::getExternalPitchP() const
544 {
545 return external.pitchP;
546 }
547
548 int Surface::getExternalSliceB() const
549 {
550 return external.sliceB;
551 }
552
553 int Surface::getExternalSliceP() const
554 {
555 return external.sliceP;
556 }
557
558 Format Surface::getInternalFormat() const
559 {
560 return internal.format;
561 }
562
563 int Surface::getInternalPitchB() const
564 {
565 return internal.pitchB;
566 }
567
568 int Surface::getInternalPitchP() const
569 {
570 return internal.pitchP;
571 }
572
573 int Surface::getInternalSliceB() const
574 {
575 return internal.sliceB;
576 }
577
578 int Surface::getInternalSliceP() const
579 {
580 return internal.sliceP;
581 }
582
Alexis Hetua52dfbd2016-10-05 17:03:30 -0400583 Format Surface::getStencilFormat() const
584 {
585 return stencil.format;
586 }
587
Nicolas Capens0bac2852016-05-07 06:09:58 -0400588 int Surface::getStencilPitchB() const
589 {
590 return stencil.pitchB;
591 }
592
593 int Surface::getStencilSliceB() const
594 {
595 return stencil.sliceB;
596 }
597
598 int Surface::getMultiSampleCount() const
599 {
600 return sw::min(internal.depth, 4);
601 }
602
603 int Surface::getSuperSampleCount() const
604 {
605 return internal.depth > 4 ? internal.depth / 4 : 1;
606 }
607
608 bool Surface::isExternalDirty() const
609 {
610 return external.buffer && external.buffer != internal.buffer && external.dirty;
611 }
612}
613
614#endif // sw_Surface_hpp