SpaceTransforms.hlsl 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. #ifndef UNITY_SPACE_TRANSFORMS_INCLUDED
  2. #define UNITY_SPACE_TRANSFORMS_INCLUDED
  3. // Return the PreTranslated ObjectToWorld Matrix (i.e matrix with _WorldSpaceCameraPos apply to it if we use camera relative rendering)
  4. float4x4 GetObjectToWorldMatrix()
  5. {
  6. return UNITY_MATRIX_M;
  7. }
  8. float4x4 GetWorldToObjectMatrix()
  9. {
  10. return UNITY_MATRIX_I_M;
  11. }
  12. float4x4 GetWorldToViewMatrix()
  13. {
  14. return UNITY_MATRIX_V;
  15. }
  16. // Transform to homogenous clip space
  17. float4x4 GetWorldToHClipMatrix()
  18. {
  19. return UNITY_MATRIX_VP;
  20. }
  21. // Transform to homogenous clip space
  22. float4x4 GetViewToHClipMatrix()
  23. {
  24. return UNITY_MATRIX_P;
  25. }
  26. // This function always return the absolute position in WS
  27. float3 GetAbsolutePositionWS(float3 positionRWS)
  28. {
  29. #if (SHADEROPTIONS_CAMERA_RELATIVE_RENDERING != 0)
  30. positionRWS += _WorldSpaceCameraPos;
  31. #endif
  32. return positionRWS;
  33. }
  34. // This function return the camera relative position in WS
  35. float3 GetCameraRelativePositionWS(float3 positionWS)
  36. {
  37. #if (SHADEROPTIONS_CAMERA_RELATIVE_RENDERING != 0)
  38. positionWS -= _WorldSpaceCameraPos;
  39. #endif
  40. return positionWS;
  41. }
  42. real GetOddNegativeScale()
  43. {
  44. return unity_WorldTransformParams.w;
  45. }
  46. float3 TransformObjectToWorld(float3 positionOS)
  47. {
  48. return mul(GetObjectToWorldMatrix(), float4(positionOS, 1.0)).xyz;
  49. }
  50. float3 TransformWorldToObject(float3 positionWS)
  51. {
  52. return mul(GetWorldToObjectMatrix(), float4(positionWS, 1.0)).xyz;
  53. }
  54. float3 TransformWorldToView(float3 positionWS)
  55. {
  56. return mul(GetWorldToViewMatrix(), float4(positionWS, 1.0)).xyz;
  57. }
  58. // Transforms position from object space to homogenous space
  59. float4 TransformObjectToHClip(float3 positionOS)
  60. {
  61. // More efficient than computing M*VP matrix product
  62. return mul(GetWorldToHClipMatrix(), mul(GetObjectToWorldMatrix(), float4(positionOS, 1.0)));
  63. }
  64. // Tranforms position from world space to homogenous space
  65. float4 TransformWorldToHClip(float3 positionWS)
  66. {
  67. return mul(GetWorldToHClipMatrix(), float4(positionWS, 1.0));
  68. }
  69. // Tranforms position from view space to homogenous space
  70. float4 TransformWViewToHClip(float3 positionVS)
  71. {
  72. return mul(GetViewToHClipMatrix(), float4(positionVS, 1.0));
  73. }
  74. real3 TransformObjectToWorldDir(real3 dirOS)
  75. {
  76. // Normalize to support uniform scaling
  77. return SafeNormalize(mul((real3x3)GetObjectToWorldMatrix(), dirOS));
  78. }
  79. real3 TransformWorldToObjectDir(real3 dirWS)
  80. {
  81. // Normalize to support uniform scaling
  82. return normalize(mul((real3x3)GetWorldToObjectMatrix(), dirWS));
  83. }
  84. real3 TransformWorldToViewDir(real3 dirWS)
  85. {
  86. return mul((real3x3)GetWorldToViewMatrix(), dirWS).xyz;
  87. }
  88. // Tranforms vector from world space to homogenous space
  89. real3 TransformWorldToHClipDir(real3 directionWS)
  90. {
  91. return mul((real3x3)GetWorldToHClipMatrix(), directionWS);
  92. }
  93. // Transforms normal from object to world space
  94. float3 TransformObjectToWorldNormal(float3 normalOS)
  95. {
  96. #ifdef UNITY_ASSUME_UNIFORM_SCALING
  97. return TransformObjectToWorldDir(normalOS);
  98. #else
  99. // Normal need to be multiply by inverse transpose
  100. return SafeNormalize(mul(normalOS, (float3x3)GetWorldToObjectMatrix()));
  101. #endif
  102. }
  103. // Transforms normal from world to object space
  104. float3 TransformWorldToObjectNormal(float3 normalWS)
  105. {
  106. #ifdef UNITY_ASSUME_UNIFORM_SCALING
  107. return TransformWorldToObjectDir(normalWS);
  108. #else
  109. // Normal need to be multiply by inverse transpose
  110. return SafeNormalize(mul(normalWS, (float3x3)GetObjectToWorldMatrix()));
  111. #endif
  112. }
  113. real3x3 CreateTangentToWorld(real3 normal, real3 tangent, real flipSign)
  114. {
  115. // For odd-negative scale transforms we need to flip the sign
  116. real sgn = flipSign * GetOddNegativeScale();
  117. real3 bitangent = cross(normal, tangent) * sgn;
  118. return real3x3(tangent, bitangent, normal);
  119. }
  120. real3 TransformTangentToWorld(real3 dirTS, real3x3 tangentToWorld)
  121. {
  122. // Note matrix is in row major convention with left multiplication as it is build on the fly
  123. return mul(dirTS, tangentToWorld);
  124. }
  125. real3 TransformWorldToTangent(real3 dirWS, real3x3 tangentToWorld)
  126. {
  127. // Note matrix is in row major convention with left multiplication as it is build on the fly
  128. float3 row0 = tangentToWorld[0];
  129. float3 row1 = tangentToWorld[1];
  130. float3 row2 = tangentToWorld[2];
  131. // these are the columns of the inverse matrix but scaled by the determinant
  132. float3 col0 = cross(row1, row2);
  133. float3 col1 = cross(row2, row0);
  134. float3 col2 = cross(row0, row1);
  135. float determinant = dot(row0, col0);
  136. float sgn = determinant<0.0 ? (-1.0) : 1.0;
  137. // inverse transposed but scaled by determinant
  138. real3x3 matTBN_I_T = real3x3(col0, col1, col2);
  139. // remove transpose part by using matrix as the first arg in mul()
  140. // this makes it the exact inverse of what TransformTangentToWorld() does.
  141. return SafeNormalize( sgn * mul(matTBN_I_T, dirWS) );
  142. }
  143. real3 TransformTangentToObject(real3 dirTS, real3x3 tangentToWorld)
  144. {
  145. // Note matrix is in row major convention with left multiplication as it is build on the fly
  146. real3 normalWS = TransformTangentToWorld(dirTS, tangentToWorld);
  147. return TransformWorldToObjectNormal(normalWS);
  148. }
  149. real3 TransformObjectToTangent(real3 dirOS, real3x3 tangentToWorld)
  150. {
  151. // Note matrix is in row major convention with left multiplication as it is build on the fly
  152. float3 normalWS = TransformObjectToWorldNormal(dirOS);
  153. return TransformWorldToTangent(normalWS, tangentToWorld);
  154. }
  155. #endif