ffx_fsr1.h 59 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200
  1. //_____________________________________________________________/\_______________________________________________________________
  2. //==============================================================================================================================
  3. //
  4. //
  5. // AMD FidelityFX SUPER RESOLUTION [FSR 1] ::: SPATIAL SCALING & EXTRAS - v1.20210629
  6. //
  7. //
  8. //------------------------------------------------------------------------------------------------------------------------------
  9. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  10. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  11. //------------------------------------------------------------------------------------------------------------------------------
  12. // FidelityFX Super Resolution Sample
  13. //
  14. // Copyright (c) 2021 Advanced Micro Devices, Inc. All rights reserved.
  15. // Permission is hereby granted, free of charge, to any person obtaining a copy
  16. // of this software and associated documentation files(the "Software"), to deal
  17. // in the Software without restriction, including without limitation the rights
  18. // to use, copy, modify, merge, publish, distribute, sublicense, and / or sell
  19. // copies of the Software, and to permit persons to whom the Software is
  20. // furnished to do so, subject to the following conditions :
  21. // The above copyright notice and this permission notice shall be included in
  22. // all copies or substantial portions of the Software.
  23. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  24. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  25. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE
  26. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  27. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  28. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  29. // THE SOFTWARE.
  30. //------------------------------------------------------------------------------------------------------------------------------
  31. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  32. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  33. //------------------------------------------------------------------------------------------------------------------------------
  34. // ABOUT
  35. // =====
  36. // FSR is a collection of algorithms relating to generating a higher resolution image.
  37. // This specific header focuses on single-image non-temporal image scaling, and related tools.
  38. //
  39. // The core functions are EASU and RCAS:
  40. // [EASU] Edge Adaptive Spatial Upsampling ....... 1x to 4x area range spatial scaling, clamped adaptive elliptical filter.
  41. // [RCAS] Robust Contrast Adaptive Sharpening .... A non-scaling variation on CAS.
  42. // RCAS needs to be applied after EASU as a separate pass.
  43. //
  44. // Optional utility functions are:
  45. // [LFGA] Linear Film Grain Applicator ........... Tool to apply film grain after scaling.
  46. // [SRTM] Simple Reversible Tone-Mapper .......... Linear HDR {0 to FP16_MAX} to {0 to 1} and back.
  47. // [TEPD] Temporal Energy Preserving Dither ...... Temporally energy preserving dithered {0 to 1} linear to gamma 2.0 conversion.
  48. // See each individual sub-section for inline documentation.
  49. //------------------------------------------------------------------------------------------------------------------------------
  50. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  51. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  52. //------------------------------------------------------------------------------------------------------------------------------
  53. // FUNCTION PERMUTATIONS
  54. // =====================
  55. // *F() ..... Single item computation with 32-bit.
  56. // *H() ..... Single item computation with 16-bit, with packing (aka two 16-bit ops in parallel) when possible.
  57. // *Hx2() ... Processing two items in parallel with 16-bit, easier packing.
  58. // Not all interfaces in this file have a *Hx2() form.
  59. //==============================================================================================================================
  60. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  61. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  62. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  63. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  64. //_____________________________________________________________/\_______________________________________________________________
  65. //==============================================================================================================================
  66. //
  67. // FSR - [EASU] EDGE ADAPTIVE SPATIAL UPSAMPLING
  68. //
  69. //------------------------------------------------------------------------------------------------------------------------------
  70. // EASU provides a high quality spatial-only scaling at relatively low cost.
  71. // Meaning EASU is appropiate for laptops and other low-end GPUs.
  72. // Quality from 1x to 4x area scaling is good.
  73. //------------------------------------------------------------------------------------------------------------------------------
  74. // The scalar uses a modified fast approximation to the standard lanczos(size=2) kernel.
  75. // EASU runs in a single pass, so it applies a directionally and anisotropically adaptive radial lanczos.
  76. // This is also kept as simple as possible to have minimum runtime.
  77. //------------------------------------------------------------------------------------------------------------------------------
  78. // The lanzcos filter has negative lobes, so by itself it will introduce ringing.
  79. // To remove all ringing, the algorithm uses the nearest 2x2 input texels as a neighborhood,
  80. // and limits output to the minimum and maximum of that neighborhood.
  81. //------------------------------------------------------------------------------------------------------------------------------
  82. // Input image requirements:
  83. //
  84. // Color needs to be encoded as 3 channel[red, green, blue](e.g.XYZ not supported)
  85. // Each channel needs to be in the range[0, 1]
  86. // Any color primaries are supported
  87. // Display / tonemapping curve needs to be as if presenting to sRGB display or similar(e.g.Gamma 2.0)
  88. // There should be no banding in the input
  89. // There should be no high amplitude noise in the input
  90. // There should be no noise in the input that is not at input pixel granularity
  91. // For performance purposes, use 32bpp formats
  92. //------------------------------------------------------------------------------------------------------------------------------
  93. // Best to apply EASU at the end of the frame after tonemapping
  94. // but before film grain or composite of the UI.
  95. //------------------------------------------------------------------------------------------------------------------------------
  96. // Example of including this header for D3D HLSL :
  97. //
  98. // #define A_GPU 1
  99. // #define A_HLSL 1
  100. // #define A_HALF 1
  101. // #include "ffx_a.h"
  102. // #define FSR_EASU_H 1
  103. // #define FSR_RCAS_H 1
  104. // //declare input callbacks
  105. // #include "ffx_fsr1.h"
  106. //
  107. // Example of including this header for Vulkan GLSL :
  108. //
  109. // #define A_GPU 1
  110. // #define A_GLSL 1
  111. // #define A_HALF 1
  112. // #include "ffx_a.h"
  113. // #define FSR_EASU_H 1
  114. // #define FSR_RCAS_H 1
  115. // //declare input callbacks
  116. // #include "ffx_fsr1.h"
  117. //
  118. // Example of including this header for Vulkan HLSL :
  119. //
  120. // #define A_GPU 1
  121. // #define A_HLSL 1
  122. // #define A_HLSL_6_2 1
  123. // #define A_NO_16_BIT_CAST 1
  124. // #define A_HALF 1
  125. // #include "ffx_a.h"
  126. // #define FSR_EASU_H 1
  127. // #define FSR_RCAS_H 1
  128. // //declare input callbacks
  129. // #include "ffx_fsr1.h"
  130. //
  131. // Example of declaring the required input callbacks for GLSL :
  132. // The callbacks need to gather4 for each color channel using the specified texture coordinate 'p'.
  133. // EASU uses gather4 to reduce position computation logic and for free Arrays of Structures to Structures of Arrays conversion.
  134. //
  135. // AH4 FsrEasuRH(AF2 p){return AH4(textureGather(sampler2D(tex,sam),p,0));}
  136. // AH4 FsrEasuGH(AF2 p){return AH4(textureGather(sampler2D(tex,sam),p,1));}
  137. // AH4 FsrEasuBH(AF2 p){return AH4(textureGather(sampler2D(tex,sam),p,2));}
  138. // ...
  139. // The FsrEasuCon function needs to be called from the CPU or GPU to set up constants.
  140. // The difference in viewport and input image size is there to support Dynamic Resolution Scaling.
  141. // To use FsrEasuCon() on the CPU, define A_CPU before including ffx_a and ffx_fsr1.
  142. // Including a GPU example here, the 'con0' through 'con3' values would be stored out to a constant buffer.
  143. // AU4 con0,con1,con2,con3;
  144. // FsrEasuCon(con0,con1,con2,con3,
  145. // 1920.0,1080.0, // Viewport size (top left aligned) in the input image which is to be scaled.
  146. // 3840.0,2160.0, // The size of the input image.
  147. // 2560.0,1440.0); // The output resolution.
  148. //==============================================================================================================================
  149. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  150. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  151. //_____________________________________________________________/\_______________________________________________________________
  152. //==============================================================================================================================
  153. // CONSTANT SETUP
  154. //==============================================================================================================================
  155. // Call to setup required constant values (works on CPU or GPU).
  156. A_STATIC void FsrEasuCon(
  157. outAU4 con0,
  158. outAU4 con1,
  159. outAU4 con2,
  160. outAU4 con3,
  161. // This the rendered image resolution being upscaled
  162. AF1 inputViewportInPixelsX,
  163. AF1 inputViewportInPixelsY,
  164. // This is the resolution of the resource containing the input image (useful for dynamic resolution)
  165. AF1 inputSizeInPixelsX,
  166. AF1 inputSizeInPixelsY,
  167. // This is the display resolution which the input image gets upscaled to
  168. AF1 outputSizeInPixelsX,
  169. AF1 outputSizeInPixelsY){
  170. // Output integer position to a pixel position in viewport.
  171. con0[0]=AU1_AF1(inputViewportInPixelsX*ARcpF1(outputSizeInPixelsX));
  172. con0[1]=AU1_AF1(inputViewportInPixelsY*ARcpF1(outputSizeInPixelsY));
  173. con0[2]=AU1_AF1(AF1_(0.5)*inputViewportInPixelsX*ARcpF1(outputSizeInPixelsX)-AF1_(0.5));
  174. con0[3]=AU1_AF1(AF1_(0.5)*inputViewportInPixelsY*ARcpF1(outputSizeInPixelsY)-AF1_(0.5));
  175. // Viewport pixel position to normalized image space.
  176. // This is used to get upper-left of 'F' tap.
  177. con1[0]=AU1_AF1(ARcpF1(inputSizeInPixelsX));
  178. con1[1]=AU1_AF1(ARcpF1(inputSizeInPixelsY));
  179. // Centers of gather4, first offset from upper-left of 'F'.
  180. // +---+---+
  181. // | | |
  182. // +--(0)--+
  183. // | b | c |
  184. // +---F---+---+---+
  185. // | e | f | g | h |
  186. // +--(1)--+--(2)--+
  187. // | i | j | k | l |
  188. // +---+---+---+---+
  189. // | n | o |
  190. // +--(3)--+
  191. // | | |
  192. // +---+---+
  193. con1[2]=AU1_AF1(AF1_( 1.0)*ARcpF1(inputSizeInPixelsX));
  194. con1[3]=AU1_AF1(AF1_(-1.0)*ARcpF1(inputSizeInPixelsY));
  195. // These are from (0) instead of 'F'.
  196. con2[0]=AU1_AF1(AF1_(-1.0)*ARcpF1(inputSizeInPixelsX));
  197. con2[1]=AU1_AF1(AF1_( 2.0)*ARcpF1(inputSizeInPixelsY));
  198. con2[2]=AU1_AF1(AF1_( 1.0)*ARcpF1(inputSizeInPixelsX));
  199. con2[3]=AU1_AF1(AF1_( 2.0)*ARcpF1(inputSizeInPixelsY));
  200. con3[0]=AU1_AF1(AF1_( 0.0)*ARcpF1(inputSizeInPixelsX));
  201. con3[1]=AU1_AF1(AF1_( 4.0)*ARcpF1(inputSizeInPixelsY));
  202. con3[2]=con3[3]=0;}
  203. //If the an offset into the input image resource
  204. A_STATIC void FsrEasuConOffset(
  205. outAU4 con0,
  206. outAU4 con1,
  207. outAU4 con2,
  208. outAU4 con3,
  209. // This the rendered image resolution being upscaled
  210. AF1 inputViewportInPixelsX,
  211. AF1 inputViewportInPixelsY,
  212. // This is the resolution of the resource containing the input image (useful for dynamic resolution)
  213. AF1 inputSizeInPixelsX,
  214. AF1 inputSizeInPixelsY,
  215. // This is the display resolution which the input image gets upscaled to
  216. AF1 outputSizeInPixelsX,
  217. AF1 outputSizeInPixelsY,
  218. // This is the input image offset into the resource containing it (useful for dynamic resolution)
  219. AF1 inputOffsetInPixelsX,
  220. AF1 inputOffsetInPixelsY) {
  221. FsrEasuCon(con0, con1, con2, con3, inputViewportInPixelsX, inputViewportInPixelsY, inputSizeInPixelsX, inputSizeInPixelsY, outputSizeInPixelsX, outputSizeInPixelsY);
  222. con0[2] = AU1_AF1(AF1_(0.5) * inputViewportInPixelsX * ARcpF1(outputSizeInPixelsX) - AF1_(0.5) + inputOffsetInPixelsX);
  223. con0[3] = AU1_AF1(AF1_(0.5) * inputViewportInPixelsY * ARcpF1(outputSizeInPixelsY) - AF1_(0.5) + inputOffsetInPixelsY);
  224. }
  225. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  226. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  227. //_____________________________________________________________/\_______________________________________________________________
  228. //==============================================================================================================================
  229. // NON-PACKED 32-BIT VERSION
  230. //==============================================================================================================================
  231. #if defined(A_GPU)&&defined(FSR_EASU_F)
  232. // Input callback prototypes, need to be implemented by calling shader
  233. AF4 FsrEasuRF(AF2 p);
  234. AF4 FsrEasuGF(AF2 p);
  235. AF4 FsrEasuBF(AF2 p);
  236. //------------------------------------------------------------------------------------------------------------------------------
  237. // Filtering for a given tap for the scalar.
  238. void FsrEasuTapF(
  239. inout AF3 aC, // Accumulated color, with negative lobe.
  240. inout AF1 aW, // Accumulated weight.
  241. AF2 off, // Pixel offset from resolve position to tap.
  242. AF2 dir, // Gradient direction.
  243. AF2 len, // Length.
  244. AF1 lob, // Negative lobe strength.
  245. AF1 clp, // Clipping point.
  246. AF3 c){ // Tap color.
  247. // Rotate offset by direction.
  248. AF2 v;
  249. v.x=(off.x*( dir.x))+(off.y*dir.y);
  250. v.y=(off.x*(-dir.y))+(off.y*dir.x);
  251. // Anisotropy.
  252. v*=len;
  253. // Compute distance^2.
  254. AF1 d2=v.x*v.x+v.y*v.y;
  255. // Limit to the window as at corner, 2 taps can easily be outside.
  256. d2=min(d2,clp);
  257. // Approximation of lancos2 without sin() or rcp(), or sqrt() to get x.
  258. // (25/16 * (2/5 * x^2 - 1)^2 - (25/16 - 1)) * (1/4 * x^2 - 1)^2
  259. // |_______________________________________| |_______________|
  260. // base window
  261. // The general form of the 'base' is,
  262. // (a*(b*x^2-1)^2-(a-1))
  263. // Where 'a=1/(2*b-b^2)' and 'b' moves around the negative lobe.
  264. AF1 wB=AF1_(2.0/5.0)*d2+AF1_(-1.0);
  265. AF1 wA=lob*d2+AF1_(-1.0);
  266. wB*=wB;
  267. wA*=wA;
  268. wB=AF1_(25.0/16.0)*wB+AF1_(-(25.0/16.0-1.0));
  269. AF1 w=wB*wA;
  270. // Do weighted average.
  271. aC+=c*w;aW+=w;}
  272. //------------------------------------------------------------------------------------------------------------------------------
  273. // Accumulate direction and length.
  274. void FsrEasuSetF(
  275. inout AF2 dir,
  276. inout AF1 len,
  277. AF2 pp,
  278. AP1 biS,AP1 biT,AP1 biU,AP1 biV,
  279. AF1 lA,AF1 lB,AF1 lC,AF1 lD,AF1 lE){
  280. // Compute bilinear weight, branches factor out as predicates are compiler time immediates.
  281. // s t
  282. // u v
  283. AF1 w = AF1_(0.0);
  284. if(biS)w=(AF1_(1.0)-pp.x)*(AF1_(1.0)-pp.y);
  285. if(biT)w= pp.x *(AF1_(1.0)-pp.y);
  286. if(biU)w=(AF1_(1.0)-pp.x)* pp.y ;
  287. if(biV)w= pp.x * pp.y ;
  288. // Direction is the '+' diff.
  289. // a
  290. // b c d
  291. // e
  292. // Then takes magnitude from abs average of both sides of 'c'.
  293. // Length converts gradient reversal to 0, smoothly to non-reversal at 1, shaped, then adding horz and vert terms.
  294. AF1 dc=lD-lC;
  295. AF1 cb=lC-lB;
  296. AF1 lenX=max(abs(dc),abs(cb));
  297. lenX=APrxLoRcpF1(lenX);
  298. AF1 dirX=lD-lB;
  299. dir.x+=dirX*w;
  300. lenX=ASatF1(abs(dirX)*lenX);
  301. lenX*=lenX;
  302. len+=lenX*w;
  303. // Repeat for the y axis.
  304. AF1 ec=lE-lC;
  305. AF1 ca=lC-lA;
  306. AF1 lenY=max(abs(ec),abs(ca));
  307. lenY=APrxLoRcpF1(lenY);
  308. AF1 dirY=lE-lA;
  309. dir.y+=dirY*w;
  310. lenY=ASatF1(abs(dirY)*lenY);
  311. lenY*=lenY;
  312. len+=lenY*w;}
  313. //------------------------------------------------------------------------------------------------------------------------------
  314. void FsrEasuF(
  315. out AF3 pix,
  316. AU2 ip, // Integer pixel position in output.
  317. AU4 con0, // Constants generated by FsrEasuCon().
  318. AU4 con1,
  319. AU4 con2,
  320. AU4 con3){
  321. //------------------------------------------------------------------------------------------------------------------------------
  322. // Get position of 'f'.
  323. AF2 pp=AF2(ip)*AF2_AU2(con0.xy)+AF2_AU2(con0.zw);
  324. AF2 fp=floor(pp);
  325. pp-=fp;
  326. //------------------------------------------------------------------------------------------------------------------------------
  327. // 12-tap kernel.
  328. // b c
  329. // e f g h
  330. // i j k l
  331. // n o
  332. // Gather 4 ordering.
  333. // a b
  334. // r g
  335. // For packed FP16, need either {rg} or {ab} so using the following setup for gather in all versions,
  336. // a b <- unused (z)
  337. // r g
  338. // a b a b
  339. // r g r g
  340. // a b
  341. // r g <- unused (z)
  342. // Allowing dead-code removal to remove the 'z's.
  343. AF2 p0=fp*AF2_AU2(con1.xy)+AF2_AU2(con1.zw);
  344. // These are from p0 to avoid pulling two constants on pre-Navi hardware.
  345. AF2 p1=p0+AF2_AU2(con2.xy);
  346. AF2 p2=p0+AF2_AU2(con2.zw);
  347. AF2 p3=p0+AF2_AU2(con3.xy);
  348. AF4 bczzR=FsrEasuRF(p0);
  349. AF4 bczzG=FsrEasuGF(p0);
  350. AF4 bczzB=FsrEasuBF(p0);
  351. AF4 ijfeR=FsrEasuRF(p1);
  352. AF4 ijfeG=FsrEasuGF(p1);
  353. AF4 ijfeB=FsrEasuBF(p1);
  354. AF4 klhgR=FsrEasuRF(p2);
  355. AF4 klhgG=FsrEasuGF(p2);
  356. AF4 klhgB=FsrEasuBF(p2);
  357. AF4 zzonR=FsrEasuRF(p3);
  358. AF4 zzonG=FsrEasuGF(p3);
  359. AF4 zzonB=FsrEasuBF(p3);
  360. //------------------------------------------------------------------------------------------------------------------------------
  361. // Simplest multi-channel approximate luma possible (luma times 2, in 2 FMA/MAD).
  362. AF4 bczzL=bczzB*AF4_(0.5)+(bczzR*AF4_(0.5)+bczzG);
  363. AF4 ijfeL=ijfeB*AF4_(0.5)+(ijfeR*AF4_(0.5)+ijfeG);
  364. AF4 klhgL=klhgB*AF4_(0.5)+(klhgR*AF4_(0.5)+klhgG);
  365. AF4 zzonL=zzonB*AF4_(0.5)+(zzonR*AF4_(0.5)+zzonG);
  366. // Rename.
  367. AF1 bL=bczzL.x;
  368. AF1 cL=bczzL.y;
  369. AF1 iL=ijfeL.x;
  370. AF1 jL=ijfeL.y;
  371. AF1 fL=ijfeL.z;
  372. AF1 eL=ijfeL.w;
  373. AF1 kL=klhgL.x;
  374. AF1 lL=klhgL.y;
  375. AF1 hL=klhgL.z;
  376. AF1 gL=klhgL.w;
  377. AF1 oL=zzonL.z;
  378. AF1 nL=zzonL.w;
  379. // Accumulate for bilinear interpolation.
  380. AF2 dir=AF2_(0.0);
  381. AF1 len=AF1_(0.0);
  382. FsrEasuSetF(dir,len,pp,true, false,false,false,bL,eL,fL,gL,jL);
  383. FsrEasuSetF(dir,len,pp,false,true ,false,false,cL,fL,gL,hL,kL);
  384. FsrEasuSetF(dir,len,pp,false,false,true ,false,fL,iL,jL,kL,nL);
  385. FsrEasuSetF(dir,len,pp,false,false,false,true ,gL,jL,kL,lL,oL);
  386. //------------------------------------------------------------------------------------------------------------------------------
  387. // Normalize with approximation, and cleanup close to zero.
  388. AF2 dir2=dir*dir;
  389. AF1 dirR=dir2.x+dir2.y;
  390. AP1 zro=dirR<AF1_(1.0/32768.0);
  391. dirR=APrxLoRsqF1(dirR);
  392. dirR=zro?AF1_(1.0):dirR;
  393. dir.x=zro?AF1_(1.0):dir.x;
  394. dir*=AF2_(dirR);
  395. // Transform from {0 to 2} to {0 to 1} range, and shape with square.
  396. len=len*AF1_(0.5);
  397. len*=len;
  398. // Stretch kernel {1.0 vert|horz, to sqrt(2.0) on diagonal}.
  399. AF1 stretch=(dir.x*dir.x+dir.y*dir.y)*APrxLoRcpF1(max(abs(dir.x),abs(dir.y)));
  400. // Anisotropic length after rotation,
  401. // x := 1.0 lerp to 'stretch' on edges
  402. // y := 1.0 lerp to 2x on edges
  403. AF2 len2=AF2(AF1_(1.0)+(stretch-AF1_(1.0))*len,AF1_(1.0)+AF1_(-0.5)*len);
  404. // Based on the amount of 'edge',
  405. // the window shifts from +/-{sqrt(2.0) to slightly beyond 2.0}.
  406. AF1 lob=AF1_(0.5)+AF1_((1.0/4.0-0.04)-0.5)*len;
  407. // Set distance^2 clipping point to the end of the adjustable window.
  408. AF1 clp=APrxLoRcpF1(lob);
  409. //------------------------------------------------------------------------------------------------------------------------------
  410. // Accumulation mixed with min/max of 4 nearest.
  411. // b c
  412. // e f g h
  413. // i j k l
  414. // n o
  415. AF3 min4=min(AMin3F3(AF3(ijfeR.z,ijfeG.z,ijfeB.z),AF3(klhgR.w,klhgG.w,klhgB.w),AF3(ijfeR.y,ijfeG.y,ijfeB.y)),
  416. AF3(klhgR.x,klhgG.x,klhgB.x));
  417. AF3 max4=max(AMax3F3(AF3(ijfeR.z,ijfeG.z,ijfeB.z),AF3(klhgR.w,klhgG.w,klhgB.w),AF3(ijfeR.y,ijfeG.y,ijfeB.y)),
  418. AF3(klhgR.x,klhgG.x,klhgB.x));
  419. // Accumulation.
  420. AF3 aC=AF3_(0.0);
  421. AF1 aW=AF1_(0.0);
  422. FsrEasuTapF(aC,aW,AF2( 0.0,-1.0)-pp,dir,len2,lob,clp,AF3(bczzR.x,bczzG.x,bczzB.x)); // b
  423. FsrEasuTapF(aC,aW,AF2( 1.0,-1.0)-pp,dir,len2,lob,clp,AF3(bczzR.y,bczzG.y,bczzB.y)); // c
  424. FsrEasuTapF(aC,aW,AF2(-1.0, 1.0)-pp,dir,len2,lob,clp,AF3(ijfeR.x,ijfeG.x,ijfeB.x)); // i
  425. FsrEasuTapF(aC,aW,AF2( 0.0, 1.0)-pp,dir,len2,lob,clp,AF3(ijfeR.y,ijfeG.y,ijfeB.y)); // j
  426. FsrEasuTapF(aC,aW,AF2( 0.0, 0.0)-pp,dir,len2,lob,clp,AF3(ijfeR.z,ijfeG.z,ijfeB.z)); // f
  427. FsrEasuTapF(aC,aW,AF2(-1.0, 0.0)-pp,dir,len2,lob,clp,AF3(ijfeR.w,ijfeG.w,ijfeB.w)); // e
  428. FsrEasuTapF(aC,aW,AF2( 1.0, 1.0)-pp,dir,len2,lob,clp,AF3(klhgR.x,klhgG.x,klhgB.x)); // k
  429. FsrEasuTapF(aC,aW,AF2( 2.0, 1.0)-pp,dir,len2,lob,clp,AF3(klhgR.y,klhgG.y,klhgB.y)); // l
  430. FsrEasuTapF(aC,aW,AF2( 2.0, 0.0)-pp,dir,len2,lob,clp,AF3(klhgR.z,klhgG.z,klhgB.z)); // h
  431. FsrEasuTapF(aC,aW,AF2( 1.0, 0.0)-pp,dir,len2,lob,clp,AF3(klhgR.w,klhgG.w,klhgB.w)); // g
  432. FsrEasuTapF(aC,aW,AF2( 1.0, 2.0)-pp,dir,len2,lob,clp,AF3(zzonR.z,zzonG.z,zzonB.z)); // o
  433. FsrEasuTapF(aC,aW,AF2( 0.0, 2.0)-pp,dir,len2,lob,clp,AF3(zzonR.w,zzonG.w,zzonB.w)); // n
  434. //------------------------------------------------------------------------------------------------------------------------------
  435. // Normalize and dering.
  436. pix=min(max4,max(min4,aC*AF3_(ARcpF1(aW))));}
  437. #endif
  438. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  439. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  440. //_____________________________________________________________/\_______________________________________________________________
  441. //==============================================================================================================================
  442. // PACKED 16-BIT VERSION
  443. //==============================================================================================================================
  444. #if defined(A_GPU)&&defined(A_HALF)&&defined(FSR_EASU_H)
  445. // Input callback prototypes, need to be implemented by calling shader
  446. AH4 FsrEasuRH(AF2 p);
  447. AH4 FsrEasuGH(AF2 p);
  448. AH4 FsrEasuBH(AF2 p);
  449. //------------------------------------------------------------------------------------------------------------------------------
  450. // This runs 2 taps in parallel.
  451. void FsrEasuTapH(
  452. inout AH2 aCR,inout AH2 aCG,inout AH2 aCB,
  453. inout AH2 aW,
  454. AH2 offX,AH2 offY,
  455. AH2 dir,
  456. AH2 len,
  457. AH1 lob,
  458. AH1 clp,
  459. AH2 cR,AH2 cG,AH2 cB){
  460. AH2 vX,vY;
  461. vX=offX* dir.xx +offY*dir.yy;
  462. vY=offX*(-dir.yy)+offY*dir.xx;
  463. vX*=len.x;vY*=len.y;
  464. AH2 d2=vX*vX+vY*vY;
  465. d2=min(d2,AH2_(clp));
  466. AH2 wB=AH2_(2.0/5.0)*d2+AH2_(-1.0);
  467. AH2 wA=AH2_(lob)*d2+AH2_(-1.0);
  468. wB*=wB;
  469. wA*=wA;
  470. wB=AH2_(25.0/16.0)*wB+AH2_(-(25.0/16.0-1.0));
  471. AH2 w=wB*wA;
  472. aCR+=cR*w;aCG+=cG*w;aCB+=cB*w;aW+=w;}
  473. //------------------------------------------------------------------------------------------------------------------------------
  474. // This runs 2 taps in parallel.
  475. void FsrEasuSetH(
  476. inout AH2 dirPX,inout AH2 dirPY,
  477. inout AH2 lenP,
  478. AH2 pp,
  479. AP1 biST,AP1 biUV,
  480. AH2 lA,AH2 lB,AH2 lC,AH2 lD,AH2 lE){
  481. AH2 w = AH2_(0.0);
  482. if(biST)w=(AH2(1.0,0.0)+AH2(-pp.x,pp.x))*AH2_(AH1_(1.0)-pp.y);
  483. if(biUV)w=(AH2(1.0,0.0)+AH2(-pp.x,pp.x))*AH2_( pp.y);
  484. // ABS is not free in the packed FP16 path.
  485. AH2 dc=lD-lC;
  486. AH2 cb=lC-lB;
  487. AH2 lenX=max(abs(dc),abs(cb));
  488. lenX=ARcpH2(lenX);
  489. AH2 dirX=lD-lB;
  490. dirPX+=dirX*w;
  491. lenX=ASatH2(abs(dirX)*lenX);
  492. lenX*=lenX;
  493. lenP+=lenX*w;
  494. AH2 ec=lE-lC;
  495. AH2 ca=lC-lA;
  496. AH2 lenY=max(abs(ec),abs(ca));
  497. lenY=ARcpH2(lenY);
  498. AH2 dirY=lE-lA;
  499. dirPY+=dirY*w;
  500. lenY=ASatH2(abs(dirY)*lenY);
  501. lenY*=lenY;
  502. lenP+=lenY*w;}
  503. //------------------------------------------------------------------------------------------------------------------------------
  504. void FsrEasuH(
  505. out AH3 pix,
  506. AU2 ip,
  507. AU4 con0,
  508. AU4 con1,
  509. AU4 con2,
  510. AU4 con3){
  511. //------------------------------------------------------------------------------------------------------------------------------
  512. AF2 pp=AF2(ip)*AF2_AU2(con0.xy)+AF2_AU2(con0.zw);
  513. AF2 fp=floor(pp);
  514. pp-=fp;
  515. AH2 ppp=AH2(pp);
  516. //------------------------------------------------------------------------------------------------------------------------------
  517. AF2 p0=fp*AF2_AU2(con1.xy)+AF2_AU2(con1.zw);
  518. AF2 p1=p0+AF2_AU2(con2.xy);
  519. AF2 p2=p0+AF2_AU2(con2.zw);
  520. AF2 p3=p0+AF2_AU2(con3.xy);
  521. AH4 bczzR=FsrEasuRH(p0);
  522. AH4 bczzG=FsrEasuGH(p0);
  523. AH4 bczzB=FsrEasuBH(p0);
  524. AH4 ijfeR=FsrEasuRH(p1);
  525. AH4 ijfeG=FsrEasuGH(p1);
  526. AH4 ijfeB=FsrEasuBH(p1);
  527. AH4 klhgR=FsrEasuRH(p2);
  528. AH4 klhgG=FsrEasuGH(p2);
  529. AH4 klhgB=FsrEasuBH(p2);
  530. AH4 zzonR=FsrEasuRH(p3);
  531. AH4 zzonG=FsrEasuGH(p3);
  532. AH4 zzonB=FsrEasuBH(p3);
  533. //------------------------------------------------------------------------------------------------------------------------------
  534. AH4 bczzL=bczzB*AH4_(0.5)+(bczzR*AH4_(0.5)+bczzG);
  535. AH4 ijfeL=ijfeB*AH4_(0.5)+(ijfeR*AH4_(0.5)+ijfeG);
  536. AH4 klhgL=klhgB*AH4_(0.5)+(klhgR*AH4_(0.5)+klhgG);
  537. AH4 zzonL=zzonB*AH4_(0.5)+(zzonR*AH4_(0.5)+zzonG);
  538. AH1 bL=bczzL.x;
  539. AH1 cL=bczzL.y;
  540. AH1 iL=ijfeL.x;
  541. AH1 jL=ijfeL.y;
  542. AH1 fL=ijfeL.z;
  543. AH1 eL=ijfeL.w;
  544. AH1 kL=klhgL.x;
  545. AH1 lL=klhgL.y;
  546. AH1 hL=klhgL.z;
  547. AH1 gL=klhgL.w;
  548. AH1 oL=zzonL.z;
  549. AH1 nL=zzonL.w;
  550. // This part is different, accumulating 2 taps in parallel.
  551. AH2 dirPX=AH2_(0.0);
  552. AH2 dirPY=AH2_(0.0);
  553. AH2 lenP=AH2_(0.0);
  554. FsrEasuSetH(dirPX,dirPY,lenP,ppp,true, false,AH2(bL,cL),AH2(eL,fL),AH2(fL,gL),AH2(gL,hL),AH2(jL,kL));
  555. FsrEasuSetH(dirPX,dirPY,lenP,ppp,false,true ,AH2(fL,gL),AH2(iL,jL),AH2(jL,kL),AH2(kL,lL),AH2(nL,oL));
  556. AH2 dir=AH2(dirPX.r+dirPX.g,dirPY.r+dirPY.g);
  557. AH1 len=lenP.r+lenP.g;
  558. //------------------------------------------------------------------------------------------------------------------------------
  559. AH2 dir2=dir*dir;
  560. AH1 dirR=dir2.x+dir2.y;
  561. AP1 zro=dirR<AH1_(1.0/32768.0);
  562. dirR=APrxLoRsqH1(dirR);
  563. dirR=zro?AH1_(1.0):dirR;
  564. dir.x=zro?AH1_(1.0):dir.x;
  565. dir*=AH2_(dirR);
  566. len=len*AH1_(0.5);
  567. len*=len;
  568. AH1 stretch=(dir.x*dir.x+dir.y*dir.y)*APrxLoRcpH1(max(abs(dir.x),abs(dir.y)));
  569. AH2 len2=AH2(AH1_(1.0)+(stretch-AH1_(1.0))*len,AH1_(1.0)+AH1_(-0.5)*len);
  570. AH1 lob=AH1_(0.5)+AH1_((1.0/4.0-0.04)-0.5)*len;
  571. AH1 clp=APrxLoRcpH1(lob);
  572. //------------------------------------------------------------------------------------------------------------------------------
  573. // FP16 is different, using packed trick to do min and max in same operation.
  574. AH2 bothR=max(max(AH2(-ijfeR.z,ijfeR.z),AH2(-klhgR.w,klhgR.w)),max(AH2(-ijfeR.y,ijfeR.y),AH2(-klhgR.x,klhgR.x)));
  575. AH2 bothG=max(max(AH2(-ijfeG.z,ijfeG.z),AH2(-klhgG.w,klhgG.w)),max(AH2(-ijfeG.y,ijfeG.y),AH2(-klhgG.x,klhgG.x)));
  576. AH2 bothB=max(max(AH2(-ijfeB.z,ijfeB.z),AH2(-klhgB.w,klhgB.w)),max(AH2(-ijfeB.y,ijfeB.y),AH2(-klhgB.x,klhgB.x)));
  577. // This part is different for FP16, working pairs of taps at a time.
  578. AH2 pR=AH2_(0.0);
  579. AH2 pG=AH2_(0.0);
  580. AH2 pB=AH2_(0.0);
  581. AH2 pW=AH2_(0.0);
  582. FsrEasuTapH(pR,pG,pB,pW,AH2( 0.0, 1.0)-ppp.xx,AH2(-1.0,-1.0)-ppp.yy,dir,len2,lob,clp,bczzR.xy,bczzG.xy,bczzB.xy);
  583. FsrEasuTapH(pR,pG,pB,pW,AH2(-1.0, 0.0)-ppp.xx,AH2( 1.0, 1.0)-ppp.yy,dir,len2,lob,clp,ijfeR.xy,ijfeG.xy,ijfeB.xy);
  584. FsrEasuTapH(pR,pG,pB,pW,AH2( 0.0,-1.0)-ppp.xx,AH2( 0.0, 0.0)-ppp.yy,dir,len2,lob,clp,ijfeR.zw,ijfeG.zw,ijfeB.zw);
  585. FsrEasuTapH(pR,pG,pB,pW,AH2( 1.0, 2.0)-ppp.xx,AH2( 1.0, 1.0)-ppp.yy,dir,len2,lob,clp,klhgR.xy,klhgG.xy,klhgB.xy);
  586. FsrEasuTapH(pR,pG,pB,pW,AH2( 2.0, 1.0)-ppp.xx,AH2( 0.0, 0.0)-ppp.yy,dir,len2,lob,clp,klhgR.zw,klhgG.zw,klhgB.zw);
  587. FsrEasuTapH(pR,pG,pB,pW,AH2( 1.0, 0.0)-ppp.xx,AH2( 2.0, 2.0)-ppp.yy,dir,len2,lob,clp,zzonR.zw,zzonG.zw,zzonB.zw);
  588. AH3 aC=AH3(pR.x+pR.y,pG.x+pG.y,pB.x+pB.y);
  589. AH1 aW=pW.x+pW.y;
  590. //------------------------------------------------------------------------------------------------------------------------------
  591. // Slightly different for FP16 version due to combined min and max.
  592. pix=min(AH3(bothR.y,bothG.y,bothB.y),max(-AH3(bothR.x,bothG.x,bothB.x),aC*AH3_(ARcpH1(aW))));}
  593. #endif
  594. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  595. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  596. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  597. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  598. //_____________________________________________________________/\_______________________________________________________________
  599. //==============================================================================================================================
  600. //
  601. // FSR - [RCAS] ROBUST CONTRAST ADAPTIVE SHARPENING
  602. //
  603. //------------------------------------------------------------------------------------------------------------------------------
  604. // CAS uses a simplified mechanism to convert local contrast into a variable amount of sharpness.
  605. // RCAS uses a more exact mechanism, solving for the maximum local sharpness possible before clipping.
  606. // RCAS also has a built in process to limit sharpening of what it detects as possible noise.
  607. // RCAS sharper does not support scaling, as it should be applied after EASU scaling.
  608. // Pass EASU output straight into RCAS, no color conversions necessary.
  609. //------------------------------------------------------------------------------------------------------------------------------
  610. // RCAS is based on the following logic.
  611. // RCAS uses a 5 tap filter in a cross pattern (same as CAS),
  612. // w n
  613. // w 1 w for taps w m e
  614. // w s
  615. // Where 'w' is the negative lobe weight.
  616. // output = (w*(n+e+w+s)+m)/(4*w+1)
  617. // RCAS solves for 'w' by seeing where the signal might clip out of the {0 to 1} input range,
  618. // 0 == (w*(n+e+w+s)+m)/(4*w+1) -> w = -m/(n+e+w+s)
  619. // 1 == (w*(n+e+w+s)+m)/(4*w+1) -> w = (1-m)/(n+e+w+s-4*1)
  620. // Then chooses the 'w' which results in no clipping, limits 'w', and multiplies by the 'sharp' amount.
  621. // This solution above has issues with MSAA input as the steps along the gradient cause edge detection issues.
  622. // So RCAS uses 4x the maximum and 4x the minimum (depending on equation)in place of the individual taps.
  623. // As well as switching from 'm' to either the minimum or maximum (depending on side), to help in energy conservation.
  624. // This stabilizes RCAS.
  625. // RCAS does a simple highpass which is normalized against the local contrast then shaped,
  626. // 0.25
  627. // 0.25 -1 0.25
  628. // 0.25
  629. // This is used as a noise detection filter, to reduce the effect of RCAS on grain, and focus on real edges.
  630. //
  631. // GLSL example for the required callbacks :
  632. //
  633. // AH4 FsrRcasLoadH(ASW2 p){return AH4(imageLoad(imgSrc,ASU2(p)));}
  634. // void FsrRcasInputH(inout AH1 r,inout AH1 g,inout AH1 b)
  635. // {
  636. // //do any simple input color conversions here or leave empty if none needed
  637. // }
  638. //
  639. // FsrRcasCon need to be called from the CPU or GPU to set up constants.
  640. // Including a GPU example here, the 'con' value would be stored out to a constant buffer.
  641. //
  642. // AU4 con;
  643. // FsrRcasCon(con,
  644. // 0.0); // The scale is {0.0 := maximum sharpness, to N>0, where N is the number of stops (halving) of the reduction of sharpness}.
  645. // ---------------
  646. // RCAS sharpening supports a CAS-like pass-through alpha via,
  647. // #define FSR_RCAS_PASSTHROUGH_ALPHA 1
  648. // RCAS also supports a define to enable a more expensive path to avoid some sharpening of noise.
  649. // Would suggest it is better to apply film grain after RCAS sharpening (and after scaling) instead of using this define,
  650. // #define FSR_RCAS_DENOISE 1
  651. //==============================================================================================================================
  652. // This is set at the limit of providing unnatural results for sharpening.
  653. #define FSR_RCAS_LIMIT (0.25-(1.0/16.0))
  654. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  655. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  656. //_____________________________________________________________/\_______________________________________________________________
  657. //==============================================================================================================================
  658. // CONSTANT SETUP
  659. //==============================================================================================================================
  660. // Call to setup required constant values (works on CPU or GPU).
  661. A_STATIC void FsrRcasCon(
  662. outAU4 con,
  663. // The scale is {0.0 := maximum, to N>0, where N is the number of stops (halving) of the reduction of sharpness}.
  664. AF1 sharpness){
  665. // Transform from stops to linear value.
  666. sharpness=AExp2F1(-sharpness);
  667. varAF2(hSharp)=initAF2(sharpness,sharpness);
  668. con[0]=AU1_AF1(sharpness);
  669. con[1]=AU1_AH2_AF2(hSharp);
  670. con[2]=0;
  671. con[3]=0;}
  672. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  673. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  674. //_____________________________________________________________/\_______________________________________________________________
  675. //==============================================================================================================================
  676. // NON-PACKED 32-BIT VERSION
  677. //==============================================================================================================================
  678. #if defined(A_GPU)&&defined(FSR_RCAS_F)
  679. // Input callback prototypes that need to be implemented by calling shader
  680. AF4 FsrRcasLoadF(ASU2 p);
  681. void FsrRcasInputF(inout AF1 r,inout AF1 g,inout AF1 b);
  682. //------------------------------------------------------------------------------------------------------------------------------
  683. void FsrRcasF(
  684. out AF1 pixR, // Output values, non-vector so port between RcasFilter() and RcasFilterH() is easy.
  685. out AF1 pixG,
  686. out AF1 pixB,
  687. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  688. out AF1 pixA,
  689. #endif
  690. AU2 ip, // Integer pixel position in output.
  691. AU4 con){ // Constant generated by RcasSetup().
  692. // Algorithm uses minimal 3x3 pixel neighborhood.
  693. // b
  694. // d e f
  695. // h
  696. ASU2 sp=ASU2(ip);
  697. AF3 b=FsrRcasLoadF(sp+ASU2( 0,-1)).rgb;
  698. AF3 d=FsrRcasLoadF(sp+ASU2(-1, 0)).rgb;
  699. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  700. AF4 ee=FsrRcasLoadF(sp);
  701. AF3 e=ee.rgb;pixA=ee.a;
  702. #else
  703. AF3 e=FsrRcasLoadF(sp).rgb;
  704. #endif
  705. AF3 f=FsrRcasLoadF(sp+ASU2( 1, 0)).rgb;
  706. AF3 h=FsrRcasLoadF(sp+ASU2( 0, 1)).rgb;
  707. // Rename (32-bit) or regroup (16-bit).
  708. AF1 bR=b.r;
  709. AF1 bG=b.g;
  710. AF1 bB=b.b;
  711. AF1 dR=d.r;
  712. AF1 dG=d.g;
  713. AF1 dB=d.b;
  714. AF1 eR=e.r;
  715. AF1 eG=e.g;
  716. AF1 eB=e.b;
  717. AF1 fR=f.r;
  718. AF1 fG=f.g;
  719. AF1 fB=f.b;
  720. AF1 hR=h.r;
  721. AF1 hG=h.g;
  722. AF1 hB=h.b;
  723. // Run optional input transform.
  724. FsrRcasInputF(bR,bG,bB);
  725. FsrRcasInputF(dR,dG,dB);
  726. FsrRcasInputF(eR,eG,eB);
  727. FsrRcasInputF(fR,fG,fB);
  728. FsrRcasInputF(hR,hG,hB);
  729. // Luma times 2.
  730. AF1 bL=bB*AF1_(0.5)+(bR*AF1_(0.5)+bG);
  731. AF1 dL=dB*AF1_(0.5)+(dR*AF1_(0.5)+dG);
  732. AF1 eL=eB*AF1_(0.5)+(eR*AF1_(0.5)+eG);
  733. AF1 fL=fB*AF1_(0.5)+(fR*AF1_(0.5)+fG);
  734. AF1 hL=hB*AF1_(0.5)+(hR*AF1_(0.5)+hG);
  735. // Noise detection.
  736. AF1 nz=AF1_(0.25)*bL+AF1_(0.25)*dL+AF1_(0.25)*fL+AF1_(0.25)*hL-eL;
  737. nz=ASatF1(abs(nz)*APrxMedRcpF1(AMax3F1(AMax3F1(bL,dL,eL),fL,hL)-AMin3F1(AMin3F1(bL,dL,eL),fL,hL)));
  738. nz=AF1_(-0.5)*nz+AF1_(1.0);
  739. // Min and max of ring.
  740. AF1 mn4R=min(AMin3F1(bR,dR,fR),hR);
  741. AF1 mn4G=min(AMin3F1(bG,dG,fG),hG);
  742. AF1 mn4B=min(AMin3F1(bB,dB,fB),hB);
  743. AF1 mx4R=max(AMax3F1(bR,dR,fR),hR);
  744. AF1 mx4G=max(AMax3F1(bG,dG,fG),hG);
  745. AF1 mx4B=max(AMax3F1(bB,dB,fB),hB);
  746. // Immediate constants for peak range.
  747. AF2 peakC=AF2(1.0,-1.0*4.0);
  748. // Limiters, these need to be high precision RCPs.
  749. AF1 hitMinR=min(mn4R,eR)*ARcpF1(AF1_(4.0)*mx4R);
  750. AF1 hitMinG=min(mn4G,eG)*ARcpF1(AF1_(4.0)*mx4G);
  751. AF1 hitMinB=min(mn4B,eB)*ARcpF1(AF1_(4.0)*mx4B);
  752. AF1 hitMaxR=(peakC.x-max(mx4R,eR))*ARcpF1(AF1_(4.0)*mn4R+peakC.y);
  753. AF1 hitMaxG=(peakC.x-max(mx4G,eG))*ARcpF1(AF1_(4.0)*mn4G+peakC.y);
  754. AF1 hitMaxB=(peakC.x-max(mx4B,eB))*ARcpF1(AF1_(4.0)*mn4B+peakC.y);
  755. AF1 lobeR=max(-hitMinR,hitMaxR);
  756. AF1 lobeG=max(-hitMinG,hitMaxG);
  757. AF1 lobeB=max(-hitMinB,hitMaxB);
  758. AF1 lobe=max(AF1_(-FSR_RCAS_LIMIT),min(AMax3F1(lobeR,lobeG,lobeB),AF1_(0.0)))*AF1_AU1(con.x);
  759. // Apply noise removal.
  760. #ifdef FSR_RCAS_DENOISE
  761. lobe*=nz;
  762. #endif
  763. // Resolve, which needs the medium precision rcp approximation to avoid visible tonality changes.
  764. AF1 rcpL=APrxMedRcpF1(AF1_(4.0)*lobe+AF1_(1.0));
  765. pixR=(lobe*bR+lobe*dR+lobe*hR+lobe*fR+eR)*rcpL;
  766. pixG=(lobe*bG+lobe*dG+lobe*hG+lobe*fG+eG)*rcpL;
  767. pixB=(lobe*bB+lobe*dB+lobe*hB+lobe*fB+eB)*rcpL;
  768. return;}
  769. #endif
  770. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  771. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  772. //_____________________________________________________________/\_______________________________________________________________
  773. //==============================================================================================================================
  774. // NON-PACKED 16-BIT VERSION
  775. //==============================================================================================================================
  776. #if defined(A_GPU)&&defined(A_HALF)&&defined(FSR_RCAS_H)
  777. // Input callback prototypes that need to be implemented by calling shader
  778. AH4 FsrRcasLoadH(ASW2 p);
  779. void FsrRcasInputH(inout AH1 r,inout AH1 g,inout AH1 b);
  780. //------------------------------------------------------------------------------------------------------------------------------
  781. void FsrRcasH(
  782. out AH1 pixR, // Output values, non-vector so port between RcasFilter() and RcasFilterH() is easy.
  783. out AH1 pixG,
  784. out AH1 pixB,
  785. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  786. out AH1 pixA,
  787. #endif
  788. AU2 ip, // Integer pixel position in output.
  789. AU4 con){ // Constant generated by RcasSetup().
  790. // Sharpening algorithm uses minimal 3x3 pixel neighborhood.
  791. // b
  792. // d e f
  793. // h
  794. ASW2 sp=ASW2(ip);
  795. AH3 b=FsrRcasLoadH(sp+ASW2( 0,-1)).rgb;
  796. AH3 d=FsrRcasLoadH(sp+ASW2(-1, 0)).rgb;
  797. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  798. AH4 ee=FsrRcasLoadH(sp);
  799. AH3 e=ee.rgb;pixA=ee.a;
  800. #else
  801. AH3 e=FsrRcasLoadH(sp).rgb;
  802. #endif
  803. AH3 f=FsrRcasLoadH(sp+ASW2( 1, 0)).rgb;
  804. AH3 h=FsrRcasLoadH(sp+ASW2( 0, 1)).rgb;
  805. // Rename (32-bit) or regroup (16-bit).
  806. AH1 bR=b.r;
  807. AH1 bG=b.g;
  808. AH1 bB=b.b;
  809. AH1 dR=d.r;
  810. AH1 dG=d.g;
  811. AH1 dB=d.b;
  812. AH1 eR=e.r;
  813. AH1 eG=e.g;
  814. AH1 eB=e.b;
  815. AH1 fR=f.r;
  816. AH1 fG=f.g;
  817. AH1 fB=f.b;
  818. AH1 hR=h.r;
  819. AH1 hG=h.g;
  820. AH1 hB=h.b;
  821. // Run optional input transform.
  822. FsrRcasInputH(bR,bG,bB);
  823. FsrRcasInputH(dR,dG,dB);
  824. FsrRcasInputH(eR,eG,eB);
  825. FsrRcasInputH(fR,fG,fB);
  826. FsrRcasInputH(hR,hG,hB);
  827. // Luma times 2.
  828. AH1 bL=bB*AH1_(0.5)+(bR*AH1_(0.5)+bG);
  829. AH1 dL=dB*AH1_(0.5)+(dR*AH1_(0.5)+dG);
  830. AH1 eL=eB*AH1_(0.5)+(eR*AH1_(0.5)+eG);
  831. AH1 fL=fB*AH1_(0.5)+(fR*AH1_(0.5)+fG);
  832. AH1 hL=hB*AH1_(0.5)+(hR*AH1_(0.5)+hG);
  833. // Noise detection.
  834. AH1 nz=AH1_(0.25)*bL+AH1_(0.25)*dL+AH1_(0.25)*fL+AH1_(0.25)*hL-eL;
  835. nz=ASatH1(abs(nz)*APrxMedRcpH1(AMax3H1(AMax3H1(bL,dL,eL),fL,hL)-AMin3H1(AMin3H1(bL,dL,eL),fL,hL)));
  836. nz=AH1_(-0.5)*nz+AH1_(1.0);
  837. // Min and max of ring.
  838. AH1 mn4R=min(AMin3H1(bR,dR,fR),hR);
  839. AH1 mn4G=min(AMin3H1(bG,dG,fG),hG);
  840. AH1 mn4B=min(AMin3H1(bB,dB,fB),hB);
  841. AH1 mx4R=max(AMax3H1(bR,dR,fR),hR);
  842. AH1 mx4G=max(AMax3H1(bG,dG,fG),hG);
  843. AH1 mx4B=max(AMax3H1(bB,dB,fB),hB);
  844. // Immediate constants for peak range.
  845. AH2 peakC=AH2(1.0,-1.0*4.0);
  846. // Limiters, these need to be high precision RCPs.
  847. AH1 hitMinR=min(mn4R,eR)*ARcpH1(AH1_(4.0)*mx4R);
  848. AH1 hitMinG=min(mn4G,eG)*ARcpH1(AH1_(4.0)*mx4G);
  849. AH1 hitMinB=min(mn4B,eB)*ARcpH1(AH1_(4.0)*mx4B);
  850. AH1 hitMaxR=(peakC.x-max(mx4R,eR))*ARcpH1(AH1_(4.0)*mn4R+peakC.y);
  851. AH1 hitMaxG=(peakC.x-max(mx4G,eG))*ARcpH1(AH1_(4.0)*mn4G+peakC.y);
  852. AH1 hitMaxB=(peakC.x-max(mx4B,eB))*ARcpH1(AH1_(4.0)*mn4B+peakC.y);
  853. AH1 lobeR=max(-hitMinR,hitMaxR);
  854. AH1 lobeG=max(-hitMinG,hitMaxG);
  855. AH1 lobeB=max(-hitMinB,hitMaxB);
  856. AH1 lobe=max(AH1_(-FSR_RCAS_LIMIT),min(AMax3H1(lobeR,lobeG,lobeB),AH1_(0.0)))*AH2_AU1(con.y).x;
  857. // Apply noise removal.
  858. #ifdef FSR_RCAS_DENOISE
  859. lobe*=nz;
  860. #endif
  861. // Resolve, which needs the medium precision rcp approximation to avoid visible tonality changes.
  862. AH1 rcpL=APrxMedRcpH1(AH1_(4.0)*lobe+AH1_(1.0));
  863. pixR=(lobe*bR+lobe*dR+lobe*hR+lobe*fR+eR)*rcpL;
  864. pixG=(lobe*bG+lobe*dG+lobe*hG+lobe*fG+eG)*rcpL;
  865. pixB=(lobe*bB+lobe*dB+lobe*hB+lobe*fB+eB)*rcpL;}
  866. #endif
  867. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  868. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  869. //_____________________________________________________________/\_______________________________________________________________
  870. //==============================================================================================================================
  871. // PACKED 16-BIT VERSION
  872. //==============================================================================================================================
  873. #if defined(A_GPU)&&defined(A_HALF)&&defined(FSR_RCAS_HX2)
  874. // Input callback prototypes that need to be implemented by the calling shader
  875. AH4 FsrRcasLoadHx2(ASW2 p);
  876. void FsrRcasInputHx2(inout AH2 r,inout AH2 g,inout AH2 b);
  877. //------------------------------------------------------------------------------------------------------------------------------
  878. // Can be used to convert from packed Structures of Arrays to Arrays of Structures for store.
  879. void FsrRcasDepackHx2(out AH4 pix0,out AH4 pix1,AH2 pixR,AH2 pixG,AH2 pixB){
  880. #ifdef A_HLSL
  881. // Invoke a slower path for DX only, since it won't allow uninitialized values.
  882. pix0.a=pix1.a=0.0;
  883. #endif
  884. pix0.rgb=AH3(pixR.x,pixG.x,pixB.x);
  885. pix1.rgb=AH3(pixR.y,pixG.y,pixB.y);}
  886. //------------------------------------------------------------------------------------------------------------------------------
  887. void FsrRcasHx2(
  888. // Output values are for 2 8x8 tiles in a 16x8 region.
  889. // pix<R,G,B>.x = left 8x8 tile
  890. // pix<R,G,B>.y = right 8x8 tile
  891. // This enables later processing to easily be packed as well.
  892. out AH2 pixR,
  893. out AH2 pixG,
  894. out AH2 pixB,
  895. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  896. out AH2 pixA,
  897. #endif
  898. AU2 ip, // Integer pixel position in output.
  899. AU4 con){ // Constant generated by RcasSetup().
  900. // No scaling algorithm uses minimal 3x3 pixel neighborhood.
  901. ASW2 sp0=ASW2(ip);
  902. AH3 b0=FsrRcasLoadHx2(sp0+ASW2( 0,-1)).rgb;
  903. AH3 d0=FsrRcasLoadHx2(sp0+ASW2(-1, 0)).rgb;
  904. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  905. AH4 ee0=FsrRcasLoadHx2(sp0);
  906. AH3 e0=ee0.rgb;pixA.r=ee0.a;
  907. #else
  908. AH3 e0=FsrRcasLoadHx2(sp0).rgb;
  909. #endif
  910. AH3 f0=FsrRcasLoadHx2(sp0+ASW2( 1, 0)).rgb;
  911. AH3 h0=FsrRcasLoadHx2(sp0+ASW2( 0, 1)).rgb;
  912. ASW2 sp1=sp0+ASW2(8,0);
  913. AH3 b1=FsrRcasLoadHx2(sp1+ASW2( 0,-1)).rgb;
  914. AH3 d1=FsrRcasLoadHx2(sp1+ASW2(-1, 0)).rgb;
  915. #ifdef FSR_RCAS_PASSTHROUGH_ALPHA
  916. AH4 ee1=FsrRcasLoadHx2(sp1);
  917. AH3 e1=ee1.rgb;pixA.g=ee1.a;
  918. #else
  919. AH3 e1=FsrRcasLoadHx2(sp1).rgb;
  920. #endif
  921. AH3 f1=FsrRcasLoadHx2(sp1+ASW2( 1, 0)).rgb;
  922. AH3 h1=FsrRcasLoadHx2(sp1+ASW2( 0, 1)).rgb;
  923. // Arrays of Structures to Structures of Arrays conversion.
  924. AH2 bR=AH2(b0.r,b1.r);
  925. AH2 bG=AH2(b0.g,b1.g);
  926. AH2 bB=AH2(b0.b,b1.b);
  927. AH2 dR=AH2(d0.r,d1.r);
  928. AH2 dG=AH2(d0.g,d1.g);
  929. AH2 dB=AH2(d0.b,d1.b);
  930. AH2 eR=AH2(e0.r,e1.r);
  931. AH2 eG=AH2(e0.g,e1.g);
  932. AH2 eB=AH2(e0.b,e1.b);
  933. AH2 fR=AH2(f0.r,f1.r);
  934. AH2 fG=AH2(f0.g,f1.g);
  935. AH2 fB=AH2(f0.b,f1.b);
  936. AH2 hR=AH2(h0.r,h1.r);
  937. AH2 hG=AH2(h0.g,h1.g);
  938. AH2 hB=AH2(h0.b,h1.b);
  939. // Run optional input transform.
  940. FsrRcasInputHx2(bR,bG,bB);
  941. FsrRcasInputHx2(dR,dG,dB);
  942. FsrRcasInputHx2(eR,eG,eB);
  943. FsrRcasInputHx2(fR,fG,fB);
  944. FsrRcasInputHx2(hR,hG,hB);
  945. // Luma times 2.
  946. AH2 bL=bB*AH2_(0.5)+(bR*AH2_(0.5)+bG);
  947. AH2 dL=dB*AH2_(0.5)+(dR*AH2_(0.5)+dG);
  948. AH2 eL=eB*AH2_(0.5)+(eR*AH2_(0.5)+eG);
  949. AH2 fL=fB*AH2_(0.5)+(fR*AH2_(0.5)+fG);
  950. AH2 hL=hB*AH2_(0.5)+(hR*AH2_(0.5)+hG);
  951. // Noise detection.
  952. AH2 nz=AH2_(0.25)*bL+AH2_(0.25)*dL+AH2_(0.25)*fL+AH2_(0.25)*hL-eL;
  953. nz=ASatH2(abs(nz)*APrxMedRcpH2(AMax3H2(AMax3H2(bL,dL,eL),fL,hL)-AMin3H2(AMin3H2(bL,dL,eL),fL,hL)));
  954. nz=AH2_(-0.5)*nz+AH2_(1.0);
  955. // Min and max of ring.
  956. AH2 mn4R=min(AMin3H2(bR,dR,fR),hR);
  957. AH2 mn4G=min(AMin3H2(bG,dG,fG),hG);
  958. AH2 mn4B=min(AMin3H2(bB,dB,fB),hB);
  959. AH2 mx4R=max(AMax3H2(bR,dR,fR),hR);
  960. AH2 mx4G=max(AMax3H2(bG,dG,fG),hG);
  961. AH2 mx4B=max(AMax3H2(bB,dB,fB),hB);
  962. // Immediate constants for peak range.
  963. AH2 peakC=AH2(1.0,-1.0*4.0);
  964. // Limiters, these need to be high precision RCPs.
  965. AH2 hitMinR=min(mn4R,eR)*ARcpH2(AH2_(4.0)*mx4R);
  966. AH2 hitMinG=min(mn4G,eG)*ARcpH2(AH2_(4.0)*mx4G);
  967. AH2 hitMinB=min(mn4B,eB)*ARcpH2(AH2_(4.0)*mx4B);
  968. AH2 hitMaxR=(peakC.x-max(mx4R,eR))*ARcpH2(AH2_(4.0)*mn4R+peakC.y);
  969. AH2 hitMaxG=(peakC.x-max(mx4G,eG))*ARcpH2(AH2_(4.0)*mn4G+peakC.y);
  970. AH2 hitMaxB=(peakC.x-max(mx4B,eB))*ARcpH2(AH2_(4.0)*mn4B+peakC.y);
  971. AH2 lobeR=max(-hitMinR,hitMaxR);
  972. AH2 lobeG=max(-hitMinG,hitMaxG);
  973. AH2 lobeB=max(-hitMinB,hitMaxB);
  974. AH2 lobe=max(AH2_(-FSR_RCAS_LIMIT),min(AMax3H2(lobeR,lobeG,lobeB),AH2_(0.0)))*AH2_(AH2_AU1(con.y).x);
  975. // Apply noise removal.
  976. #ifdef FSR_RCAS_DENOISE
  977. lobe*=nz;
  978. #endif
  979. // Resolve, which needs the medium precision rcp approximation to avoid visible tonality changes.
  980. AH2 rcpL=APrxMedRcpH2(AH2_(4.0)*lobe+AH2_(1.0));
  981. pixR=(lobe*bR+lobe*dR+lobe*hR+lobe*fR+eR)*rcpL;
  982. pixG=(lobe*bG+lobe*dG+lobe*hG+lobe*fG+eG)*rcpL;
  983. pixB=(lobe*bB+lobe*dB+lobe*hB+lobe*fB+eB)*rcpL;}
  984. #endif
  985. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  986. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  987. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  988. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  989. //_____________________________________________________________/\_______________________________________________________________
  990. //==============================================================================================================================
  991. //
  992. // FSR - [LFGA] LINEAR FILM GRAIN APPLICATOR
  993. //
  994. //------------------------------------------------------------------------------------------------------------------------------
  995. // Adding output-resolution film grain after scaling is a good way to mask both rendering and scaling artifacts.
  996. // Suggest using tiled blue noise as film grain input, with peak noise frequency set for a specific look and feel.
  997. // The 'Lfga*()' functions provide a convenient way to introduce grain.
  998. // These functions limit grain based on distance to signal limits.
  999. // This is done so that the grain is temporally energy preserving, and thus won't modify image tonality.
  1000. // Grain application should be done in a linear colorspace.
  1001. // The grain should be temporally changing, but have a temporal sum per pixel that adds to zero (non-biased).
  1002. //------------------------------------------------------------------------------------------------------------------------------
  1003. // Usage,
  1004. // FsrLfga*(
  1005. // color, // In/out linear colorspace color {0 to 1} ranged.
  1006. // grain, // Per pixel grain texture value {-0.5 to 0.5} ranged, input is 3-channel to support colored grain.
  1007. // amount); // Amount of grain (0 to 1} ranged.
  1008. //------------------------------------------------------------------------------------------------------------------------------
  1009. // Example if grain texture is monochrome: 'FsrLfgaF(color,AF3_(grain),amount)'
  1010. //==============================================================================================================================
  1011. #if defined(A_GPU)
  1012. // Maximum grain is the minimum distance to the signal limit.
  1013. void FsrLfgaF(inout AF3 c,AF3 t,AF1 a){c+=(t*AF3_(a))*min(AF3_(1.0)-c,c);}
  1014. #endif
  1015. //==============================================================================================================================
  1016. #if defined(A_GPU)&&defined(A_HALF)
  1017. // Half precision version (slower).
  1018. void FsrLfgaH(inout AH3 c,AH3 t,AH1 a){c+=(t*AH3_(a))*min(AH3_(1.0)-c,c);}
  1019. //------------------------------------------------------------------------------------------------------------------------------
  1020. // Packed half precision version (faster).
  1021. void FsrLfgaHx2(inout AH2 cR,inout AH2 cG,inout AH2 cB,AH2 tR,AH2 tG,AH2 tB,AH1 a){
  1022. cR+=(tR*AH2_(a))*min(AH2_(1.0)-cR,cR);cG+=(tG*AH2_(a))*min(AH2_(1.0)-cG,cG);cB+=(tB*AH2_(a))*min(AH2_(1.0)-cB,cB);}
  1023. #endif
  1024. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1025. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1026. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1027. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1028. //_____________________________________________________________/\_______________________________________________________________
  1029. //==============================================================================================================================
  1030. //
  1031. // FSR - [SRTM] SIMPLE REVERSIBLE TONE-MAPPER
  1032. //
  1033. //------------------------------------------------------------------------------------------------------------------------------
  1034. // This provides a way to take linear HDR color {0 to FP16_MAX} and convert it into a temporary {0 to 1} ranged post-tonemapped linear.
  1035. // The tonemapper preserves RGB ratio, which helps maintain HDR color bleed during filtering.
  1036. //------------------------------------------------------------------------------------------------------------------------------
  1037. // Reversible tonemapper usage,
  1038. // FsrSrtm*(color); // {0 to FP16_MAX} converted to {0 to 1}.
  1039. // FsrSrtmInv*(color); // {0 to 1} converted into {0 to 32768, output peak safe for FP16}.
  1040. //==============================================================================================================================
  1041. #if defined(A_GPU)
  1042. void FsrSrtmF(inout AF3 c){c*=AF3_(ARcpF1(AMax3F1(c.r,c.g,c.b)+AF1_(1.0)));}
  1043. // The extra max solves the c=1.0 case (which is a /0).
  1044. void FsrSrtmInvF(inout AF3 c){c*=AF3_(ARcpF1(max(AF1_(1.0/32768.0),AF1_(1.0)-AMax3F1(c.r,c.g,c.b))));}
  1045. #endif
  1046. //==============================================================================================================================
  1047. #if defined(A_GPU)&&defined(A_HALF)
  1048. void FsrSrtmH(inout AH3 c){c*=AH3_(ARcpH1(AMax3H1(c.r,c.g,c.b)+AH1_(1.0)));}
  1049. void FsrSrtmInvH(inout AH3 c){c*=AH3_(ARcpH1(max(AH1_(1.0/32768.0),AH1_(1.0)-AMax3H1(c.r,c.g,c.b))));}
  1050. //------------------------------------------------------------------------------------------------------------------------------
  1051. void FsrSrtmHx2(inout AH2 cR,inout AH2 cG,inout AH2 cB){
  1052. AH2 rcp=ARcpH2(AMax3H2(cR,cG,cB)+AH2_(1.0));cR*=rcp;cG*=rcp;cB*=rcp;}
  1053. void FsrSrtmInvHx2(inout AH2 cR,inout AH2 cG,inout AH2 cB){
  1054. AH2 rcp=ARcpH2(max(AH2_(1.0/32768.0),AH2_(1.0)-AMax3H2(cR,cG,cB)));cR*=rcp;cG*=rcp;cB*=rcp;}
  1055. #endif
  1056. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1057. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1058. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1059. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1060. //_____________________________________________________________/\_______________________________________________________________
  1061. //==============================================================================================================================
  1062. //
  1063. // FSR - [TEPD] TEMPORAL ENERGY PRESERVING DITHER
  1064. //
  1065. //------------------------------------------------------------------------------------------------------------------------------
  1066. // Temporally energy preserving dithered {0 to 1} linear to gamma 2.0 conversion.
  1067. // Gamma 2.0 is used so that the conversion back to linear is just to square the color.
  1068. // The conversion comes in 8-bit and 10-bit modes, designed for output to 8-bit UNORM or 10:10:10:2 respectively.
  1069. // Given good non-biased temporal blue noise as dither input,
  1070. // the output dither will temporally conserve energy.
  1071. // This is done by choosing the linear nearest step point instead of perceptual nearest.
  1072. // See code below for details.
  1073. //------------------------------------------------------------------------------------------------------------------------------
  1074. // DX SPEC RULES FOR FLOAT->UNORM 8-BIT CONVERSION
  1075. // ===============================================
  1076. // - Output is 'uint(floor(saturate(n)*255.0+0.5))'.
  1077. // - Thus rounding is to nearest.
  1078. // - NaN gets converted to zero.
  1079. // - INF is clamped to {0.0 to 1.0}.
  1080. //==============================================================================================================================
  1081. #if defined(A_GPU)
  1082. // Hand tuned integer position to dither value, with more values than simple checkerboard.
  1083. // Only 32-bit has enough precision for this compddation.
  1084. // Output is {0 to <1}.
  1085. AF1 FsrTepdDitF(AU2 p,AU1 f){
  1086. AF1 x=AF1_(p.x+f);
  1087. AF1 y=AF1_(p.y);
  1088. // The 1.61803 golden ratio.
  1089. AF1 a=AF1_((1.0+sqrt(5.0))/2.0);
  1090. // Number designed to provide a good visual pattern.
  1091. AF1 b=AF1_(1.0/3.69);
  1092. x=x*a+(y*b);
  1093. return AFractF1(x);}
  1094. //------------------------------------------------------------------------------------------------------------------------------
  1095. // This version is 8-bit gamma 2.0.
  1096. // The 'c' input is {0 to 1}.
  1097. // Output is {0 to 1} ready for image store.
  1098. void FsrTepdC8F(inout AF3 c,AF1 dit){
  1099. AF3 n=sqrt(c);
  1100. n=floor(n*AF3_(255.0))*AF3_(1.0/255.0);
  1101. AF3 a=n*n;
  1102. AF3 b=n+AF3_(1.0/255.0);b=b*b;
  1103. // Ratio of 'a' to 'b' required to produce 'c'.
  1104. // APrxLoRcpF1() won't work here (at least for very high dynamic ranges).
  1105. // APrxMedRcpF1() is an IADD,FMA,MUL.
  1106. AF3 r=(c-b)*APrxMedRcpF3(a-b);
  1107. // Use the ratio as a cutoff to choose 'a' or 'b'.
  1108. // AGtZeroF1() is a MUL.
  1109. c=ASatF3(n+AGtZeroF3(AF3_(dit)-r)*AF3_(1.0/255.0));}
  1110. //------------------------------------------------------------------------------------------------------------------------------
  1111. // This version is 10-bit gamma 2.0.
  1112. // The 'c' input is {0 to 1}.
  1113. // Output is {0 to 1} ready for image store.
  1114. void FsrTepdC10F(inout AF3 c,AF1 dit){
  1115. AF3 n=sqrt(c);
  1116. n=floor(n*AF3_(1023.0))*AF3_(1.0/1023.0);
  1117. AF3 a=n*n;
  1118. AF3 b=n+AF3_(1.0/1023.0);b=b*b;
  1119. AF3 r=(c-b)*APrxMedRcpF3(a-b);
  1120. c=ASatF3(n+AGtZeroF3(AF3_(dit)-r)*AF3_(1.0/1023.0));}
  1121. #endif
  1122. //==============================================================================================================================
  1123. #if defined(A_GPU)&&defined(A_HALF)
  1124. AH1 FsrTepdDitH(AU2 p,AU1 f){
  1125. AF1 x=AF1_(p.x+f);
  1126. AF1 y=AF1_(p.y);
  1127. AF1 a=AF1_((1.0+sqrt(5.0))/2.0);
  1128. AF1 b=AF1_(1.0/3.69);
  1129. x=x*a+(y*b);
  1130. return AH1(AFractF1(x));}
  1131. //------------------------------------------------------------------------------------------------------------------------------
  1132. void FsrTepdC8H(inout AH3 c,AH1 dit){
  1133. AH3 n=sqrt(c);
  1134. n=floor(n*AH3_(255.0))*AH3_(1.0/255.0);
  1135. AH3 a=n*n;
  1136. AH3 b=n+AH3_(1.0/255.0);b=b*b;
  1137. AH3 r=(c-b)*APrxMedRcpH3(a-b);
  1138. c=ASatH3(n+AGtZeroH3(AH3_(dit)-r)*AH3_(1.0/255.0));}
  1139. //------------------------------------------------------------------------------------------------------------------------------
  1140. void FsrTepdC10H(inout AH3 c,AH1 dit){
  1141. AH3 n=sqrt(c);
  1142. n=floor(n*AH3_(1023.0))*AH3_(1.0/1023.0);
  1143. AH3 a=n*n;
  1144. AH3 b=n+AH3_(1.0/1023.0);b=b*b;
  1145. AH3 r=(c-b)*APrxMedRcpH3(a-b);
  1146. c=ASatH3(n+AGtZeroH3(AH3_(dit)-r)*AH3_(1.0/1023.0));}
  1147. //==============================================================================================================================
  1148. // This computes dither for positions 'p' and 'p+{8,0}'.
  1149. AH2 FsrTepdDitHx2(AU2 p,AU1 f){
  1150. AF2 x;
  1151. x.x=AF1_(p.x+f);
  1152. x.y=x.x+AF1_(8.0);
  1153. AF1 y=AF1_(p.y);
  1154. AF1 a=AF1_((1.0+sqrt(5.0))/2.0);
  1155. AF1 b=AF1_(1.0/3.69);
  1156. x=x*AF2_(a)+AF2_(y*b);
  1157. return AH2(AFractF2(x));}
  1158. //------------------------------------------------------------------------------------------------------------------------------
  1159. void FsrTepdC8Hx2(inout AH2 cR,inout AH2 cG,inout AH2 cB,AH2 dit){
  1160. AH2 nR=sqrt(cR);
  1161. AH2 nG=sqrt(cG);
  1162. AH2 nB=sqrt(cB);
  1163. nR=floor(nR*AH2_(255.0))*AH2_(1.0/255.0);
  1164. nG=floor(nG*AH2_(255.0))*AH2_(1.0/255.0);
  1165. nB=floor(nB*AH2_(255.0))*AH2_(1.0/255.0);
  1166. AH2 aR=nR*nR;
  1167. AH2 aG=nG*nG;
  1168. AH2 aB=nB*nB;
  1169. AH2 bR=nR+AH2_(1.0/255.0);bR=bR*bR;
  1170. AH2 bG=nG+AH2_(1.0/255.0);bG=bG*bG;
  1171. AH2 bB=nB+AH2_(1.0/255.0);bB=bB*bB;
  1172. AH2 rR=(cR-bR)*APrxMedRcpH2(aR-bR);
  1173. AH2 rG=(cG-bG)*APrxMedRcpH2(aG-bG);
  1174. AH2 rB=(cB-bB)*APrxMedRcpH2(aB-bB);
  1175. cR=ASatH2(nR+AGtZeroH2(dit-rR)*AH2_(1.0/255.0));
  1176. cG=ASatH2(nG+AGtZeroH2(dit-rG)*AH2_(1.0/255.0));
  1177. cB=ASatH2(nB+AGtZeroH2(dit-rB)*AH2_(1.0/255.0));}
  1178. //------------------------------------------------------------------------------------------------------------------------------
  1179. void FsrTepdC10Hx2(inout AH2 cR,inout AH2 cG,inout AH2 cB,AH2 dit){
  1180. AH2 nR=sqrt(cR);
  1181. AH2 nG=sqrt(cG);
  1182. AH2 nB=sqrt(cB);
  1183. nR=floor(nR*AH2_(1023.0))*AH2_(1.0/1023.0);
  1184. nG=floor(nG*AH2_(1023.0))*AH2_(1.0/1023.0);
  1185. nB=floor(nB*AH2_(1023.0))*AH2_(1.0/1023.0);
  1186. AH2 aR=nR*nR;
  1187. AH2 aG=nG*nG;
  1188. AH2 aB=nB*nB;
  1189. AH2 bR=nR+AH2_(1.0/1023.0);bR=bR*bR;
  1190. AH2 bG=nG+AH2_(1.0/1023.0);bG=bG*bG;
  1191. AH2 bB=nB+AH2_(1.0/1023.0);bB=bB*bB;
  1192. AH2 rR=(cR-bR)*APrxMedRcpH2(aR-bR);
  1193. AH2 rG=(cG-bG)*APrxMedRcpH2(aG-bG);
  1194. AH2 rB=(cB-bB)*APrxMedRcpH2(aB-bB);
  1195. cR=ASatH2(nR+AGtZeroH2(dit-rR)*AH2_(1.0/1023.0));
  1196. cG=ASatH2(nG+AGtZeroH2(dit-rG)*AH2_(1.0/1023.0));
  1197. cB=ASatH2(nB+AGtZeroH2(dit-rB)*AH2_(1.0/1023.0));}
  1198. #endif