Шейдер искажения LUT для линз VROne
Я пытаюсь создать шейдер, который искажает и зеркально отображает видеопоток, чтобы бифокальные линзы видели его как VROne.
VROne предоставляет пакет Unity, где я нашел таблицы поиска (LUT), которые определяют преобразование искажения.
У нас есть набор из 6 справочных таблиц, по 2 для каждого цвета.
LUT_XB, LUT_YB, а также LUT_XG,LUTYG, LUT_XR, LUT_YR. Результат показан здесь.
Код, предоставленный Zeiss:
//
// Copyright (C) 2014 - Carl Zeiss AG
//
// Distortion Shader
Shader "VROneSDK/LUTDistortion" {
Properties {
_MainTex ("Base (RGB)", 2D) = "white" {}
// We have a set of LUTs.
// for each color two. One for the x-direction
// and one for the y-direction. _LUT<direction>Tex<Color>
_LUTXTexR ("LUTXR", 2D) = "white" {}
_LUTYTexR ("LUTYR", 2D) = "white" {}
_LUTXTexG ("LUTXG", 2D) = "white" {}
_LUTYTexG ("LUTYG", 2D) = "white" {}
_LUTXTexB ("LUTXB", 2D) = "white" {}
_LUTYTexB ("LUTYB", 2D) = "white" {}
}
SubShader {
Pass {
CGPROGRAM
// We need to define a vertex shader even though
// we don't use one. Without this is an invalid
// shader for GLES.
#pragma target 3.0
#pragma vertex vert_img
// Fragement shader is function frag.
#pragma fragment frag
// Use unity
#include "UnityCG.cginc"
// We will use two variables _MainTex contains
// the source texture
uniform sampler2D _MainTex;
// One Texture for each LUT (direction and color).
uniform sampler2D _LUTXTexR;
uniform sampler2D _LUTYTexR;
uniform sampler2D _LUTXTexG;
uniform sampler2D _LUTYTexG;
uniform sampler2D _LUTXTexB;
uniform sampler2D _LUTYTexB;
// mirror flag for right eye (textures are for left
// eye and mirrored for right eye)
uniform bool _isMirrored;
// Decoding a color value from the
// texture into a float. Similar to unitys
// DecodeFloatRGBA and DecodeFloatRG.
float DecodeFloatRGB(float3 rgb) {
return dot(rgb, float3(1.0,1.0/255.0,1.0/65025.0));
}
float mirrored(float coord) {
if (_isMirrored) {
return 1.0f - coord;
}
return coord;
}
// compute new lookup position from a LUT.zz
// For each color we extract the rgb value at the
// coordinate we are interested in. This rgb value
// indicates which pixel (or interpolated pixel) we
// need to map.
float2 LUTDistortionR(float2 coord)
{
float3 lookupX = tex2D(_LUTXTexR, coord).rgb;
float3 lookupY = tex2D(_LUTYTexR, coord).rgb;
return float2(mirrored(DecodeFloatRGB(lookupX)),DecodeFloatRGB(lookupY));
}
float2 LUTDistortionG(float2 coord)
{
float3 lookupX = tex2D(_LUTXTexG, coord).rgb;
float3 lookupY = tex2D(_LUTYTexG, coord).rgb;
return float2(mirrored(DecodeFloatRGB(lookupX)),DecodeFloatRGB(lookupY));
}
float2 LUTDistortionB(float2 coord)
{
float3 lookupX = tex2D(_LUTXTexB, coord).rgb;
float3 lookupY = tex2D(_LUTYTexB, coord).rgb;
return float2(mirrored(DecodeFloatRGB(lookupX)),DecodeFloatRGB(lookupY));
}
float4 frag(v2f_img i) : COLOR
{
// our result will be initialized to 0/0/0.
float3 res = float3(0.0f,0.0f,0.0f);
// Get the target (u,v) coordinate (i.uv)
// which is where we will draw the pixel.
// What we will draw, depends on the color
// and the distortion, which we can look up in
// the LUT. We do this for each color and do
// not put xy in rb or similar to allow us to
// improve precision with the DecodeFloatRGB method,
// as can be seen above.
// since textures are for left eye only, we need to
// "mirror" the input coordinate for the right eye.
float2 coord = float2(mirrored(i.uv.x), i.uv.y);
float2 xyR = LUTDistortionR(coord);
if (xyR.x <= 0.0f || xyR.y <= 0.0f || xyR.x >= 1.0f || xyR.y >= 1.0f) {
// set alpha to 1 and return.
return float4(res, 1.0f);
}
float2 xyG = LUTDistortionG(coord);
if (xyG.x <= 0.0f || xyG.y <= 0.0f || xyG.x >= 1.0f || xyG.y >= 1.0f) {
// set alpha to 1 and return.
return float4(res, 1.0f);
}
float2 xyB = LUTDistortionB(coord);
if (xyB.x <= 0.0f || xyB.y <= 0.0f || xyG.x >= 1.0f || xyG.y >= 1.0f) {
// set alpha to 1 and return.
return float4(res, 1.0f);
}
res = float3(tex2D(_MainTex,xyR).r,
tex2D(_MainTex,xyG).g,
tex2D(_MainTex,xyB).b);
// set alpha to 1 and return.
return float4(res, 1.0f);
}
ENDCG
}
}
Я хочу создать фрагментный шейдер, который преобразует поток кадра YUV (я создал sampler2D для каждого y, u, v: sampler2D s_texture_y...).
Код, который я реализовал:
NSString *const vertexShaderString = SHADER_STRING
(
attribute vec4 position;
attribute vec2 texcoord;
uniform mat4 modelViewProjectionMatrix;
varying vec2 v_texcoord;
void main()
{
gl_Position = modelViewProjectionMatrix * position;
v_texcoord = texcoord.xy;
}
);
NSString *const distortionVROneShaderString = SHADER_STRING
(
precision highp float;
varying highp vec2 v_texcoord;
uniform sampler2D s_texture_y;
uniform sampler2D s_texture_u;
uniform sampler2D s_texture_v;
uniform sampler2D LUTXTexR;
uniform sampler2D LUTYTexR;
uniform sampler2D LUTXTexG;
uniform sampler2D LUTYTexG;
uniform sampler2D LUTXTexB;
uniform sampler2D LUTYTexB;
float DecodeFloatRGB(vec3 rgb) {
return dot(rgb, vec3(1.0,1.0/255.0,1.0/65025.0));
}
vec2 LUTDistortionR(vec2 coord)
{
vec3 lookupX = texture2D(LUTXTexR, coord).rgb;
vec3 lookupY = texture2D(LUTYTexR, coord).rgb;
return vec2(DecodeFloatRGB(lookupX),DecodeFloatRGB(lookupY));
}
vec2 LUTDistortionG(vec2 coord)
{
vec3 lookupX = texture2D(LUTXTexG, coord).rgb;
vec3 lookupY = texture2D(LUTYTexG, coord).rgb;
return vec2(DecodeFloatRGB(lookupX),DecodeFloatRGB(lookupY));
}
vec2 LUTDistortionB(vec2 coord)
{
vec3 lookupX = texture2D(LUTXTexB, coord).rgb;
vec3 lookupY = texture2D(LUTYTexB, coord).rgb;
return vec2(DecodeFloatRGB(lookupX),DecodeFloatRGB(lookupY));
}
void main()
{
highp float y = texture2D(s_texture_y, v_texcoord).r * 1.0;
highp float u = texture2D(s_texture_u, v_texcoord).r - 0.5;
highp float v = texture2D(s_texture_v, v_texcoord).r - 0.5;
highp float r = y + 1.402 * v;
highp float g = y - 0.344 * u - 0.714 * v;
highp float b = y + 1.772 * u;
vec3 textureRGB = vec3(r,g,b);
vec3 res = vec3(0.0,0.0,0.0);
vec2 xyR = LUTDistortionR(v_texcoord);
vec2 xyG = LUTDistortionG(v_texcoord);
vec2 xyB = LUTDistortionB(v_texcoord);
gl_FragColor = vec4(texture2D(s_texture_y, xyR).r,texture2D(s_texture_u,xyG).r,texture2D(s_texture_v, xyB).r,1.0);// vec4(res,1.0);
}
);
Результат, который я получаю с этим кодом, похоже, не искажает видеопоток, я подозреваю, что импорт LUT как LUTXTexR, LUTXTexG, LUTXTexB ... должен быть выполнен как bufferSample.
Я застрял, любая помощь, указание или ссылка на учебник приветствуется, так как я не мог найти соответствующие учебники для этого конкретного приложения.