1#!/usr/bin/env python 2 3CopyRight = ''' 4/************************************************************************** 5 * 6 * Copyright 2009 VMware, Inc. 7 * All Rights Reserved. 8 * 9 * Permission is hereby granted, free of charge, to any person obtaining a 10 * copy of this software and associated documentation files (the 11 * "Software"), to deal in the Software without restriction, including 12 * without limitation the rights to use, copy, modify, merge, publish, 13 * distribute, sub license, and/or sell copies of the Software, and to 14 * permit persons to whom the Software is furnished to do so, subject to 15 * the following conditions: 16 * 17 * The above copyright notice and this permission notice (including the 18 * next paragraph) shall be included in all copies or substantial portions 19 * of the Software. 20 * 21 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 22 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 23 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. 24 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR 25 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, 26 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE 27 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 28 * 29 **************************************************************************/ 30 31/** 32 * @file 33 * Pixel format accessor functions. 34 * 35 * @author Jose Fonseca <jfonseca@vmware.com> 36 */ 37''' 38 39 40import sys 41import os.path 42 43sys.path.insert(0, os.path.join(os.path.dirname(sys.argv[0]), '../../auxiliary/util')) 44 45from u_format_pack import * 46 47 48def is_format_supported(format): 49 '''Determines whether we actually have the plumbing necessary to generate the 50 to read/write to/from this format.''' 51 52 # FIXME: Ideally we would support any format combination here. 53 54 if format.name == 'PIPE_FORMAT_R11G11B10_FLOAT': 55 return True; 56 57 if format.name == 'PIPE_FORMAT_R9G9B9E5_FLOAT': 58 return True; 59 60 if format.layout != PLAIN: 61 return False 62 63 for i in range(4): 64 channel = format.channels[i] 65 if channel.type not in (VOID, UNSIGNED, SIGNED, FLOAT): 66 return False 67 if channel.type == FLOAT and channel.size not in (16, 32 ,64): 68 return False 69 70 if format.colorspace not in ('rgb', 'srgb'): 71 return False 72 73 return True 74 75 76def generate_format_read(format, dst_channel, dst_native_type, dst_suffix): 77 '''Generate the function to read pixels from a particular format''' 78 79 name = format.short_name() 80 81 src_native_type = native_type(format) 82 83 print 'static void' 84 print 'lp_tile_%s_swizzle_%s(%s * restrict dst, const uint8_t * restrict src, unsigned src_stride, unsigned x0, unsigned y0)' % (name, dst_suffix, dst_native_type) 85 print '{' 86 print ' unsigned x, y;' 87 print ' const uint8_t *src_row = src + y0*src_stride;' 88 print ' for (y = 0; y < TILE_SIZE; ++y) {' 89 print ' const %s *src_pixel = (const %s *)(src_row + x0*%u);' % (src_native_type, src_native_type, format.stride()) 90 print ' for (x = 0; x < TILE_SIZE; ++x) {' 91 92 names = ['']*4 93 if format.colorspace in ('rgb', 'srgb'): 94 for i in range(4): 95 swizzle = format.swizzles[i] 96 if swizzle < 4: 97 names[swizzle] += 'rgba'[i] 98 elif format.colorspace == 'zs': 99 swizzle = format.swizzles[0] 100 if swizzle < 4: 101 names[swizzle] = 'z' 102 else: 103 assert False 104 else: 105 assert False 106 107 if format.name == 'PIPE_FORMAT_R11G11B10_FLOAT': 108 print ' float tmp[3];' 109 print ' uint8_t r, g, b;' 110 print ' r11g11b10f_to_float3(*src_pixel++, tmp);' 111 for i in range(3): 112 print ' %s = tmp[%d] * 0xff;' % (names[i], i) 113 elif format.name == 'PIPE_FORMAT_R9G9B9E5_FLOAT': 114 print ' float tmp[3];' 115 print ' uint8_t r, g, b;' 116 print ' rgb9e5_to_float3(*src_pixel++, tmp);' 117 for i in range(3): 118 print ' %s = tmp[%d] * 0xff;' % (names[i], i) 119 elif format.layout == PLAIN: 120 if not format.is_array(): 121 print ' %s pixel = *src_pixel++;' % src_native_type 122 shift = 0; 123 for i in range(4): 124 src_channel = format.channels[i] 125 width = src_channel.size 126 if names[i]: 127 value = 'pixel' 128 mask = (1 << width) - 1 129 if shift: 130 value = '(%s >> %u)' % (value, shift) 131 if shift + width < format.block_size(): 132 value = '(%s & 0x%x)' % (value, mask) 133 value = conversion_expr(src_channel, dst_channel, dst_native_type, value, clamp=False) 134 print ' %s %s = %s;' % (dst_native_type, names[i], value) 135 shift += width 136 else: 137 for i in range(4): 138 if names[i]: 139 print ' %s %s;' % (dst_native_type, names[i]) 140 for i in range(4): 141 src_channel = format.channels[i] 142 if names[i]: 143 value = '(*src_pixel++)' 144 value = conversion_expr(src_channel, dst_channel, dst_native_type, value, clamp=False) 145 print ' %s = %s;' % (names[i], value) 146 elif src_channel.size: 147 print ' ++src_pixel;' 148 else: 149 assert False 150 151 for i in range(4): 152 if format.colorspace in ('rgb', 'srgb'): 153 swizzle = format.swizzles[i] 154 if swizzle < 4: 155 value = names[swizzle] 156 elif swizzle == SWIZZLE_0: 157 value = '0' 158 elif swizzle == SWIZZLE_1: 159 value = get_one(dst_channel) 160 else: 161 assert False 162 elif format.colorspace == 'zs': 163 if i < 3: 164 value = 'z' 165 else: 166 value = get_one(dst_channel) 167 else: 168 assert False 169 print ' TILE_PIXEL(dst, x, y, %u) = %s; /* %s */' % (i, value, 'rgba'[i]) 170 171 print ' }' 172 print ' src_row += src_stride;' 173 print ' }' 174 print '}' 175 print 176 177 178def pack_rgba(format, src_channel, r, g, b, a): 179 """Return an expression for packing r, g, b, a into a pixel of the 180 given format. Ex: '(b << 24) | (g << 16) | (r << 8) | (a << 0)' 181 """ 182 assert format.colorspace in ('rgb', 'srgb') 183 inv_swizzle = format.inv_swizzles() 184 shift = 0 185 expr = None 186 for i in range(4): 187 # choose r, g, b, or a depending on the inverse swizzle term 188 if inv_swizzle[i] == 0: 189 value = r 190 elif inv_swizzle[i] == 1: 191 value = g 192 elif inv_swizzle[i] == 2: 193 value = b 194 elif inv_swizzle[i] == 3: 195 value = a 196 else: 197 value = None 198 199 if value: 200 dst_channel = format.channels[i] 201 dst_native_type = native_type(format) 202 value = conversion_expr(src_channel, dst_channel, dst_native_type, value, clamp=False) 203 term = "((%s) << %d)" % (value, shift) 204 if expr: 205 expr = expr + " | " + term 206 else: 207 expr = term 208 209 width = format.channels[i].size 210 shift = shift + width 211 return expr 212 213 214def emit_unrolled_unswizzle_code(format, src_channel): 215 '''Emit code for writing a block based on unrolled loops. 216 This is considerably faster than the TILE_PIXEL-based code below. 217 ''' 218 dst_native_type = 'uint%u_t' % format.block_size() 219 print ' const unsigned dstpix_stride = dst_stride / %d;' % format.stride() 220 print ' %s *dstpix = (%s *) dst;' % (dst_native_type, dst_native_type) 221 print ' unsigned int qx, qy, i;' 222 print 223 print ' for (qy = 0; qy < TILE_SIZE; qy += TILE_VECTOR_HEIGHT) {' 224 print ' const unsigned py = y0 + qy;' 225 print ' for (qx = 0; qx < TILE_SIZE; qx += TILE_VECTOR_WIDTH) {' 226 print ' const unsigned px = x0 + qx;' 227 print ' const uint8_t *r = src + 0 * TILE_C_STRIDE;' 228 print ' const uint8_t *g = src + 1 * TILE_C_STRIDE;' 229 print ' const uint8_t *b = src + 2 * TILE_C_STRIDE;' 230 print ' const uint8_t *a = src + 3 * TILE_C_STRIDE;' 231 print ' (void) r; (void) g; (void) b; (void) a; /* silence warnings */' 232 print ' for (i = 0; i < TILE_C_STRIDE; i += 2) {' 233 print ' const uint32_t pixel0 = %s;' % pack_rgba(format, src_channel, "r[i+0]", "g[i+0]", "b[i+0]", "a[i+0]") 234 print ' const uint32_t pixel1 = %s;' % pack_rgba(format, src_channel, "r[i+1]", "g[i+1]", "b[i+1]", "a[i+1]") 235 print ' const unsigned offset = (py + tile_y_offset[i]) * dstpix_stride + (px + tile_x_offset[i]);' 236 print ' dstpix[offset + 0] = pixel0;' 237 print ' dstpix[offset + 1] = pixel1;' 238 print ' }' 239 print ' src += TILE_X_STRIDE;' 240 print ' }' 241 print ' }' 242 243 244def emit_tile_pixel_unswizzle_code(format, src_channel): 245 '''Emit code for writing a block based on the TILE_PIXEL macro.''' 246 dst_native_type = native_type(format) 247 248 inv_swizzle = format.inv_swizzles() 249 250 print ' unsigned x, y;' 251 print ' uint8_t *dst_row = dst + y0*dst_stride;' 252 print ' for (y = 0; y < TILE_SIZE; ++y) {' 253 print ' %s *dst_pixel = (%s *)(dst_row + x0*%u);' % (dst_native_type, dst_native_type, format.stride()) 254 print ' for (x = 0; x < TILE_SIZE; ++x) {' 255 256 if format.name == 'PIPE_FORMAT_R11G11B10_FLOAT': 257 print ' float tmp[3];' 258 for i in range(3): 259 print ' tmp[%d] = ubyte_to_float(TILE_PIXEL(src, x, y, %u));' % (i, inv_swizzle[i]) 260 print ' *dst_pixel++ = float3_to_r11g11b10f(tmp);' 261 elif format.name == 'PIPE_FORMAT_R9G9B9E5_FLOAT': 262 print ' float tmp[3];' 263 for i in range(3): 264 print ' tmp[%d] = ubyte_to_float(TILE_PIXEL(src, x, y, %u));' % (i, inv_swizzle[i]) 265 print ' *dst_pixel++ = float3_to_rgb9e5(tmp);' 266 elif format.layout == PLAIN: 267 if not format.is_array(): 268 print ' %s pixel = 0;' % dst_native_type 269 shift = 0; 270 for i in range(4): 271 dst_channel = format.channels[i] 272 width = dst_channel.size 273 if inv_swizzle[i] is not None: 274 value = 'TILE_PIXEL(src, x, y, %u)' % inv_swizzle[i] 275 value = conversion_expr(src_channel, dst_channel, dst_native_type, value, clamp=False) 276 if shift: 277 value = '(%s << %u)' % (value, shift) 278 print ' pixel |= %s;' % value 279 shift += width 280 print ' *dst_pixel++ = pixel;' 281 else: 282 for i in range(4): 283 dst_channel = format.channels[i] 284 if inv_swizzle[i] is not None: 285 value = 'TILE_PIXEL(src, x, y, %u)' % inv_swizzle[i] 286 value = conversion_expr(src_channel, dst_channel, dst_native_type, value, clamp=False) 287 print ' *dst_pixel++ = %s;' % value 288 elif dst_channel.size: 289 print ' ++dst_pixel;' 290 else: 291 assert False 292 293 print ' }' 294 print ' dst_row += dst_stride;' 295 print ' }' 296 297 298def generate_format_write(format, src_channel, src_native_type, src_suffix): 299 '''Generate the function to write pixels to a particular format''' 300 301 name = format.short_name() 302 303 print 'static void' 304 print 'lp_tile_%s_unswizzle_%s(const %s * restrict src, uint8_t * restrict dst, unsigned dst_stride, unsigned x0, unsigned y0)' % (name, src_suffix, src_native_type) 305 print '{' 306 if format.layout == PLAIN \ 307 and format.colorspace == 'rgb' \ 308 and format.block_size() <= 32 \ 309 and format.is_pot() \ 310 and not format.is_mixed() \ 311 and (format.channels[0].type == UNSIGNED \ 312 or format.channels[1].type == UNSIGNED): 313 emit_unrolled_unswizzle_code(format, src_channel) 314 else: 315 emit_tile_pixel_unswizzle_code(format, src_channel) 316 print '}' 317 print 318 319 320def generate_sse2(): 321 print ''' 322#if defined(PIPE_ARCH_SSE) 323 324#include "util/u_sse.h" 325 326static ALWAYS_INLINE void 327swz4( const __m128i * restrict x, 328 const __m128i * restrict y, 329 const __m128i * restrict z, 330 const __m128i * restrict w, 331 __m128i * restrict a, 332 __m128i * restrict b, 333 __m128i * restrict c, 334 __m128i * restrict d) 335{ 336 __m128i i, j, k, l; 337 __m128i m, n, o, p; 338 __m128i e, f, g, h; 339 340 m = _mm_unpacklo_epi8(*x,*y); 341 n = _mm_unpackhi_epi8(*x,*y); 342 o = _mm_unpacklo_epi8(*z,*w); 343 p = _mm_unpackhi_epi8(*z,*w); 344 345 i = _mm_unpacklo_epi16(m,n); 346 j = _mm_unpackhi_epi16(m,n); 347 k = _mm_unpacklo_epi16(o,p); 348 l = _mm_unpackhi_epi16(o,p); 349 350 e = _mm_unpacklo_epi8(i,j); 351 f = _mm_unpackhi_epi8(i,j); 352 g = _mm_unpacklo_epi8(k,l); 353 h = _mm_unpackhi_epi8(k,l); 354 355 *a = _mm_unpacklo_epi64(e,g); 356 *b = _mm_unpackhi_epi64(e,g); 357 *c = _mm_unpacklo_epi64(f,h); 358 *d = _mm_unpackhi_epi64(f,h); 359} 360 361static ALWAYS_INLINE void 362unswz4( const __m128i * restrict a, 363 const __m128i * restrict b, 364 const __m128i * restrict c, 365 const __m128i * restrict d, 366 __m128i * restrict x, 367 __m128i * restrict y, 368 __m128i * restrict z, 369 __m128i * restrict w) 370{ 371 __m128i i, j, k, l; 372 __m128i m, n, o, p; 373 374 i = _mm_unpacklo_epi8(*a,*b); 375 j = _mm_unpackhi_epi8(*a,*b); 376 k = _mm_unpacklo_epi8(*c,*d); 377 l = _mm_unpackhi_epi8(*c,*d); 378 379 m = _mm_unpacklo_epi16(i,k); 380 n = _mm_unpackhi_epi16(i,k); 381 o = _mm_unpacklo_epi16(j,l); 382 p = _mm_unpackhi_epi16(j,l); 383 384 *x = _mm_unpacklo_epi64(m,n); 385 *y = _mm_unpackhi_epi64(m,n); 386 *z = _mm_unpacklo_epi64(o,p); 387 *w = _mm_unpackhi_epi64(o,p); 388} 389 390static void 391lp_tile_b8g8r8a8_unorm_swizzle_4ub_sse2(uint8_t * restrict dst, 392 const uint8_t * restrict src, unsigned src_stride, 393 unsigned x0, unsigned y0) 394{ 395 __m128i *dst128 = (__m128i *) dst; 396 unsigned x, y; 397 398 src += y0 * src_stride; 399 src += x0 * sizeof(uint32_t); 400 401 for (y = 0; y < TILE_SIZE; y += 4) { 402 const uint8_t *src_row = src; 403 404 for (x = 0; x < TILE_SIZE; x += 4) { 405 swz4((const __m128i *) (src_row + 0 * src_stride), 406 (const __m128i *) (src_row + 1 * src_stride), 407 (const __m128i *) (src_row + 2 * src_stride), 408 (const __m128i *) (src_row + 3 * src_stride), 409 dst128 + 2, /* b */ 410 dst128 + 1, /* g */ 411 dst128 + 0, /* r */ 412 dst128 + 3); /* a */ 413 414 dst128 += 4; 415 src_row += sizeof(__m128i); 416 } 417 418 src += 4 * src_stride; 419 } 420} 421 422static void 423lp_tile_b8g8r8a8_unorm_unswizzle_4ub_sse2(const uint8_t * restrict src, 424 uint8_t * restrict dst, unsigned dst_stride, 425 unsigned x0, unsigned y0) 426{ 427 unsigned int x, y; 428 const __m128i *src128 = (const __m128i *) src; 429 430 dst += y0 * dst_stride; 431 dst += x0 * sizeof(uint32_t); 432 433 for (y = 0; y < TILE_SIZE; y += 4) { 434 const uint8_t *dst_row = dst; 435 436 for (x = 0; x < TILE_SIZE; x += 4) { 437 unswz4( &src128[2], /* b */ 438 &src128[1], /* g */ 439 &src128[0], /* r */ 440 &src128[3], /* a */ 441 (__m128i *) (dst_row + 0 * dst_stride), 442 (__m128i *) (dst_row + 1 * dst_stride), 443 (__m128i *) (dst_row + 2 * dst_stride), 444 (__m128i *) (dst_row + 3 * dst_stride)); 445 446 src128 += 4; 447 dst_row += sizeof(__m128i);; 448 } 449 450 dst += 4 * dst_stride; 451 } 452} 453 454static void 455lp_tile_b8g8r8x8_unorm_swizzle_4ub_sse2(uint8_t * restrict dst, 456 const uint8_t * restrict src, unsigned src_stride, 457 unsigned x0, unsigned y0) 458{ 459 __m128i *dst128 = (__m128i *) dst; 460 unsigned x, y; 461 462 src += y0 * src_stride; 463 src += x0 * sizeof(uint32_t); 464 465 for (y = 0; y < TILE_SIZE; y += 4) { 466 const uint8_t *src_row = src; 467 468 for (x = 0; x < TILE_SIZE; x += 4) { 469 swz4((const __m128i *) (src_row + 0 * src_stride), 470 (const __m128i *) (src_row + 1 * src_stride), 471 (const __m128i *) (src_row + 2 * src_stride), 472 (const __m128i *) (src_row + 3 * src_stride), 473 dst128 + 2, /* b */ 474 dst128 + 1, /* g */ 475 dst128 + 0, /* r */ 476 dst128 + 3); /* a */ 477 478 dst128 += 4; 479 src_row += sizeof(__m128i); 480 } 481 482 src += 4 * src_stride; 483 } 484} 485 486static void 487lp_tile_b8g8r8x8_unorm_unswizzle_4ub_sse2(const uint8_t * restrict src, 488 uint8_t * restrict dst, unsigned dst_stride, 489 unsigned x0, unsigned y0) 490{ 491 unsigned int x, y; 492 const __m128i *src128 = (const __m128i *) src; 493 494 dst += y0 * dst_stride; 495 dst += x0 * sizeof(uint32_t); 496 497 for (y = 0; y < TILE_SIZE; y += 4) { 498 const uint8_t *dst_row = dst; 499 500 for (x = 0; x < TILE_SIZE; x += 4) { 501 unswz4( &src128[2], /* b */ 502 &src128[1], /* g */ 503 &src128[0], /* r */ 504 &src128[3], /* a */ 505 (__m128i *) (dst_row + 0 * dst_stride), 506 (__m128i *) (dst_row + 1 * dst_stride), 507 (__m128i *) (dst_row + 2 * dst_stride), 508 (__m128i *) (dst_row + 3 * dst_stride)); 509 510 src128 += 4; 511 dst_row += sizeof(__m128i);; 512 } 513 514 dst += 4 * dst_stride; 515 } 516} 517 518#endif /* PIPE_ARCH_SSE */ 519''' 520 521 522def generate_swizzle(formats, dst_channel, dst_native_type, dst_suffix): 523 '''Generate the dispatch function to read pixels from any format''' 524 525 for format in formats: 526 if is_format_supported(format): 527 generate_format_read(format, dst_channel, dst_native_type, dst_suffix) 528 529 print 'void' 530 print 'lp_tile_swizzle_%s(enum pipe_format format, %s *dst, const void *src, unsigned src_stride, unsigned x, unsigned y)' % (dst_suffix, dst_native_type) 531 print '{' 532 print ' void (*func)(%s * restrict dst, const uint8_t * restrict src, unsigned src_stride, unsigned x0, unsigned y0);' % dst_native_type 533 print '#ifdef DEBUG' 534 print ' lp_tile_swizzle_count += 1;' 535 print '#endif' 536 print ' switch(format) {' 537 for format in formats: 538 if is_format_supported(format): 539 print ' case %s:' % format.name 540 func_name = 'lp_tile_%s_swizzle_%s' % (format.short_name(), dst_suffix) 541 if format.name == 'PIPE_FORMAT_B8G8R8A8_UNORM' or format.name == 'PIPE_FORMAT_B8G8R8X8_UNORM': 542 print '#ifdef PIPE_ARCH_SSE' 543 print ' func = util_cpu_caps.has_sse2 ? %s_sse2 : %s;' % (func_name, func_name) 544 print '#else' 545 print ' func = %s;' % (func_name,) 546 print '#endif' 547 else: 548 print ' func = %s;' % (func_name,) 549 print ' break;' 550 print ' default:' 551 print ' debug_printf("%s: unsupported format %s\\n", __FUNCTION__, util_format_name(format));' 552 print ' return;' 553 print ' }' 554 print ' func(dst, (const uint8_t *)src, src_stride, x, y);' 555 print '}' 556 print 557 558 559def generate_unswizzle(formats, src_channel, src_native_type, src_suffix): 560 '''Generate the dispatch function to write pixels to any format''' 561 562 for format in formats: 563 if is_format_supported(format): 564 generate_format_write(format, src_channel, src_native_type, src_suffix) 565 566 print 'void' 567 print 'lp_tile_unswizzle_%s(enum pipe_format format, const %s *src, void *dst, unsigned dst_stride, unsigned x, unsigned y)' % (src_suffix, src_native_type) 568 569 print '{' 570 print ' void (*func)(const %s * restrict src, uint8_t * restrict dst, unsigned dst_stride, unsigned x0, unsigned y0);' % src_native_type 571 print '#ifdef DEBUG' 572 print ' lp_tile_unswizzle_count += 1;' 573 print '#endif' 574 print ' switch(format) {' 575 for format in formats: 576 if is_format_supported(format): 577 print ' case %s:' % format.name 578 func_name = 'lp_tile_%s_unswizzle_%s' % (format.short_name(), src_suffix) 579 if format.name == 'PIPE_FORMAT_B8G8R8A8_UNORM' or format.name == 'PIPE_FORMAT_B8G8R8X8_UNORM': 580 print '#ifdef PIPE_ARCH_SSE' 581 print ' func = util_cpu_caps.has_sse2 ? %s_sse2 : %s;' % (func_name, func_name) 582 print '#else' 583 print ' func = %s;' % (func_name,) 584 print '#endif' 585 else: 586 print ' func = %s;' % (func_name,) 587 print ' break;' 588 print ' default:' 589 print ' debug_printf("%s: unsupported format %s\\n", __FUNCTION__, util_format_name(format));' 590 print ' return;' 591 print ' }' 592 print ' func(src, (uint8_t *)dst, dst_stride, x, y);' 593 print '}' 594 print 595 596 597def main(): 598 formats = [] 599 for arg in sys.argv[1:]: 600 formats.extend(parse(arg)) 601 602 print '/* This file is autogenerated by lp_tile_soa.py from u_format.csv. Do not edit directly. */' 603 print 604 # This will print the copyright message on the top of this file 605 print CopyRight.strip() 606 print 607 print '#include "pipe/p_compiler.h"' 608 print '#include "util/u_math.h"' 609 print '#include "util/u_format.h"' 610 print '#include "util/u_format_r11g11b10f.h"' 611 print '#include "util/u_format_rgb9e5.h"' 612 print '#include "util/u_half.h"' 613 print '#include "util/u_cpu_detect.h"' 614 print '#include "lp_tile_soa.h"' 615 print 616 print '#ifdef DEBUG' 617 print 'unsigned lp_tile_unswizzle_count = 0;' 618 print 'unsigned lp_tile_swizzle_count = 0;' 619 print '#endif' 620 print 621 print 'const unsigned char' 622 print 'tile_offset[TILE_VECTOR_HEIGHT][TILE_VECTOR_WIDTH] = {' 623 print ' { 0, 1, 4, 5},' 624 print ' { 2, 3, 6, 7},' 625 print ' { 8, 9, 12, 13},' 626 print ' { 10, 11, 14, 15}' 627 print '};' 628 print 629 print '/* Note: these lookup tables could be replaced with some' 630 print ' * bit-twiddling code, but this is a little faster.' 631 print ' */' 632 print 'static unsigned tile_x_offset[TILE_VECTOR_WIDTH * TILE_VECTOR_HEIGHT] = {' 633 print ' 0, 1, 0, 1, 2, 3, 2, 3,' 634 print ' 0, 1, 0, 1, 2, 3, 2, 3' 635 print '};' 636 print 637 print 'static unsigned tile_y_offset[TILE_VECTOR_WIDTH * TILE_VECTOR_HEIGHT] = {' 638 print ' 0, 0, 1, 1, 0, 0, 1, 1,' 639 print ' 2, 2, 3, 3, 2, 2, 3, 3' 640 print '};' 641 print 642 643 generate_sse2() 644 645 channel = Channel(UNSIGNED, True, False, 8) 646 native_type = 'uint8_t' 647 suffix = '4ub' 648 649 generate_swizzle(formats, channel, native_type, suffix) 650 generate_unswizzle(formats, channel, native_type, suffix) 651 652 653if __name__ == '__main__': 654 main() 655