2013-01-04 17:04:55 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
"""
|
2013-04-11 15:33:10 +02:00
|
|
|
Python 3 script which converts simple RetroArch Cg shaders to modern GLSL (ES) format.
|
2013-01-04 17:04:55 +01:00
|
|
|
Author: Hans-Kristian Arntzen (Themaister)
|
|
|
|
License: Public domain
|
|
|
|
"""
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import errno
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
batch_mode = False
|
|
|
|
|
|
|
|
def log(*arg):
|
|
|
|
if not batch_mode:
|
|
|
|
print(*arg)
|
|
|
|
|
|
|
|
def remove_comments(source_lines):
|
|
|
|
ret = []
|
|
|
|
killed_comments = [line.split('//')[0] for line in source_lines]
|
|
|
|
for i in filter(lambda line: len(line) > 0, killed_comments):
|
|
|
|
ret.append(i)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def keep_line_if(func, lines):
|
|
|
|
ret = []
|
|
|
|
for i in filter(func, lines):
|
|
|
|
ret.append(i)
|
|
|
|
return ret
|
|
|
|
|
2013-04-13 02:41:23 +02:00
|
|
|
def replace_global_in(source):
|
|
|
|
split_source = source.split('\n')
|
|
|
|
replace_table = [
|
|
|
|
('IN.video_size', 'InputSize'),
|
|
|
|
('IN.texture_size', 'TextureSize'),
|
|
|
|
('IN.output_size', 'OutputSize'),
|
|
|
|
('IN.frame_count', 'FrameCount'),
|
|
|
|
('IN.frame_direction', 'FrameDirection')
|
|
|
|
]
|
|
|
|
|
|
|
|
for line in split_source:
|
|
|
|
if '//var' in line:
|
|
|
|
for index, replace in enumerate(replace_table):
|
|
|
|
orig = line.split(' ')[2]
|
|
|
|
if replace[0] == orig:
|
|
|
|
replace_table[index] = (line.split(':')[2].split(' ')[1], replace_table[index][1])
|
|
|
|
|
|
|
|
log('Replace globals:', replace_table)
|
|
|
|
|
|
|
|
for replace in replace_table:
|
|
|
|
if replace[0]:
|
|
|
|
source = source.replace(replace[0], replace[1])
|
|
|
|
|
|
|
|
return source
|
|
|
|
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
def replace_global_vertex(source):
|
2013-04-13 02:41:23 +02:00
|
|
|
|
|
|
|
source = replace_global_in(source)
|
2013-01-04 17:04:55 +01:00
|
|
|
replace_table = [
|
2013-04-05 14:29:12 +02:00
|
|
|
('POSITION', 'VertexCoord'),
|
2013-04-13 02:07:37 +02:00
|
|
|
('TEXCOORD1', 'LUTTexCoord'),
|
2013-04-05 14:29:12 +02:00
|
|
|
('TEXCOORD0', 'TexCoord'),
|
|
|
|
('TEXCOORD', 'TexCoord'),
|
|
|
|
('uniform vec4 _modelViewProj1[4]', 'uniform mat4 MVPMatrix'),
|
|
|
|
('_modelViewProj1', 'MVPMatrix'),
|
|
|
|
('MVPMatrix[0]', 'MVPMatrix_[0]'),
|
|
|
|
('MVPMatrix[1]', 'MVPMatrix_[1]'),
|
|
|
|
('MVPMatrix[2]', 'MVPMatrix_[2]'),
|
|
|
|
('MVPMatrix[3]', 'MVPMatrix_[3]'),
|
2013-04-13 02:41:23 +02:00
|
|
|
|
2013-04-05 14:29:12 +02:00
|
|
|
('FrameCount', 'float(FrameCount)'),
|
2013-01-04 17:04:55 +01:00
|
|
|
('input', 'input_dummy'), # 'input' is reserved in GLSL.
|
|
|
|
('output', 'output_dummy'), # 'output' is reserved in GLSL.
|
|
|
|
]
|
|
|
|
|
|
|
|
for replacement in replace_table:
|
|
|
|
source = source.replace(replacement[0], replacement[1])
|
|
|
|
|
|
|
|
return source
|
|
|
|
|
|
|
|
def translate_varyings(varyings, source):
|
|
|
|
dictionary = {}
|
|
|
|
for varying in varyings:
|
|
|
|
for line in source:
|
|
|
|
if (varying in line) and ('//var' in line):
|
|
|
|
log('Found line for', varying + ':', line)
|
2013-04-13 16:32:52 +02:00
|
|
|
dictionary[varying] = 'VAR' + line.split(':')[0].split('.')[-1].strip()
|
2013-01-04 17:04:55 +01:00
|
|
|
break
|
|
|
|
|
|
|
|
return dictionary
|
|
|
|
|
2013-04-13 01:43:40 +02:00
|
|
|
def no_uniform(elem):
|
|
|
|
banned = [
|
|
|
|
'_video_size',
|
|
|
|
'_texture_size',
|
|
|
|
'_output_size',
|
2013-04-13 13:47:13 +02:00
|
|
|
'_output_dummy_size',
|
2013-04-13 01:43:40 +02:00
|
|
|
'_frame_count',
|
|
|
|
'_frame_direction',
|
|
|
|
'sampler2D'
|
|
|
|
]
|
|
|
|
|
|
|
|
for ban in banned:
|
|
|
|
if ban in elem:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
def destructify_varyings(source):
|
2013-04-13 01:43:40 +02:00
|
|
|
# We have to change varying structs that Cg support to single varyings for GL.
|
|
|
|
# Varying structs aren't supported until later versions
|
2013-01-04 17:04:55 +01:00
|
|
|
# of GLSL.
|
2013-04-12 00:20:58 +02:00
|
|
|
|
|
|
|
# Global structs are sometimes used to store temporary data.
|
|
|
|
# Don't try to remove this as it breaks compile.
|
|
|
|
vout_lines = []
|
|
|
|
for line in source:
|
2013-04-13 01:43:40 +02:00
|
|
|
if ('//var' in line) and (('$vout.' in line) or ('$vin.' in line)):
|
2013-04-12 00:20:58 +02:00
|
|
|
vout_lines.append(line)
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
struct_types = []
|
2013-04-13 16:32:52 +02:00
|
|
|
for line in source[1:]:
|
2013-01-04 17:04:55 +01:00
|
|
|
if 'struct' in line:
|
2013-04-13 01:43:40 +02:00
|
|
|
struct_type = line.split(' ')[1]
|
|
|
|
if struct_type not in struct_types:
|
|
|
|
struct_types.append(struct_type)
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
log('Struct types:', struct_types)
|
|
|
|
|
|
|
|
last_struct_decl_line = 0
|
|
|
|
varyings = []
|
|
|
|
varyings_name = []
|
|
|
|
# Find all varyings in structs and make them "global" varyings.
|
|
|
|
for struct in struct_types:
|
|
|
|
for i, line in enumerate(source):
|
2013-04-13 01:43:40 +02:00
|
|
|
if ('struct ' + struct) in line:
|
2013-01-04 17:04:55 +01:00
|
|
|
j = i + 1
|
|
|
|
while (j < len(source)) and ('};' not in source[j]):
|
|
|
|
j += 1
|
2013-04-13 01:43:40 +02:00
|
|
|
|
|
|
|
lines = ['varying ' + string for string in source[i + 1 : j]]
|
|
|
|
varyings.extend(lines)
|
2013-01-04 17:04:55 +01:00
|
|
|
names = [string.strip().split(' ')[1].split(';')[0].strip() for string in source[i + 1 : j]]
|
|
|
|
varyings_name.extend(names)
|
|
|
|
log('Found elements in struct', struct + ':', names)
|
|
|
|
last_struct_decl_line = j
|
|
|
|
|
2013-04-13 01:58:25 +02:00
|
|
|
# Must have explicit uniform sampler2D in struct.
|
|
|
|
for index in range(i, j + 1):
|
|
|
|
if 'sampler2D' in source[index]:
|
2013-04-16 15:31:28 +02:00
|
|
|
source[index] = 'float _placeholder{};'.format(index)
|
2013-04-13 01:58:25 +02:00
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
varyings_tmp = varyings
|
|
|
|
varyings = []
|
|
|
|
variables = []
|
|
|
|
|
|
|
|
# Don't include useless varyings like IN.video_size, IN.texture_size, etc as they are not actual varyings ...
|
2013-04-13 01:43:40 +02:00
|
|
|
for i in filter(no_uniform, varyings_tmp):
|
2013-01-04 17:04:55 +01:00
|
|
|
varyings.append(i)
|
|
|
|
|
|
|
|
# Find any global variable struct that is supposed to be the output varying, and redirect all references to it to
|
|
|
|
# the actual varyings we just declared ...
|
|
|
|
# Globals only come before main() ...
|
|
|
|
# Make sure to only look after all struct declarations as there might be overlap.
|
|
|
|
for line in source[last_struct_decl_line:]:
|
|
|
|
if 'void main()' in line:
|
|
|
|
break
|
|
|
|
|
|
|
|
for struct in struct_types:
|
|
|
|
if struct in line:
|
|
|
|
variable = line.split(' ')[1].split(';')[0]
|
2013-04-12 00:20:58 +02:00
|
|
|
|
|
|
|
# Only redirect if the struct is actually used as vertex output.
|
|
|
|
for vout_line in vout_lines:
|
|
|
|
if variable in vout_line:
|
|
|
|
log('Found struct variable for', struct + ':', variable)
|
|
|
|
variables.append(variable)
|
|
|
|
break
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
varyings_dict = translate_varyings(varyings_name, source)
|
|
|
|
log('Varyings dict:', varyings_dict)
|
|
|
|
|
|
|
|
# Append all varyings. Keep the structs as they might be used as regular values.
|
|
|
|
for varying in varyings:
|
|
|
|
source.insert(1, varying)
|
|
|
|
|
2013-04-13 16:32:52 +02:00
|
|
|
log('Variables:', variables)
|
|
|
|
log('Varying names:', varyings_name)
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
# Replace struct access with global access, e.g. (_co1._c00 => _c00)
|
|
|
|
# Also replace mangled Cg name with 'real' name.
|
|
|
|
for index, _ in enumerate(source):
|
|
|
|
for variable in variables:
|
2013-04-13 16:32:52 +02:00
|
|
|
for varying_name in varyings_dict:
|
|
|
|
trans_from = variable + '.' + varying_name
|
|
|
|
trans_to = varyings_dict[varying_name]
|
|
|
|
source[index] = source[index].replace(trans_from, trans_to);
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
for index, _ in enumerate(source):
|
|
|
|
for varying_name in varyings_name:
|
|
|
|
if varying_name in varyings_dict:
|
|
|
|
source[index] = source[index].replace(varying_name, varyings_dict[varying_name])
|
|
|
|
|
2013-04-13 01:58:25 +02:00
|
|
|
# Replace union <struct>. Sometimes we get collision in vertex and fragment.
|
|
|
|
for index, line in enumerate(source):
|
|
|
|
for struct_type in struct_types:
|
|
|
|
line = line.replace('uniform ' + struct_type, struct_type)
|
|
|
|
source[index] = line
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
return source
|
|
|
|
|
2013-04-13 01:43:40 +02:00
|
|
|
def translate_varying(cg):
|
|
|
|
#log('Translate:', cg)
|
|
|
|
translations = {
|
|
|
|
'ORIG.tex_coord' : 'OrigTexCoord',
|
|
|
|
'PREV.tex_coord' : 'PrevTexCoord',
|
|
|
|
'PREV1.tex_coord' : 'Prev1TexCoord',
|
|
|
|
'PREV2.tex_coord' : 'Prev2TexCoord',
|
|
|
|
'PREV3.tex_coord' : 'Prev3TexCoord',
|
|
|
|
'PREV4.tex_coord' : 'Prev4TexCoord',
|
|
|
|
'PREV5.tex_coord' : 'Prev5TexCoord',
|
|
|
|
'PREV6.tex_coord' : 'Prev6TexCoord',
|
|
|
|
'PASS1.tex_coord' : 'Pass1TexCoord',
|
|
|
|
'PASS2.tex_coord' : 'Pass2TexCoord',
|
|
|
|
'PASS3.tex_coord' : 'Pass3TexCoord',
|
|
|
|
'PASS4.tex_coord' : 'Pass4TexCoord',
|
|
|
|
'PASS5.tex_coord' : 'Pass5TexCoord',
|
|
|
|
'PASS6.tex_coord' : 'Pass6TexCoord',
|
|
|
|
'PASS7.tex_coord' : 'Pass7TexCoord',
|
|
|
|
'PASS8.tex_coord' : 'Pass8TexCoord',
|
|
|
|
}
|
|
|
|
|
|
|
|
if cg in translations:
|
|
|
|
return translations[cg]
|
|
|
|
else:
|
|
|
|
return cg
|
|
|
|
|
2013-04-13 16:32:52 +02:00
|
|
|
def translate_texture_size(cg):
|
|
|
|
log('Translate:', cg)
|
|
|
|
translations = {
|
|
|
|
'ORIG.texture_size' : 'OrigTextureSize',
|
|
|
|
'PREV.texture_size' : 'PrevTextureSize',
|
|
|
|
'PREV1.texture_size' : 'Prev1TextureSize',
|
|
|
|
'PREV2.texture_size' : 'Prev2TextureSize',
|
|
|
|
'PREV3.texture_size' : 'Prev3TextureSize',
|
|
|
|
'PREV4.texture_size' : 'Prev4TextureSize',
|
|
|
|
'PREV5.texture_size' : 'Prev5TextureSize',
|
|
|
|
'PREV6.texture_size' : 'Prev6TextureSize',
|
|
|
|
'PASS1.texture_size' : 'Pass1TextureSize',
|
|
|
|
'PASS2.texture_size' : 'Pass2TextureSize',
|
|
|
|
'PASS3.texture_size' : 'Pass3TextureSize',
|
|
|
|
'PASS4.texture_size' : 'Pass4TextureSize',
|
|
|
|
'PASS5.texture_size' : 'Pass5TextureSize',
|
|
|
|
'PASS6.texture_size' : 'Pass6TextureSize',
|
|
|
|
'PASS7.texture_size' : 'Pass7TextureSize',
|
|
|
|
'PASS8.texture_size' : 'Pass8TextureSize',
|
|
|
|
}
|
|
|
|
|
|
|
|
if cg in translations:
|
|
|
|
return translations[cg]
|
|
|
|
else:
|
|
|
|
return cg
|
|
|
|
|
|
|
|
|
2013-04-13 01:43:40 +02:00
|
|
|
|
|
|
|
def replace_varyings(source):
|
|
|
|
ret = []
|
|
|
|
translations = []
|
|
|
|
attribs = []
|
2013-04-13 16:32:52 +02:00
|
|
|
uniforms = []
|
2013-04-13 01:43:40 +02:00
|
|
|
for index, line in enumerate(source):
|
|
|
|
if ('//var' in line) and ('$vin.' in line):
|
|
|
|
orig = line.split(' ')[2]
|
|
|
|
translated = translate_varying(orig)
|
|
|
|
if translated != orig and translated not in attribs:
|
|
|
|
cg_attrib = line.split(':')[2].split(' ')[1]
|
|
|
|
translations.append((cg_attrib, translated))
|
|
|
|
attribs.append(translated)
|
2013-04-13 16:32:52 +02:00
|
|
|
elif '//var' in line:
|
|
|
|
orig = line.split(' ')[2]
|
|
|
|
translated = translate_texture_size(orig)
|
|
|
|
if translated != orig and translated not in uniforms:
|
|
|
|
cg_uniform = line.split(':')[2].split(' ')[1]
|
|
|
|
translations.append((cg_uniform, translated))
|
|
|
|
uniforms.append(translated)
|
2013-04-13 01:43:40 +02:00
|
|
|
|
|
|
|
for index, line in enumerate(source):
|
|
|
|
if 'void main()' in line:
|
|
|
|
for attrib in attribs:
|
|
|
|
source.insert(index, 'attribute vec2 ' + attrib + ';')
|
2013-04-13 16:32:52 +02:00
|
|
|
for uniform in uniforms:
|
|
|
|
source.insert(index, '#endif')
|
|
|
|
source.insert(index, 'uniform vec2 ' + uniform + ';')
|
|
|
|
source.insert(index, '#else')
|
|
|
|
source.insert(index, 'uniform mediump vec2 ' + uniform + ';')
|
|
|
|
source.insert(index, '#ifdef GL_ES')
|
2013-04-13 01:43:40 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
for line in source:
|
|
|
|
for trans in translations:
|
|
|
|
line = line.replace(trans[0], trans[1])
|
|
|
|
ret.append(line)
|
|
|
|
|
|
|
|
return ret
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
def hack_source_vertex(source):
|
|
|
|
transpose_index = 2
|
2013-04-16 15:31:28 +02:00
|
|
|
ref_index = 0
|
2013-01-04 17:04:55 +01:00
|
|
|
for index, line in enumerate(source):
|
|
|
|
if 'void main()' in line:
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index + 2, ' mat4 MVPMatrix_ = transpose_(MVPMatrix);') # transpose() is GLSL 1.20+, doesn't exist in GLSL ES 1.0
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#endif')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform vec2 InputSize;')
|
|
|
|
source.insert(index, 'uniform vec2 TextureSize;')
|
|
|
|
source.insert(index, 'uniform vec2 OutputSize;')
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#else')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform mediump vec2 InputSize;')
|
|
|
|
source.insert(index, 'uniform mediump vec2 TextureSize;')
|
|
|
|
source.insert(index, 'uniform mediump vec2 OutputSize;')
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#ifdef GL_ES')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform int FrameCount;')
|
2013-04-12 00:20:58 +02:00
|
|
|
source.insert(index, 'uniform int FrameDirection;')
|
2013-01-26 14:24:42 +01:00
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
source.insert(index, """
|
|
|
|
mat4 transpose_(mat4 matrix)
|
|
|
|
{
|
|
|
|
mat4 ret;
|
2013-05-31 14:04:20 +02:00
|
|
|
for (int i = 0; i != 4; i++)
|
|
|
|
for (int j = 0; j != 4; j++)
|
2013-01-04 17:04:55 +01:00
|
|
|
ret[i][j] = matrix[j][i];
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
""")
|
2013-04-16 15:31:28 +02:00
|
|
|
ref_index = index
|
2013-01-04 17:04:55 +01:00
|
|
|
break
|
|
|
|
|
2013-04-16 15:31:28 +02:00
|
|
|
# Fix samplers in vertex shader (supported by GLSL).
|
|
|
|
translations = []
|
|
|
|
added_samplers = []
|
|
|
|
translated_samplers = []
|
|
|
|
struct_texunit0 = False # If True, we have to append uniform sampler2D Texture manually ...
|
|
|
|
for line in source:
|
|
|
|
if ('TEXUNIT0' in line) and ('semantic' not in line):
|
|
|
|
main_sampler = (line.split(':')[2].split(' ')[1], 'Texture')
|
|
|
|
if len(main_sampler[0]) > 0:
|
|
|
|
translations.append(main_sampler)
|
|
|
|
log('Vertex: Sampler:', main_sampler[0], '->', main_sampler[1])
|
|
|
|
struct_texunit0 = '.' in main_sampler[0]
|
|
|
|
elif '//var sampler2D' in line:
|
|
|
|
cg_texture = line.split(' ')[2]
|
|
|
|
translated = translate_texture(cg_texture)
|
2013-04-16 15:44:41 +02:00
|
|
|
orig_name = translated
|
2013-04-16 15:31:28 +02:00
|
|
|
new_name = line.split(':')[2].split(' ')[1]
|
2013-04-16 15:44:41 +02:00
|
|
|
log('Vertex: Sampler:', new_name, '->', orig_name)
|
|
|
|
if len(new_name) > 0:
|
|
|
|
if translated != cg_texture and translated not in translated_samplers:
|
|
|
|
translated_samplers.append(translated)
|
|
|
|
added_samplers.append('uniform sampler2D ' + translated + ';')
|
2013-04-16 15:31:28 +02:00
|
|
|
translations.append((new_name, orig_name))
|
|
|
|
|
|
|
|
for sampler in added_samplers:
|
|
|
|
source.insert(ref_index, sampler)
|
|
|
|
if struct_texunit0:
|
|
|
|
source.insert(ref_index, 'uniform sampler2D Texture;')
|
|
|
|
for index, line in enumerate(source):
|
|
|
|
for translation in translations:
|
|
|
|
source[index] = source[index].replace(translation[0], translation[1])
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
source = destructify_varyings(source)
|
2013-04-13 01:43:40 +02:00
|
|
|
source = replace_varyings(source)
|
2013-01-04 17:04:55 +01:00
|
|
|
return source
|
|
|
|
|
|
|
|
def replace_global_fragment(source):
|
2013-04-13 02:41:23 +02:00
|
|
|
source = replace_global_in(source)
|
2013-01-04 17:04:55 +01:00
|
|
|
replace_table = [
|
2013-04-05 14:29:12 +02:00
|
|
|
('FrameCount', 'float(FrameCount)'),
|
2013-01-04 17:04:55 +01:00
|
|
|
('input', 'input_dummy'),
|
|
|
|
('output', 'output_dummy'), # 'output' is reserved in GLSL.
|
|
|
|
]
|
|
|
|
|
|
|
|
for replacement in replace_table:
|
|
|
|
source = source.replace(replacement[0], replacement[1])
|
|
|
|
|
|
|
|
return source
|
|
|
|
|
2013-04-13 16:32:52 +02:00
|
|
|
|
2013-04-13 01:43:40 +02:00
|
|
|
def translate_texture(cg):
|
|
|
|
log('Translate:', cg)
|
|
|
|
translations = {
|
|
|
|
'ORIG.texture' : 'OrigTexture',
|
|
|
|
'PREV.texture' : 'PrevTexture',
|
|
|
|
'PREV1.texture' : 'Prev1Texture',
|
|
|
|
'PREV2.texture' : 'Prev2Texture',
|
|
|
|
'PREV3.texture' : 'Prev3Texture',
|
|
|
|
'PREV4.texture' : 'Prev4Texture',
|
|
|
|
'PREV5.texture' : 'Prev5Texture',
|
|
|
|
'PREV6.texture' : 'Prev6Texture',
|
|
|
|
'PASS1.texture' : 'Pass1Texture',
|
|
|
|
'PASS2.texture' : 'Pass2Texture',
|
|
|
|
'PASS3.texture' : 'Pass3Texture',
|
|
|
|
'PASS4.texture' : 'Pass4Texture',
|
|
|
|
'PASS5.texture' : 'Pass5Texture',
|
|
|
|
'PASS6.texture' : 'Pass6Texture',
|
|
|
|
'PASS7.texture' : 'Pass7Texture',
|
|
|
|
'PASS8.texture' : 'Pass8Texture',
|
|
|
|
}
|
|
|
|
|
|
|
|
if cg in translations:
|
|
|
|
return translations[cg]
|
|
|
|
else:
|
|
|
|
return cg
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
def hack_source_fragment(source):
|
2013-04-13 01:43:40 +02:00
|
|
|
ref_index = 0
|
2013-01-04 17:04:55 +01:00
|
|
|
for index, line in enumerate(source):
|
|
|
|
if 'void main()' in line:
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#endif')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform vec2 InputSize;')
|
|
|
|
source.insert(index, 'uniform vec2 TextureSize;')
|
|
|
|
source.insert(index, 'uniform vec2 OutputSize;')
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#else')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform mediump vec2 InputSize;')
|
|
|
|
source.insert(index, 'uniform mediump vec2 TextureSize;')
|
|
|
|
source.insert(index, 'uniform mediump vec2 OutputSize;')
|
2013-01-26 14:24:42 +01:00
|
|
|
source.insert(index, '#ifdef GL_ES')
|
2013-04-05 14:29:12 +02:00
|
|
|
source.insert(index, 'uniform int FrameCount;')
|
2013-04-12 00:20:58 +02:00
|
|
|
source.insert(index, 'uniform int FrameDirection;')
|
2013-04-13 01:43:40 +02:00
|
|
|
ref_index = index
|
2013-01-04 17:04:55 +01:00
|
|
|
break
|
|
|
|
|
2013-04-13 16:32:52 +02:00
|
|
|
translations = []
|
2013-04-13 01:43:40 +02:00
|
|
|
added_samplers = []
|
|
|
|
translated_samplers = []
|
2013-04-13 16:32:52 +02:00
|
|
|
uniforms = []
|
2013-04-16 15:31:28 +02:00
|
|
|
struct_texunit0 = False # If True, we have to append uniform sampler2D Texture manually ...
|
2013-01-04 17:04:55 +01:00
|
|
|
for line in source:
|
|
|
|
if ('TEXUNIT0' in line) and ('semantic' not in line):
|
2013-04-12 00:29:27 +02:00
|
|
|
main_sampler = (line.split(':')[2].split(' ')[1], 'Texture')
|
2013-04-16 15:31:28 +02:00
|
|
|
if len(main_sampler[0]) > 0:
|
|
|
|
translations.append(main_sampler)
|
|
|
|
log('Fragment: Sampler:', main_sampler[0], '->', main_sampler[1])
|
|
|
|
struct_texunit0 = '.' in main_sampler[0]
|
2013-04-12 00:29:27 +02:00
|
|
|
elif '//var sampler2D' in line:
|
2013-04-13 01:43:40 +02:00
|
|
|
cg_texture = line.split(' ')[2]
|
|
|
|
translated = translate_texture(cg_texture)
|
2013-04-16 15:44:41 +02:00
|
|
|
orig_name = translated
|
2013-04-16 15:31:28 +02:00
|
|
|
new_name = line.split(':')[2].split(' ')[1]
|
2013-04-16 15:44:41 +02:00
|
|
|
log('Fragment: Sampler:', new_name, '->', orig_name)
|
|
|
|
if len(new_name) > 0:
|
|
|
|
if translated != cg_texture and translated not in translated_samplers:
|
|
|
|
translated_samplers.append(translated)
|
|
|
|
added_samplers.append('uniform sampler2D ' + translated + ';')
|
2013-04-16 15:31:28 +02:00
|
|
|
translations.append((new_name, orig_name))
|
2013-04-13 16:32:52 +02:00
|
|
|
elif '//var' in line:
|
|
|
|
orig = line.split(' ')[2]
|
|
|
|
translated = translate_texture_size(orig)
|
|
|
|
if translated != orig and translated not in uniforms:
|
|
|
|
cg_uniform = line.split(':')[2].split(' ')[1]
|
|
|
|
translations.append((cg_uniform, translated))
|
|
|
|
uniforms.append(translated)
|
|
|
|
|
2013-04-12 00:29:27 +02:00
|
|
|
|
2013-04-13 01:43:40 +02:00
|
|
|
for sampler in added_samplers:
|
|
|
|
source.insert(ref_index, sampler)
|
2013-04-13 16:32:52 +02:00
|
|
|
for uniform in uniforms:
|
|
|
|
source.insert(ref_index, '#endif')
|
|
|
|
source.insert(ref_index, 'uniform vec2 ' + uniform + ';')
|
|
|
|
source.insert(ref_index, '#else')
|
|
|
|
source.insert(ref_index, 'uniform mediump vec2 ' + uniform + ';')
|
|
|
|
source.insert(ref_index, '#ifdef GL_ES')
|
2013-04-16 15:31:28 +02:00
|
|
|
if struct_texunit0:
|
|
|
|
source.insert(ref_index, 'uniform sampler2D Texture;')
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
ret = []
|
|
|
|
for line in source:
|
2013-04-13 16:32:52 +02:00
|
|
|
for translation in translations:
|
|
|
|
line = line.replace(translation[0], translation[1])
|
2013-04-13 01:43:40 +02:00
|
|
|
ret.append(line)
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
ret = destructify_varyings(ret)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def validate_shader(source, target):
|
|
|
|
command = ['cgc', '-noentry', '-ogles']
|
|
|
|
p = subprocess.Popen(command, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE)
|
|
|
|
stdout_ret, stderr_ret = p.communicate(source.encode())
|
|
|
|
|
|
|
|
log('Shader:')
|
|
|
|
log('===')
|
|
|
|
log(source)
|
|
|
|
log('===')
|
|
|
|
log('CGC:', stderr_ret.decode())
|
|
|
|
|
|
|
|
return p.returncode == 0
|
|
|
|
|
|
|
|
def convert(source, dest):
|
|
|
|
vert_cmd = ['cgc', '-profile', 'glesv', '-entry', 'main_vertex', source]
|
|
|
|
p = subprocess.Popen(vert_cmd, stderr = subprocess.PIPE, stdout = subprocess.PIPE)
|
|
|
|
vertex_source, stderr_ret = p.communicate()
|
|
|
|
log(stderr_ret.decode())
|
|
|
|
vertex_source = vertex_source.decode()
|
|
|
|
|
|
|
|
if p.returncode != 0:
|
|
|
|
log('Vertex compilation failed ...')
|
|
|
|
return 1
|
|
|
|
|
|
|
|
frag_cmd = ['cgc', '-profile', 'glesf', '-entry', 'main_fragment', source]
|
|
|
|
p = subprocess.Popen(frag_cmd, stderr = subprocess.PIPE, stdout = subprocess.PIPE)
|
|
|
|
fragment_source, stderr_ret = p.communicate()
|
|
|
|
log(stderr_ret.decode())
|
|
|
|
fragment_source = fragment_source.decode()
|
|
|
|
|
|
|
|
if p.returncode != 0:
|
|
|
|
log('Vertex compilation failed ...')
|
|
|
|
return 1
|
|
|
|
|
|
|
|
vertex_source = replace_global_vertex(vertex_source)
|
|
|
|
fragment_source = replace_global_fragment(fragment_source)
|
|
|
|
|
|
|
|
vertex_source = vertex_source.split('\n')
|
|
|
|
fragment_source = fragment_source.split('\n')
|
|
|
|
|
|
|
|
# Cg think we're using row-major matrices, but we're using column major.
|
|
|
|
# Also, Cg tends to compile matrix multiplications as dot products in GLSL.
|
|
|
|
# Hack in a fix for this.
|
2013-04-13 01:43:40 +02:00
|
|
|
log('Hacking vertex')
|
2013-01-04 17:04:55 +01:00
|
|
|
vertex_source = hack_source_vertex(vertex_source)
|
2013-04-13 01:43:40 +02:00
|
|
|
log('Hacking fragment')
|
2013-01-04 17:04:55 +01:00
|
|
|
fragment_source = hack_source_fragment(fragment_source)
|
|
|
|
|
|
|
|
# We compile to GLES, but we really just want modern GL ...
|
|
|
|
vertex_source = keep_line_if(lambda line: 'precision' not in line, vertex_source)
|
|
|
|
fragment_source = keep_line_if(lambda line: 'precision' not in line, fragment_source)
|
|
|
|
|
|
|
|
# Kill all comments. Cg adds lots of useless comments.
|
|
|
|
# Remove first line. It contains the name of the cg program.
|
|
|
|
vertex_source = remove_comments(vertex_source[1:])
|
|
|
|
fragment_source = remove_comments(fragment_source[1:])
|
|
|
|
|
|
|
|
out_vertex = '\n'.join(vertex_source)
|
|
|
|
out_fragment = '\n'.join(['#ifdef GL_ES', 'precision mediump float;', '#endif'] + fragment_source)
|
|
|
|
|
|
|
|
if not validate_shader(out_vertex, 'glesv'):
|
|
|
|
log('Vertex shader does not compile ...')
|
|
|
|
return 1
|
|
|
|
|
|
|
|
if not validate_shader(out_fragment, 'glesf'):
|
|
|
|
log('Fragment shader does not compile ...')
|
|
|
|
return 1
|
|
|
|
|
|
|
|
with open(dest, 'w') as f:
|
2013-04-11 15:33:10 +02:00
|
|
|
f.write('// GLSL shader autogenerated by cg2glsl.py.\n')
|
|
|
|
f.write('#if defined(VERTEX)\n')
|
2013-01-04 17:04:55 +01:00
|
|
|
f.write(out_vertex)
|
|
|
|
f.write('\n')
|
2013-04-11 15:33:10 +02:00
|
|
|
f.write('#elif defined(FRAGMENT)\n')
|
2013-01-04 17:04:55 +01:00
|
|
|
f.write(out_fragment)
|
|
|
|
f.write('\n')
|
2013-04-11 15:33:10 +02:00
|
|
|
f.write('#endif\n')
|
2013-01-04 17:04:55 +01:00
|
|
|
return 0
|
|
|
|
|
2013-05-10 00:05:37 +02:00
|
|
|
def convert_cgp(source, dest):
|
|
|
|
string = ''
|
|
|
|
with open(source, 'r') as f:
|
|
|
|
string = f.read().replace('.cg', '.glsl')
|
|
|
|
|
|
|
|
open(dest, 'w').write(string)
|
|
|
|
|
|
|
|
def path_ext(path):
|
|
|
|
_, ext = os.path.splitext(path)
|
|
|
|
return ext
|
|
|
|
|
2013-05-20 23:46:40 +02:00
|
|
|
def convert_path(source, source_dir, dest_dir, conv):
|
|
|
|
index = 0 if source_dir[-1] == '/' else 1
|
|
|
|
return os.path.join(dest_dir, source.replace(source_dir, '')[index:]).replace(conv[0], conv[1])
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
def main():
|
|
|
|
if len(sys.argv) != 3:
|
2013-05-10 00:05:37 +02:00
|
|
|
print('Usage: {} prog.cg(p) prog.glsl(p)'.format(sys.argv[0]))
|
2013-01-04 17:11:18 +01:00
|
|
|
print('Batch mode usage: {} cg-dir out-xml-shader-dir'.format(sys.argv[0]))
|
2013-01-04 17:04:55 +01:00
|
|
|
return 1
|
|
|
|
|
|
|
|
if os.path.isdir(sys.argv[1]):
|
|
|
|
global batch_mode
|
|
|
|
batch_mode = True
|
|
|
|
try:
|
|
|
|
os.makedirs(sys.argv[2])
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno != errno.EEXIST:
|
|
|
|
raise
|
|
|
|
|
|
|
|
failed_cnt = 0
|
|
|
|
success_cnt = 0
|
|
|
|
failed_files = []
|
|
|
|
for dirname, _, filenames in os.walk(sys.argv[1]):
|
2013-05-10 00:05:37 +02:00
|
|
|
for source in filter(lambda path: path_ext(path) == '.cg', [os.path.join(dirname, filename) for filename in filenames]):
|
2013-05-20 23:46:40 +02:00
|
|
|
dest = convert_path(source, sys.argv[1], sys.argv[2], ('.cg', '.glsl'))
|
2013-01-04 17:04:55 +01:00
|
|
|
dirpath = os.path.split(dest)[0]
|
|
|
|
print('Dirpath:', dirpath)
|
|
|
|
if not os.path.isdir(dirpath):
|
|
|
|
try:
|
|
|
|
os.makedirs(dirpath)
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno != errno.EEXIST:
|
|
|
|
raise
|
|
|
|
|
|
|
|
try:
|
|
|
|
ret = convert(source, dest)
|
|
|
|
print(source, '->', dest, '...', 'suceeded!' if ret == 0 else 'failed!')
|
|
|
|
|
|
|
|
if ret == 0:
|
|
|
|
success_cnt += 1
|
|
|
|
else:
|
|
|
|
failed_cnt += 1
|
|
|
|
failed_files.append(source)
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
failed_files.append(source)
|
|
|
|
failed_cnt += 1
|
|
|
|
|
2013-05-10 00:05:37 +02:00
|
|
|
for source in filter(lambda path: path_ext(path) == '.cgp', [os.path.join(dirname, filename) for filename in filenames]):
|
2013-05-20 23:46:40 +02:00
|
|
|
dest = convert_path(source, sys.argv[1], sys.argv[2], ('.cgp', '.glslp'))
|
2013-05-10 00:05:37 +02:00
|
|
|
dirpath = os.path.split(dest)[0]
|
|
|
|
print('Dirpath:', dirpath)
|
|
|
|
if not os.path.isdir(dirpath):
|
|
|
|
try:
|
|
|
|
os.makedirs(dirpath)
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno != errno.EEXIST:
|
|
|
|
raise
|
|
|
|
|
|
|
|
try:
|
|
|
|
convert_cgp(source, dest)
|
|
|
|
success_cnt += 1
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
failed_files.append(source)
|
|
|
|
failed_cnt += 1
|
|
|
|
|
2013-01-04 17:04:55 +01:00
|
|
|
print(success_cnt, 'shaders converted successfully.')
|
|
|
|
print(failed_cnt, 'shaders failed.')
|
|
|
|
if failed_cnt > 0:
|
|
|
|
print('Failed shaders:')
|
|
|
|
for path in failed_files:
|
|
|
|
print(path)
|
|
|
|
|
|
|
|
else:
|
|
|
|
source = sys.argv[1]
|
|
|
|
dest = sys.argv[2]
|
2013-05-10 00:05:37 +02:00
|
|
|
|
|
|
|
if path_ext(source) == '.cgp':
|
|
|
|
sys.exit(convert_cgp(source, dest))
|
|
|
|
else:
|
|
|
|
sys.exit(convert(source, dest))
|
2013-01-04 17:04:55 +01:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
sys.exit(main())
|
|
|
|
|