1b8e80941Smrgfrom __future__ import print_function 2848b8605Smrg 3848b8605SmrgCopyRight = ''' 4848b8605Smrg/************************************************************************** 5848b8605Smrg * 6848b8605Smrg * Copyright 2010 VMware, Inc. 7848b8605Smrg * All Rights Reserved. 8848b8605Smrg * 9848b8605Smrg * Permission is hereby granted, free of charge, to any person obtaining a 10848b8605Smrg * copy of this software and associated documentation files (the 11848b8605Smrg * "Software"), to deal in the Software without restriction, including 12848b8605Smrg * without limitation the rights to use, copy, modify, merge, publish, 13848b8605Smrg * distribute, sub license, and/or sell copies of the Software, and to 14848b8605Smrg * permit persons to whom the Software is furnished to do so, subject to 15848b8605Smrg * the following conditions: 16848b8605Smrg * 17848b8605Smrg * The above copyright notice and this permission notice (including the 18848b8605Smrg * next paragraph) shall be included in all copies or substantial portions 19848b8605Smrg * of the Software. 20848b8605Smrg * 21848b8605Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 22848b8605Smrg * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 23848b8605Smrg * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. 24848b8605Smrg * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR 25848b8605Smrg * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, 26848b8605Smrg * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE 27848b8605Smrg * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 28848b8605Smrg * 29848b8605Smrg **************************************************************************/ 30848b8605Smrg 31848b8605Smrg/** 32848b8605Smrg * @file 33848b8605Smrg * SRGB translation. 34848b8605Smrg * 35848b8605Smrg * @author Brian Paul <brianp@vmware.com> 36848b8605Smrg * @author Michal Krol <michal@vmware.com> 37848b8605Smrg * @author Jose Fonseca <jfonseca@vmware.com> 38848b8605Smrg */ 39848b8605Smrg''' 40848b8605Smrg 41848b8605Smrg 42848b8605Smrgimport math 43848b8605Smrgimport struct 44848b8605Smrg 45848b8605Smrg 46848b8605Smrgdef srgb_to_linear(x): 47848b8605Smrg if x <= 0.04045: 48848b8605Smrg return x / 12.92 49848b8605Smrg else: 50848b8605Smrg return math.pow((x + 0.055) / 1.055, 2.4) 51848b8605Smrg 52848b8605Smrg 53848b8605Smrgdef linear_to_srgb(x): 54848b8605Smrg if x >= 0.0031308: 55848b8605Smrg return 1.055 * math.pow(x, 0.41666666) - 0.055 56848b8605Smrg else: 57848b8605Smrg return 12.92 * x 58848b8605Smrg 59848b8605Smrg 60848b8605Smrgdef generate_srgb_tables(): 61b8e80941Smrg print('const float') 62b8e80941Smrg print('util_format_srgb_8unorm_to_linear_float_table[256] = {') 63848b8605Smrg for j in range(0, 256, 4): 64b8e80941Smrg print(' ', end=' ') 65b8e80941Smrg print(' '.join(['%.7e,' % srgb_to_linear(i / 255.0) for i in range(j, j + 4)])) 66b8e80941Smrg print('};') 67b8e80941Smrg print() 68b8e80941Smrg print('const uint8_t') 69b8e80941Smrg print('util_format_srgb_to_linear_8unorm_table[256] = {') 70848b8605Smrg for j in range(0, 256, 16): 71b8e80941Smrg print(' ', end=' ') 72b8e80941Smrg print(' '.join(['%3u,' % int(srgb_to_linear(i / 255.0) * 255.0 + 0.5) for i in range(j, j + 16)])) 73b8e80941Smrg print('};') 74b8e80941Smrg print() 75b8e80941Smrg print('const uint8_t') 76b8e80941Smrg print('util_format_linear_to_srgb_8unorm_table[256] = {') 77848b8605Smrg for j in range(0, 256, 16): 78b8e80941Smrg print(' ', end=' ') 79b8e80941Smrg print(' '.join(['%3u,' % int(linear_to_srgb(i / 255.0) * 255.0 + 0.5) for i in range(j, j + 16)])) 80b8e80941Smrg print('};') 81b8e80941Smrg print() 82848b8605Smrg 83848b8605Smrg# calculate the table interpolation values used in float linear to unorm8 srgb 84848b8605Smrg numexp = 13 85848b8605Smrg mantissa_msb = 3 86848b8605Smrg# stepshift is just used to only use every x-th float to make things faster, 87848b8605Smrg# 5 is largest value which still gives exact same table as 0 88848b8605Smrg stepshift = 5 89848b8605Smrg nbuckets = numexp << mantissa_msb 90848b8605Smrg bucketsize = (1 << (23 - mantissa_msb)) >> stepshift 91848b8605Smrg mantshift = 12 92848b8605Smrg valtable = [] 93848b8605Smrg sum_aa = float(bucketsize) 94848b8605Smrg sum_ab = 0.0 95848b8605Smrg sum_bb = 0.0 96848b8605Smrg for i in range(0, bucketsize): 97848b8605Smrg j = (i << stepshift) >> mantshift 98848b8605Smrg sum_ab += j 99848b8605Smrg sum_bb += j*j 100848b8605Smrg inv_det = 1.0 / (sum_aa * sum_bb - sum_ab * sum_ab) 101848b8605Smrg 102848b8605Smrg for bucket in range(0, nbuckets): 103848b8605Smrg start = ((127 - numexp) << 23) + bucket*(bucketsize << stepshift) 104848b8605Smrg sum_a = 0.0 105848b8605Smrg sum_b = 0.0 106848b8605Smrg 107848b8605Smrg for i in range(0, bucketsize): 108848b8605Smrg j = (i << stepshift) >> mantshift 109848b8605Smrg fint = start + (i << stepshift) 110848b8605Smrg ffloat = struct.unpack('f', struct.pack('I', fint))[0] 111848b8605Smrg val = linear_to_srgb(ffloat) * 255.0 + 0.5 112848b8605Smrg sum_a += val 113848b8605Smrg sum_b += j*val 114848b8605Smrg 115848b8605Smrg solved_a = inv_det * (sum_bb*sum_a - sum_ab*sum_b) 116848b8605Smrg solved_b = inv_det * (sum_aa*sum_b - sum_ab*sum_a) 117848b8605Smrg 118848b8605Smrg scaled_a = solved_a * 65536.0 / 512.0 119848b8605Smrg scaled_b = solved_b * 65536.0 120848b8605Smrg 121848b8605Smrg int_a = int(scaled_a + 0.5) 122848b8605Smrg int_b = int(scaled_b + 0.5) 123848b8605Smrg 124848b8605Smrg valtable.append((int_a << 16) + int_b) 125848b8605Smrg 126b8e80941Smrg print('const unsigned') 127b8e80941Smrg print('util_format_linear_to_srgb_helper_table[104] = {') 128848b8605Smrg 129848b8605Smrg for j in range(0, nbuckets, 4): 130b8e80941Smrg print(' ', end=' ') 131b8e80941Smrg print(' '.join(['0x%08x,' % valtable[i] for i in range(j, j + 4)])) 132b8e80941Smrg print('};') 133b8e80941Smrg print() 134848b8605Smrg 135848b8605Smrgdef main(): 136b8e80941Smrg print('/* This file is autogenerated by u_format_srgb.py. Do not edit directly. */') 137b8e80941Smrg print() 138848b8605Smrg # This will print the copyright message on the top of this file 139b8e80941Smrg print(CopyRight.strip()) 140b8e80941Smrg print() 141b8e80941Smrg print('#include "format_srgb.h"') 142b8e80941Smrg print() 143848b8605Smrg generate_srgb_tables() 144848b8605Smrg 145848b8605Smrg 146848b8605Smrgif __name__ == '__main__': 147848b8605Smrg main() 148