blob: 48e94e1b1d010d886fb71151ef085a060a1cb2da [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);
340 static bool isNonNormalizedInteger(Format format);
341 static int componentCount(Format format);
342
343 static void setTexturePalette(unsigned int *palette);
344
345 protected:
346 sw::Resource *resource;
347
348 private:
349 typedef unsigned char byte;
350 typedef unsigned short word;
351 typedef unsigned int dword;
352 typedef uint64_t qword;
353
354 #if S3TC_SUPPORT
355 struct DXT1
356 {
357 word c0;
358 word c1;
359 dword lut;
360 };
361
362 struct DXT3
363 {
364 qword a;
365
366 word c0;
367 word c1;
368 dword lut;
369 };
370
371 struct DXT5
372 {
373 union
374 {
375 struct
376 {
377 byte a0;
378 byte a1;
379 };
380
381 qword alut; // Skip first 16 bit
382 };
383
384 word c0;
385 word c1;
386 dword clut;
387 };
388 #endif
389
390 struct ATI2
391 {
392 union
393 {
394 struct
395 {
396 byte y0;
397 byte y1;
398 };
399
400 qword ylut; // Skip first 16 bit
401 };
402
403 union
404 {
405 struct
406 {
407 byte x0;
408 byte x1;
409 };
410
411 qword xlut; // Skip first 16 bit
412 };
413 };
414
415 struct ATI1
416 {
417 union
418 {
419 struct
420 {
421 byte r0;
422 byte r1;
423 };
424
425 qword rlut; // Skip first 16 bit
426 };
427 };
428
429 static void decodeR8G8B8(Buffer &destination, const Buffer &source);
430 static void decodeX1R5G5B5(Buffer &destination, const Buffer &source);
431 static void decodeA1R5G5B5(Buffer &destination, const Buffer &source);
432 static void decodeX4R4G4B4(Buffer &destination, const Buffer &source);
433 static void decodeA4R4G4B4(Buffer &destination, const Buffer &source);
434 static void decodeP8(Buffer &destination, const Buffer &source);
435
436 #if S3TC_SUPPORT
437 static void decodeDXT1(Buffer &internal, const Buffer &external);
438 static void decodeDXT3(Buffer &internal, const Buffer &external);
439 static void decodeDXT5(Buffer &internal, const Buffer &external);
440 #endif
441 static void decodeATI1(Buffer &internal, const Buffer &external);
442 static void decodeATI2(Buffer &internal, const Buffer &external);
443 static void decodeEAC(Buffer &internal, const Buffer &external, int nbChannels, bool isSigned);
444 static void decodeETC2(Buffer &internal, const Buffer &external, int nbAlphaBits, bool isSRGB);
445 static void decodeASTC(Buffer &internal, const Buffer &external, int xSize, int ySize, int zSize, bool isSRGB);
446
447 static void update(Buffer &destination, Buffer &source);
448 static void genericUpdate(Buffer &destination, Buffer &source);
449 static void *allocateBuffer(int width, int height, int depth, Format format);
450 static void memfill4(void *buffer, int pattern, int bytes);
451
452 bool identicalFormats() const;
453 Format selectInternalFormat(Format format) const;
454
455 void resolve();
456
457 Buffer external;
458 Buffer internal;
459 Buffer stencil;
460
461 const bool lockable;
462 const bool renderTarget;
463
464 bool dirtyMipmaps;
465 unsigned int paletteUsed;
466
467 static unsigned int *palette; // FIXME: Not multi-device safe
468 static unsigned int paletteID;
469
470 bool hasParent;
471 bool ownExternal;
472 };
473}
474
475#undef min
476#undef max
477
478namespace sw
479{
480 void *Surface::lock(int x, int y, int z, Lock lock, Accessor client, bool internal)
481 {
482 return internal ? lockInternal(x, y, z, lock, client) : lockExternal(x, y, z, lock, client);
483 }
484
485 void Surface::unlock(bool internal)
486 {
487 return internal ? unlockInternal() : unlockExternal();
488 }
489
490 int Surface::getWidth() const
491 {
492 return external.width;
493 }
494
495 int Surface::getHeight() const
496 {
497 return external.height;
498 }
499
500 int Surface::getDepth() const
501 {
502 return external.depth;
503 }
504
505 Format Surface::getFormat(bool internal) const
506 {
507 return internal ? getInternalFormat() : getExternalFormat();
508 }
509
510 int Surface::getPitchB(bool internal) const
511 {
512 return internal ? getInternalPitchB() : getExternalPitchB();
513 }
514
515 int Surface::getPitchP(bool internal) const
516 {
Nicolas Capens26b41162017-02-14 22:42:58 -0500517 return internal ? getInternalPitchP() : getExternalPitchP();
Nicolas Capens0bac2852016-05-07 06:09:58 -0400518 }
519
520 int Surface::getSliceB(bool internal) const
521 {
522 return internal ? getInternalSliceB() : getExternalSliceB();
523 }
524
525 int Surface::getSliceP(bool internal) const
526 {
Nicolas Capens26b41162017-02-14 22:42:58 -0500527 return internal ? getInternalSliceP() : getExternalSliceP();
Nicolas Capens0bac2852016-05-07 06:09:58 -0400528 }
529
530 Format Surface::getExternalFormat() const
531 {
532 return external.format;
533 }
534
535 int Surface::getExternalPitchB() const
536 {
537 return external.pitchB;
538 }
539
540 int Surface::getExternalPitchP() const
541 {
542 return external.pitchP;
543 }
544
545 int Surface::getExternalSliceB() const
546 {
547 return external.sliceB;
548 }
549
550 int Surface::getExternalSliceP() const
551 {
552 return external.sliceP;
553 }
554
555 Format Surface::getInternalFormat() const
556 {
557 return internal.format;
558 }
559
560 int Surface::getInternalPitchB() const
561 {
562 return internal.pitchB;
563 }
564
565 int Surface::getInternalPitchP() const
566 {
567 return internal.pitchP;
568 }
569
570 int Surface::getInternalSliceB() const
571 {
572 return internal.sliceB;
573 }
574
575 int Surface::getInternalSliceP() const
576 {
577 return internal.sliceP;
578 }
579
Alexis Hetua52dfbd2016-10-05 17:03:30 -0400580 Format Surface::getStencilFormat() const
581 {
582 return stencil.format;
583 }
584
Nicolas Capens0bac2852016-05-07 06:09:58 -0400585 int Surface::getStencilPitchB() const
586 {
587 return stencil.pitchB;
588 }
589
590 int Surface::getStencilSliceB() const
591 {
592 return stencil.sliceB;
593 }
594
595 int Surface::getMultiSampleCount() const
596 {
597 return sw::min(internal.depth, 4);
598 }
599
600 int Surface::getSuperSampleCount() const
601 {
602 return internal.depth > 4 ? internal.depth / 4 : 1;
603 }
604
605 bool Surface::isExternalDirty() const
606 {
607 return external.buffer && external.buffer != internal.buffer && external.dirty;
608 }
609}
610
611#endif // sw_Surface_hpp