I’ve put together this simple* mod for sonkun’s CRT presets to approximate the colors my real CRTs from 1989 and 2000, using the measurements that I posted last month. You can also still use sonkun’s own colors adjustments with the more accurate color, or you can get unmodified RGB from your emulator without adjustments. Some screenshots can be found here: https://imgur.com/a/oIaoA2d
Or at least, that’s what’s supposed to happen. My way of doing this is fairly hacky and imprecise, and the accuracy of the colors depends to some degree on your display’s resolution and RetroArch’s aspect ratio and overscan-cropping settings. What I did was make a shader program that takes almost 10 hours to generate an approximate inverse LUT of sonkun’s presets, and mod that LUT into one of the shader passes. This can be done with other CRT shaders in RetroArch too.
If you want to save yourself 10 hours at the cost of some accuracy, you can just use the LUT that I generated against my bizzare “1600p” display, but for better results on your own monitor’s resolution and scaling settings, I’m also posting instructions on how to generate an inverse LUT that’s optimized for your setup, which will take 10 hours of your time. Just beware that I’m probably going to edit this post later today or tomorrow to add a better LUT generator and to improve the precision a little bit more.
oh no
Consider reading through all these instructions before starting. It’s like a meme, with how long and bizzare it is. You’ll need Java for one part.
Keep in mind that this LUT will be dependent on all the following things. If one thing changes, your LUT will be uncalibrated.
- The internal vertical resolution of the core. Make sure you do not crop vertical overscan, so that your vertical resolution is 240 or as close to 240 as possible. Both RetroArch itself and your individual cores have settings that affect vertical overscan. To make sure your core is outputting about 240 scanlines, try loading shaders_slang/misc/print_resolution.
- The final resolution that is output to your screen. The vertical resolution is especially important, as that controls the thickness of scanlines, even if the content is staying at 240 lines across cores. Consider going into RetroArch’s settings and forcing a 4:3, 5:4, or 6:5 aspect ratio across all cores.
- Curved screen vs flat screen
- Trinitron vs slot mask vs shadow mask
- Any updates to crt-guest and/or to sonkun’s presets might affect results.
In shaders_slang/crt/shaders/guest/advanced/grade, make a copy of pre-shaders-afterglow-grade.slang, and call it passthru.slang. Open passthru.slang in a text editor other than Windows notepad and locate this line:
FragColor = vec4(LUT2_output + aftglow.rgb, 1.0);
and replace it with this
FragColor = vec4(imgColor.rgb, 1.0);
and save the file.
Inside shaders_slang/sonkun/… locate the preset that you want to mod, and make a copy of it. You can add 00 to the beginning of the file name to move it to the top of the file list. Open this file in your text editor, and search for this string:
"../../../../crt/shaders/guest/advanced/grade/pre-shaders-afterglow-grade.slang"
Replace it with this:
"../../../../crt/shaders/guest/advanced/grade/passthru.slang"
and save the file. Verify that you are able to load this modified sonkun shader in RetroArch.
We are about to create the most useless shader in the history of shaders. Locate stock.slang inside your shaders_slang folder. Make a copy of it called white.slang, and locate this line inside white.slang:
FragColor = vec4(texture(Source, vTexCoord).rgb, 1.0);
Replace that with this:
FragColor = vec4(1.0);
Also in shaders_slang, locate nearest.slangp. Make a copy of it called white.slangp, and locate this line inside white.slangp:
shader0 = stock.slang
Change it to this
shader0 = white.slang
Add this new line after it
alias0 = LutIn
Download my LUT generation shaders at https://www.mediafire.com/file/odx29b6vpd2mcvl/auto-lut-invert-128.zip/file and unzip them into your shaders folder. I forgot to include a couple things. Make a copy of auto-lut-output-invert.slangp called average-screen.slangp and locate this line at the top of average-screen.slangp:
shaders = "2"
Change it to
shaders = "1"
In RetroArch, go into your hotkey settings, and make sure you know how to screenshot at any time. You will need to be able to screenshot the LUT when it’s done generating using RetroArch’s built-in screenshot feature, without accidentally causing the LUT to reset to pitch-black. Screenshots get saved into a folder named “screenshots”. Also make sure that you have no other overlays on-screen, and that you know how to show and hide your FPS just by pressing a hotkey, which I believe is F3 by default.
In RetroArch, go into a game with 240 scanlines of resolution. As described in an earlier part of these instructions, make sure you’re actually at 240 scanlines or something close enough like 239. Load shaders_slang/white.slangp. Append your modded sonkun preset. Append the average-screen.slangp file we created earlier. You should see a black screen with two white rectangles. Your FPS might drop very badly, or it might stay stable.
Scroll to the bottom of the shader settings, where you see the options “Average size”, “Point size”, and “Average scale”. Decrease “Point size” to the lowest it’ll go without the two rectangles disappearing. “Average size” is the width and height (in pixels) of a rectangle in the middle of your screen that is being averaged, so try to adjust this to something that is reasonable to you while not dropping your framerate under 60 FPS. On my PC, 200 pixels works well.
Take a screenshot of the two white rectangles, and open the screenshot in an image editor. One of the rectangles should be 255, 255, 255, while the other should be something else. In your shader settings, change “Average scale” to that something else. Screenshot again, and now both rectangles should be 255, 255, 255.
Load your modified sonkun shader again. Prepend auto-lut-input-128.slangp, and append auto-lut-output-invert.slangp. You should now see the LUT being generated, but it’s not correct yet. Quickly, in the shader’s settings, set “Afterglow Strength” to 0.0, then scroll all the way down to the settings at the bottom, and change the averager settings to whatever you set them to earlier. Starting right now, this is the part that takes 10 hours, assuming your game is running at 60 FPS constantly. BE CAREFUL because doing certain actions in RetroArch will cause the LUT to reset to a black screen.
On some computers, you might be able to do certain glitches in RetroArch to speed this up. For example, I’m able to turn off VSync, close and reopen the menu, and turn VSync back on, and this causes the process to speed up a lot until I close the menu, which then causes the whole LUT to reset to black. Even if I take a screenshot while the RetroArch menu is open, it’ll only be a screenshot of the game content plus any overlays like FPS or a notification of another screenshot.
After the 10 hours are over, use the F8 key (by default) on your keyboard to take a screenshot. It’s normal for the LUT to have some gaps where there’s no existing inverse for that RGB value.
Download this short Java program https://www.mediafire.com/file/mzs4t99f9zfnrwq/jav64.java/file and put it inside your screenshots directory. Inside jav64.java, change “2048-250109-123436.png” to the name of your screenshotted LUT. When you run this program, it should generate sonkun_fix_lut.png. Put it inside shaders_slang/crt/shaders/guest/advanced/grade, and feel free to rename the file if you’re doing this process multiple times.
In case you skipped this step, download my modded Grade shader as instructed earlier. Inside your modded sonkun preset, go back to this line:
"../../../../crt/shaders/guest/advanced/grade/passthru.slang"
and change it to this
"../../../../crt/shaders/guest/advanced/grade/pre-shaders-afterglow-grade-sonkunfix.slang"
and scroll down to the bottom where you see these textures:
textures = "SamplerLUT1;SamplerLUT2;SamplerLUT3;SamplerLUT4"
SamplerLUT1 = "../../../../crt/shaders/guest/advanced/lut/trinitron-lut.png"
SamplerLUT1_wrap_mode = "clamp_to_border"
SamplerLUT1_mipmap = "false"
SamplerLUT2 = "../../../../crt/shaders/guest/advanced/lut/inv-trinitron-lut.png"
SamplerLUT2_wrap_mode = "clamp_to_border"
SamplerLUT2_mipmap = "false"
SamplerLUT3 = "../../../../crt/shaders/guest/advanced/lut/nec-lut.png"
SamplerLUT3_wrap_mode = "clamp_to_border"
SamplerLUT3_mipmap = "false"
SamplerLUT4 = "../../../../crt/shaders/guest/advanced/lut/ntsc-lut.png"
SamplerLUT4_wrap_mode = "clamp_to_border"
SamplerLUT4_mipmap = "false"
Change it to this. If you’ve renamed your sonkun fix LUT file, make sure to reflect that here too.
textures = "SamplerLUT1;SamplerLUT2;SamplerLUT3;SamplerLUT4;ScalableSamplerLUT1"
SamplerLUT1 = "../../../../crt/shaders/guest/advanced/lut/trinitron-lut.png"
SamplerLUT1_wrap_mode = "clamp_to_border"
SamplerLUT1_mipmap = "false"
SamplerLUT2 = "../../../../crt/shaders/guest/advanced/lut/inv-trinitron-lut.png"
SamplerLUT2_wrap_mode = "clamp_to_border"
SamplerLUT2_mipmap = "false"
SamplerLUT3 = "../../../../crt/shaders/guest/advanced/lut/nec-lut.png"
SamplerLUT3_wrap_mode = "clamp_to_border"
SamplerLUT3_mipmap = "false"
SamplerLUT4 = "../../../../crt/shaders/guest/advanced/lut/ntsc-lut.png"
SamplerLUT4_wrap_mode = "clamp_to_border"
SamplerLUT4_mipmap = "false"
ScalableSamplerLUT1 = "../../../../crt/shaders/guest/advanced/grade/sonkun_fix_lut.png"
ScalableSamplerLUT1_wrap_mode = "clamp_to_border"
ScalableSamplerLUT1_mipmap = "false"
You should be able to load the preset in RetroArch, but it’s not quite ready yet. You still need to set certain settings. Read the “Color settings for NES games” and “Color settings for other consoles” sections above to do this correctly. However, there are two additional settings you’ll need to fix, because sonkun’s presets aren’t able to output darker or brighter than certain thresholds. You need to set “crt-guest-advanced color fix minimum” and “crt-guest-advanced color fix maximum” to force your colors to stay in-bounds, because out-of-bounds colors just become pitch black. For me, the values that worked were 1.0 and 240.0 respectively, but they might be different for you.