luchak [Lexaloffle Blog Feed]https://www.lexaloffle.com/bbs/?uid=24137 Filter Testbench <p> <table><tr><td> <a href="/bbs/?pid=151611#p"> <img src="/bbs/thumbs/pico64_picotron_filter_testbench-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=151611#p"> picotron_filter_testbench</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=151611#p"> [Click to Play]</a> </td></tr></table> </p> <p>This is a tool for visualizing the impulse response (top) and frequency magnitude response (bottom) of different filters. The frequency magnitude response is determined by taking the magnitude of the FFT of the impulse response.</p> <p>It's a pretty messy tool that's accumulated a few different features from various things I've been trying to do, but I want to refer to the cart in a Mastodon post so I'm posting it here. Ignore the blue and orange lines, those are from a different experiment; just focus on green and red.</p> <p>Basic usage: left/right changes cutoff, up/down changes resonance, Z/X rotate through filter modes. There are some modes labeled &quot;zep&quot; that are more relevant to the Picotron filter. The ladder filters work fine at 0 resonance, the zep modes need at least a tiny bit of resonance to produce a signal (so you'll need to hit the up arrow a bit to get them to work).</p> https://www.lexaloffle.com/bbs/?tid=143253 https://www.lexaloffle.com/bbs/?tid=143253 Sun, 21 Jul 2024 20:53:40 UTC weird transpose/matmul interaction <p>I was trying to compute the outer product of a column vector <code>x</code> with itself, like so:</p> <div> <div class=scrollable_with_touch style="width:100%; max-width:800px; overflow:auto; margin-bottom:12px"> <table style="width:100%" cellspacing=0 cellpadding=0> <tr><td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> <td background=/gfx/code_bg0.png> <div style="font-family : courier; color: #000000; display:absolute; padding-left:10px; padding-top:4px; padding-bottom:4px; "> <pre>xxt=x:matmul(x:transpose())</pre></div></td> <td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> </tr></table></div></div> <p>The resulting matrix had the right shape, but the only nonzero entries were on the diagonal. If I explicitly create a row vector and copy, then I get the correct result.</p> https://www.lexaloffle.com/bbs/?tid=142745 https://www.lexaloffle.com/bbs/?tid=142745 Tue, 18 Jun 2024 04:07:36 UTC Vectorized FFT <p> <table><tr><td> <a href="/bbs/?pid=150033#p"> <img src="/bbs/thumbs/pico64_vectorized_fft-1.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=150033#p"> vectorized_fft</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=150033#p"> [Click to Play]</a> </td></tr></table> </p> <p>(This is slower after 0.1.0h but still fairly quick overall - maybe I'll update these numbers eventually.)</p> <p>Here's some fairly fast FFT code for Picotron. It abuses the fact that <code>matmul()</code> is extremely cheap right now, and also decomposes shuffles into either matrix transposes or large-stride vector shuffles - so even if <code>matmul()</code> gets more expensive this should remain fairly efficient. A 256-point complex FFT currently takes about 1.4% CPU at 60fps.</p> <p>Hit Z or X to switch to the visualizer mode, which demonstrates feeding tracker output into the FFT to get frequency visualization. You could try taking the FFT code to use this with your own music!</p> <p>This works on complex signals only at the moment, but since it's quite fast and accepts separate real and imaginary vectors as inputs, using it for real signals shouldn't be too bad - just provide a zero imaginary part and ignore the upper half of the returned frequencies.</p> <p>General FFT code is in <code>fft.lua</code>, demo-specific code is in <code>main.lua</code>. I&rsquo;ll probably stick a MIT license or similar on the FFT code soonish - if you want to use it and I haven&rsquo;t done that by the time you read this (and the CC license won&rsquo;t work for you), get in touch and remind me!</p> https://www.lexaloffle.com/bbs/?tid=142736 https://www.lexaloffle.com/bbs/?tid=142736 Sun, 16 Jun 2024 21:57:25 UTC scalar-userdata subtraction commutes <p>Division has the same problem.</p> <div> <div class=scrollable_with_touch style="width:100%; max-width:800px; overflow:auto; margin-bottom:12px"> <table style="width:100%" cellspacing=0 cellpadding=0> <tr><td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> <td background=/gfx/code_bg0.png> <div style="font-family : courier; color: #000000; display:absolute; padding-left:10px; padding-top:4px; padding-bottom:4px; "> <pre>print((vec(1)-2)[0]) -- prints -1 print((2-vec(1))[0]) -- prints -1 print((vec(1)/2)[0]) -- prints 0.5 print((2/vec(1))[0]) -- prints 0.5</pre></div></td> <td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> </tr></table></div></div> https://www.lexaloffle.com/bbs/?tid=142723 https://www.lexaloffle.com/bbs/?tid=142723 Sat, 15 Jun 2024 19:24:50 UTC mouselock issues on Mac <p>I've seen this behavior in both sfx.p64 and #visitrack. I'm on a M1 Macbook Pro, OS X 14.3, running Picotron 0.1.0f.</p> <ol> <li>Slow drags downward don't do anything - <code>mouselock</code> just returns 0 for <code>dy</code>, regardless of how far you drag. Slow drags upward do not have this problem.</li> <li>Effective sensitivity is lower on Mac than on other platforms. I haven't verified this myself, but based on <a href="https://www.lexaloffle.com/bbs/?uid=5951"> @drakmaniso</a>'s description of how the sliders <em>should</em> be acting in #visitrack, this seems to be true.</li> </ol> https://www.lexaloffle.com/bbs/?tid=141785 https://www.lexaloffle.com/bbs/?tid=141785 Wed, 17 Apr 2024 17:02:00 UTC Picotron PCM: Yet Another Lazy Procedural Acid Cart <p>Now in stereo!</p> <p> <table><tr><td> <a href="/bbs/?pid=146405#p"> <img src="/bbs/thumbs/pico64_lazy_procedural_acid_picotron-3.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=146405#p"> lazy_procedural_acid_picotron</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=146405#p"> [Click to Play]</a> </td></tr></table> </p> <p>Just a short little proof-of-concept for streaming PCM output in Picotron. Could stand to be streamlined and cleaned up a fair bit (and commented better!), but hopefully this gets the idea across. The idea is to set up a few different sfx instruments playing different wavetables on adjacent rows of a pattern, use <code>stat()</code> calls to track the currently playing row, and update non-playing rows with new samples. There are a few gotchas to work around - setting up the instruments is a little fiddly, and Picotron crossfades instruments between rows so some samples need to be written to multiple instruments.</p> <p>Use the left and right arrows to change the audio buffer size. Values &gt;=8 seem to work well on web, and it looks like I can go as low as 3 on desktop Picotron on my machine.</p> <p>0.1.0h seems to have broken part of the visualization here - probably just an API change I haven't caught up with yet.</p> <p>IF YOU WANT TO USE PCM IN YOUR OWN CODE: copy pcm.lua from this cart, and check out how main.lua uses pcm_init, pcm_callback, pcm_start, and pcm_stop. I'll probably document this better at some point.</p> https://www.lexaloffle.com/bbs/?tid=141655 https://www.lexaloffle.com/bbs/?tid=141655 Thu, 11 Apr 2024 23:12:48 UTC Reappearing spam posts <p>The poster of <a href="https://www.lexaloffle.com/bbs/?tid=141370">https://www.lexaloffle.com/bbs/?tid=141370</a> seems to have discovered that posting a short reply to their own thread un-hides their original post after it was flagged for spam review. Not 100% sure this is a bug, but it seems like one.</p> https://www.lexaloffle.com/bbs/?tid=141371 https://www.lexaloffle.com/bbs/?tid=141371 Tue, 02 Apr 2024 02:34:28 UTC Userdata performance notes <p>Just a few notes from playing around with userdata. These notes assume 8M VM cycles/sec and large enough arrays to avoid substantial overhead and fully realize economies of scale.</p> <ul> <li>Fast ops - <code>add</code>/<code>mul</code>/<code>copy</code>/etc. - cost 1/16 cycle.</li> <li>Slow ops - <code>div</code>/<code>convert</code>/etc. - cost 1/4 cycle.</li> <li><code>matmul</code> is charged as a fast op based on the size of the output matrix. I'm a little suspicious that the answer seems to be so simple, so I'm wondering if I missed something.</li> <li><code>copy</code> and <code>memmap</code>/<code>memcpy</code> are approximately the same speed for 64 bit datatypes. For smaller datatypes, <code>memcpy</code> is proportionally faster, though of course you then have to manage strides/spans yourself. <code>memcpy</code> should also enable <code>reinterpret_cast</code> type shenanigans.</li> <li>There is substantial overhead for small spans. If you use spans of length 1 you pay 1/4 cycle/span, same as a slow op. It looks like this may be a flat cost per span, but I'm not sure. Using the full/strided forms of the ops does not seem to have noticeable additional costs beyond the per-span cost.</li> <li>For full-screen rendering, you have about 1 cycle/pixel at 480x270x60Hz. This includes whatever scaling/conversion you need to do at the end of the process. So realistically, you'll get in the neighborhood of 10 additions/multiplications per pixel. Exact numbers depend on whether you need a divide at the end, and whether or not you can work in <code>u8</code>.</li> <li>userdata flat access w/ locals seems to cost 1 cycle/element including the assignment.</li> <li>userdata <code>get</code> is 1/4 cycle / element at scale ... but each explicit assignment will cost you 1 cycle on top of this.</li> <li>userdata <code>set</code> is 1 cycle / element at scale.</li> </ul> <p>There also seems to be some interesting behavior happening where multivals, even very large multivals, do not seem to noticeably increase CPU usage when passed to <code>pack</code> or <code>set</code>. While I'm enjoying taking advantage of this for <code>f64</code> to <code>u8</code> conversions at the same cost as <code>convert</code>, I'm worried this might not last.</p> https://www.lexaloffle.com/bbs/?tid=141181 https://www.lexaloffle.com/bbs/?tid=141181 Thu, 28 Mar 2024 06:26:22 UTC Clock Life <p>Now much more screensaver-y!</p> <p> <table><tr><td> <a href="/bbs/?pid=144494#p"> <img src="/bbs/thumbs/pico64_clock_life-1.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=144494#p"> clock_life</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=144494#p"> [Click to Play]</a> </td></tr></table> </p> <p>After seeing some of the other Game of Life posts today, I thought I'd take a shot at the problem. This implementation runs at 480x270 / 60fps using a little over 50% CPU. Userdata ops are used to compute neighbor counts, and I initially tried using userdata ops to apply the changes to the grid as well - but that was just a <em>little</em> too slow (~90% CPU, so Picotron wouldn't run it full speed). This version uses color tables to apply the updates instead, which is much faster.</p> <p>Color tables seem incredibly powerful. I'm curious to see what else people do with them.</p> <h3>Changelog</h3> <ul> <li>0.2 - now the clock moves</li> </ul> https://www.lexaloffle.com/bbs/?tid=141098 https://www.lexaloffle.com/bbs/?tid=141098 Tue, 26 Mar 2024 04:31:54 UTC Wavepaper <p> <table><tr><td> <a href="/bbs/?pid=143925#p"> <img src="/bbs/thumbs/pico64_wavepaper-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=143925#p"> wavepaper</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=143925#p"> [Click to Play]</a> </td></tr></table> </p> <p>Please enjoy this very necessary desktop wallpaper.</p> <p>This is 0.1 and not 1.0 because performance is a major issue: this will run at 60fps if you have no windows open, but will drop to 30fps or possibly lower if anything at all is running on top of it. When that happens, (WARNING) <em>you may get some flickering</em>, especially if some windows overlap desktop icons.</p> <p>Despite the fact that this is basically just a reskin of the other wave sim I posted - using a different rendering effect to keep it compatible with the default palette - I still think it's kind of neat. Hopefully I'll find ways to get this running more smoothly over time. Or, failing that, maybe just ways to pause it if anything else is open.</p> <p>To install:</p> <div> <div class=scrollable_with_touch style="width:100%; max-width:800px; overflow:auto; margin-bottom:12px"> <table style="width:100%" cellspacing=0 cellpadding=0> <tr><td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> <td background=/gfx/code_bg0.png> <div style="font-family : courier; color: #000000; display:absolute; padding-left:10px; padding-top:4px; padding-bottom:4px; "> <pre>load #wavepaper save /appdata/system/wallpapers/wavepaper.p64.png</pre></div></td> <td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> </tr></table></div></div> https://www.lexaloffle.com/bbs/?tid=140928 https://www.lexaloffle.com/bbs/?tid=140928 Wed, 20 Mar 2024 03:46:47 UTC Pico Ripples <p> <table><tr><td> <a href="/bbs/?pid=143450#p"> <img src="/bbs/thumbs/pico64_pico_ripples-1.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=143450#p"> pico_ripples</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=143450#p"> [Click to Play]</a> </td></tr></table> </p> <p>A nice little raindrops-on-a-pond type of effect. Though perhaps with more of a shifting rainbow color scheme than most ponds.</p> <h3>Changelog</h3> <p>v1.01: Fixed initial splash height so smaller ripples should be more visible. This, to my eye, adds some pleasing visual variety.</p> https://www.lexaloffle.com/bbs/?tid=140778 https://www.lexaloffle.com/bbs/?tid=140778 Sun, 17 Mar 2024 06:42:27 UTC Picotron Smoke Sim <p> <table><tr><td> <a href="/bbs/?pid=143262#p"> <img src="/bbs/thumbs/pico64_picotron_smoke-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=143262#p"> picotron_smoke</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=143262#p"> [Click to Play]</a> </td></tr></table> </p> <p>Yup, it's a smoke simulation toy for Picotron! Use your mouse to play. Should run at a nice steady 60fps after the first ~1s.</p> <p>Uses userdata to make pressure projection steps almost free. That's great, since those steps were super-expensive in the PICO-8 version! Unfortunately, advection still has to be done with loops in userspace Lua code, and so that part didn't really see any speed increases, and is the new bottleneck.</p> <p>There might still be some rendering tricks to speed this up, and I need to see whether some tricks I gave up on in the PICO-8 sim, like a MAC grid or vorticity confinement, might be viable here. Unfortunately higher-order interpolation, which I know would help a lot, is probably not happening without major new userdata features.</p> https://www.lexaloffle.com/bbs/?tid=140717 https://www.lexaloffle.com/bbs/?tid=140717 Sat, 16 Mar 2024 05:20:19 UTC Tunnel effect w/ userdata <p> <table><tr><td> <a href="/bbs/?pid=143044#p"> <img src="/bbs/thumbs/pico64_userdata_tunnel-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=143044#p"> userdata_tunnel</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=143044#p"> [Click to Play]</a> </td></tr></table> </p> <p>Here's a simple tunnel effect I did to test out doing pixel effects with userdata. Not super inspired, but it does run at 60fps at full resolution!</p> https://www.lexaloffle.com/bbs/?tid=140665 https://www.lexaloffle.com/bbs/?tid=140665 Fri, 15 Mar 2024 02:39:43 UTC userdata methods and type conversions <p>I've been thinking about ways to execute fast pixel effects on Picotron, and so I'm looking for ways to perform the effect entirely in userdata-land, and display the results as sprites. I've tried a few approaches, none of which seems to be quite ideal.</p> <ol> <li> <p>Do math on <code>f64</code> userdata, then render that to the screen. This gets me fast math operations, but of course the rendering result is completely garbled. I suppose some pack-ints-into-floats type nonsense could be attempted? But with just basic arithmetic ops this seems like a bad time.</p> </li> <li> <p>Do math on <code>f64</code> userdata, then cast that to <code>u8</code> in a loop. This is a <em>lot</em> of time spend calling getters/setters in a loop, and may not be any better than <code>pset</code>, not sure.</p> </li> <li>Do math on integer userdata, then cast to <code>u8</code> using the <code>userdata:convert()</code> method and display that. This would work fine if I could figure out how to get fast fixed-point going. Unfortunately, div and mod are slow, and shifts don't seem to work.</li> </ol> <p>I think my preferred resolution would be to allow <code>convert()</code> to go from <code>f64</code> to <code>u8</code>, but anything that allows any kind of math with fractional quantities, either floating-point or fixed-point, would be great. Drawing <code>f64</code> userdata with <code>spr</code> with implicit casting would be great too. Just trying to avoid explicit loops to get to <code>u8</code>....</p> <p>Or perhaps there's some other way to get what I want with what's in Picotron already and I've just missed it. Not quite sure what's doable with the 10-15ish ops/pixel this would get best case but I'm sure something neat can happen.</p> <p>Other requests:</p> <ul> <li> <p>I would love to have min/max/abs or similar &quot;sharp&quot; functions available for userdata, to allow masking off out-of-range data or other similar conditional-ish use cases.</p> </li> <li> <p>trig would be amazing but I assume that's probably off-limits, or at least would be quite expensive.</p> </li> <li>Even more amazing would be the ability to index userdata with userdata (and at this point built-in trig is probably not necessary)</li> </ul> https://www.lexaloffle.com/bbs/?tid=140654 https://www.lexaloffle.com/bbs/?tid=140654 Thu, 14 Mar 2024 23:31:55 UTC (outdated) instrument editor confusion <p>I'm trying to make a classic synth pluck sound: a couple of detuned saw oscillators, plus a resonant lowpass filter that quickly opens up when a note starts, then closes back down. Unfortunately, I can't quite figure it out. I have an ADSR envelope modulating the filter cutoff, but since the envelope has a positive polarity and higher lowpass knob values = lower frequencies, the effect is exactly the opposite of what I want: the filter immediately closes down on note onset, then more slowly opens back up.</p> <p>Is there any way to flip the sign of this modulation? Am I thinking about this the wrong way?</p> https://www.lexaloffle.com/bbs/?tid=140648 https://www.lexaloffle.com/bbs/?tid=140648 Thu, 14 Mar 2024 20:25:34 UTC Wave Designer <p>This is a tool to allow both time-domain and frequency-domain editing of the new custom sfx waveforms.</p> <p> <table><tr><td> <a href="/bbs/?pid=142178#p"> <img src="/bbs/thumbs/pico8_wave_designer-5.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=142178#p"> wave_designer</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=142178#p"> [Click to Play]</a> </td></tr></table> </p> <p><strong>Note for existing users:</strong> use the buttons at the top-right to import/export to clipboard, it doesn't happen automatically on every change now.</p> <p>Controls:</p> <ul> <li>Mouse: click and drag to edit the wave directly (green line), harmonic magnitudes (blue bars in the middle), or harmonic phases (maroon/lavender bars on the bottom). Or click the play button to toggle preview playback, which plays sfx 1 on loop. The waveform is stored in sfx 0.</li> <li>Left/right arrow: switch between a bunch of preset waves.</li> <li>Tracker keys: play notes!</li> </ul> <p>Buttons:</p> <ul> <li>Play/pause</li> <li>Undo and redo</li> <li>Set phase to 0 or 0.25 (<code>sin</code> and <code>cos</code> buttons)</li> <li>Keep odd harmonics only (<code>odd</code> button)</li> </ul> <p>Features:</p> <ul> <li>Time and frequency domain editing, always kept in sync.</li> <li>Mouse drags are interpolated so fast drags work as expected.</li> <li>Import/export your creation to the clipboard as a PICO-8 sfx string, using the buttons at the upper right. You may need to press ctrl-c/cmd-c on the web for this to work; on desktop it should be automatic.</li> <li>You should be able to paste PICO-8 format sfx strings into the cart to load waves (including waves you've designed in this tool).</li> <li>NOTE: if you are using the web version on a Mac, you will have to press some combination of <em>both</em> cmd-C and ctrl-C to copy, and <em>both</em> cmd-V and ctrl-V in order to paste, since the web version is really really finicky about how copy/paste works. You may get a stuck note when this happens.</li> </ul> <p>Limitations:</p> <ul> <li>If you want to change the preview sound (sfx 1) you have to edit it in the tracker. You can replace sfx 1 with some other looping sfx if you want, it just needs to use wavetable instrument 0.</li> </ul> <p>Feedback is welcome!</p> https://www.lexaloffle.com/bbs/?tid=140438 https://www.lexaloffle.com/bbs/?tid=140438 Thu, 29 Feb 2024 04:59:40 UTC sfxp - sfx design tool and generator <p>Want sfx fast? sfxp can help! Adjust the sliders for new sounds, or click &quot;random&quot; for instant inspiration.</p> <p> <table><tr><td> <a href="/bbs/?pid=134424#p"> <img src="/bbs/thumbs/pico8_sfxp-1.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=134424#p"> sfxp</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=134424#p"> [Click to Play]</a> </td></tr></table> </p> <p>Controls:</p> <ul> <li>click/drag sliders and buttons</li> <li>right click the random button to mutate the current sound instead of fully randomizing it</li> <li>z/x to replay the current sound</li> <li>left arrow to undo</li> <li>right arrow to redo</li> </ul> <p>You can export either to sfxp format (to share, or to edit more later) or to PICO-8 sfx format (to paste into the PICO-8 sfx editor and use in your PICO-8 games). Export works via the clipboard, which can be kind of weird on the web version of PICO-8. I highly recommend running sfxp on desktop PICO-8 if you have it.</p> <p>You can also find <a href="https://luchak.itch.io/sfxp">sfxp on Itch</a>.</p> <p>Made for the PICO-1K Jam.</p> <p>Update 2023-09-20: undo bug fixed and redo added.</p> https://www.lexaloffle.com/bbs/?tid=54142 https://www.lexaloffle.com/bbs/?tid=54142 Sat, 16 Sep 2023 08:24:00 UTC QPA: Lossy Audio Compression for PICO-8 <p>I am pleased to announce Questionable PICO-8 Audio (QPA), a compressed audio format for PICO-8.</p> <p>(Compatibility note: if you encoded any QPA files before 2023-09-07, please see the Changes section below.)</p> <h2>DON'T YOU LECTURE ME WITH YOUR 30K PICO-8 CART</h2> <p>Oh, but I will!</p> <p> <table><tr><td> <a href="/bbs/?pid=133755#p"> <img src="/bbs/thumbs/pico8_qpa_demo-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=133755#p"> qpa_demo</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=133755#p"> [Click to Play]</a> </td></tr></table> </p> <p>Your song is synced with the clipboard - you can copy your songs out to text files (or this thread!) and paste song strings back in to the cart. You can right-click samples to preview them, and you can right-click the play button to skip the intro sample. Numeric modifiers come <em>before</em> the thing they modify. Hopefully you can figure out the rest!</p> <p>Copy/paste is a little annoying on the web player, and PCM audio can also be unreliable on the web, so consider trying this cart on desktop PICO-8 if you have it.</p> <p>You can also check out this song by me:</p> <div> <div class=scrollable_with_touch style="width:100%; max-width:800px; overflow:auto; margin-bottom:12px"> <table style="width:100%" cellspacing=0 cellpadding=0> <tr><td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> <td background=/gfx/code_bg0.png> <div style="font-family : courier; color: #000000; display:absolute; padding-left:10px; padding-top:4px; padding-bottom:4px; "> <pre>7⬆️✽0⬆️✽✽✽⬇️✽⬆️✽3⬇️▶▤웃⬆️7◀0⬆️ˇˇ♪♪♪□&and;7⬆️&and;7⬇️7⬇️&and;</pre></div></td> <td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> </tr></table></div></div> <p>Or this much better song by <a href="https://www.lexaloffle.com/bbs/?uid=40166"> <a href="https://www.lexaloffle.com/bbs/?uid=40166"> @packbat</a></a>:</p> <div> <div class=scrollable_with_touch style="width:100%; max-width:800px; overflow:auto; margin-bottom:12px"> <table style="width:100%" cellspacing=0 cellpadding=0> <tr><td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> <td background=/gfx/code_bg0.png> <div style="font-family : courier; color: #000000; display:absolute; padding-left:10px; padding-top:4px; padding-bottom:4px; "> <pre>➡️▶▒7◀▶&hellip;6◀░▶▒7◀▶&hellip;6◀★▶▒7◀▶&hellip;6◀█▶😐7◀▶⁘3◀⬆️▶⁘3◀⬇️▶▒7◀▶&hellip;6◀░▶▒7◀▶&hellip;6◀★▶▒7◀▶&hellip;6◀⬆️★⬇️⬇️⬇️⬇️▶★7◀▶▥▥⬆️2◀&and;</pre></div></td> <td background=/gfx/code_bg1.png width=16><div style="width:16px;display:block"></div></td> </tr></table></div></div> <p>(<a href="https://thirtydollar.website/">The reference</a>, if you're not familiar)</p> <h2>Why QPA?</h2> <p>Because you want to put fun sound samples in your carts, but don't want to use too many tokens or too much space!</p> <p>QPA provides three main benefits:</p> <ul> <li>Tiny decoder. A fully-featured decoder that reads from strings, does quality-level detection, and rejects invalid files is 228 tokens. A minimal decoder is just 175 tokens.</li> <li>Reasonable quality at very low data rates. QPA encodes mostly-intelligible speech at just 1.14 bits/sample (788 bytes/second) - or completely-intelligible speech, as well as usable instrument samples, at twice that rate.</li> <li>Faster-than-realtime decoding speed. Precise speed depends on the decoder implementation, but around 4x realtime is a reasonable expectation for a token-efficient decoder.</li> </ul> <p>As for how this works in practice: the demo cart PNG is just under 32k. It contains about 18k of samples (split 17k in sprite/map/etc. space, 1k in code) and 2k of other compressed code; the rest of the space is taken up by the label and other overhead. That 18k of sample data stores almost 14 seconds of audio. The samples are stored at a variety of quality levels, but more audio could have been included at lower quality, or higher-quality audio could have been used in exchange for more space. There also could have been a lot more audio if I hadn't tried to keep the cart around 30k - there's another 12k of compressed code space that could have been filled with QPA data strings!</p> <h2>What Is QPA?</h2> <p>QPA is an adaptation of the Quite OK Audio Format (QOA) to PICO-8's constraints. As such, it is a lossy, constant-bitrate format, in which the decoder attempts to predict future sample values one at a time, and compressed files store approximate differences between these predictions and the real sample values. To better understand how this scheme works, I suggest reading <a href="https://phoboslab.org/log/2023/02/qoa-time-domain-audio-compression">this introduction to QOA</a>. QPA changes some details, but the basic structure remains the same.</p> <p>So what changed to put the PICO in QPA?</p> <ul> <li><strong>Multiple quality levels.</strong> QPA offers a choice of compression ratios so you can optimize your valuable cart space. The highest compression ratio (7x) is noisy, but is still suitable for some speech, percussion, and sound effects. The lowest two compression ratios (2.25x and 1.75x) are approximately transparent for most sounds.</li> <li><strong>Minimal metadata.</strong> The only pieces of metadata in a QPA file are a magic number and a sample count. QPA includes no sample rate information, and does not support multi-channel audio (PICO-8 is mono only) or seeking to arbitrary points in a file (PICO-8 carts are small). This makes the decoder simpler.</li> <li><strong>8-bit audio only.</strong> QPA assumes an 8-bit output resolution, and is tuned to avoid single-bit output noise.</li> <li><strong>PICO-8 numbers.</strong> Almost all arithmetic works on PICO-8's native q16.16 number representation, slices are now 32 bits to match PICO-8's word size, and the format is little-endian instead of big-endian.</li> <li><strong>Smarter encoder.</strong> At PICO-8's low sample rate, it is critical to avoid high-frequency noise, so the encoder includes some light perceptual tuning to push noise into lower frequencies. It also expands QOA's use of exhaustive search for parameter selection, making it easier to incorporate new encoding strategies in the future.</li> </ul> <h2>Resources</h2> <p>To use QPA in your own carts, please check out the following resources:</p> <ul> <li><a href="https://www.lexaloffle.com/bbs/?uid=49583"> <a href="https://www.lexaloffle.com/bbs/?uid=49583"> @bikibird</a></a>'s <a href="https://bikibird.itch.io/defy">Defy encoder</a> and <a href="https://www.lexaloffle.com/bbs/?tid=47089">Defy cart</a>. You can use these to encode and play back your own audio files.</li> <li>The <a href="https://raw.githubusercontent.com/luchak/qpa-format/main/pico8/qpa_util.p8">QPA utility cart</a>. This cart makes it easy to convert QPA files to strings, and contains low-token decoding functions to copy into your own carts.</li> <li>The <a href="https://github.com/luchak/qpa-format/">QPA repository on Github</a>. Here you can find more documentation, a format spec, and a CLI encoder/decoder for QPA files. (CLI requires Node and NPM.)</li> </ul> <h2>Changes</h2> <p>On 2023-09-06 and 2023-09-07, new versions of all tools, code, and encoders were released to fix problems identified by Packbat. These new versions broke compatibility with earlier versions - if you switch to the new code, please keep in mind that you will have to re-encode your files. Hopefully this is a one-time only sort of change.</p> <h2>Future Work</h2> <p>I'd like to focus on getting better quality at approximately 1 bit/sample. It's possible that VBR or variable sample rate enhancements to QPA could get there, or it might be necessary to explore frequency-domain techniques ... although that might bode poorly for keeping decoder token count low.</p> <p>But first - I think it's time to use QPA to build a tracker. No promises on release date!</p> <h2>Acknowledgements</h2> <ul> <li>Dominic Szablewski for the QOA audio format</li> <li><a href="https://www.lexaloffle.com/bbs/?uid=40166"> <a href="https://www.lexaloffle.com/bbs/?uid=40166"> @packbat</a></a> for discussions and a whole lot of listening tests</li> <li><a href="https://www.lexaloffle.com/bbs/?uid=49583"> <a href="https://www.lexaloffle.com/bbs/?uid=49583"> @bikibird</a></a> for Defy and being willing to integrate QPA</li> <li>ferris for providing design feedback and pointers for future work</li> <li>MissMouse for the demo cart suggestion</li> <li><a href="https://www.lexaloffle.com/bbs/?uid=25898"> @NerdyTeachers</a> and <a href="https://www.lexaloffle.com/bbs/?uid=66501"> @Heracleum</a> for help with the cookie/coffee test</li> <li>The entire #music-sfx channel on the Discord for discussions and for putting up with a whole lot of QPA chatter</li> </ul> https://www.lexaloffle.com/bbs/?tid=53933 https://www.lexaloffle.com/bbs/?tid=53933 Thu, 31 Aug 2023 01:00:55 UTC Sample Piano (and also a loop slicer) <p>Would you like to play the piano ... in PICO-8?</p> <p> <table><tr><td> <a href="/bbs/?pid=132750#p"> <img src="/bbs/thumbs/pico8_sample_piano-0.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=132750#p"> sample_piano</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=132750#p"> [Click to Play]</a> </td></tr></table> </p> <p>Well, I can't imagine why, but there you go. Use the tracker keys to play. It's polyphonic.</p> <p><del>If you want to change the sample, you can convert a .WAV file to any of the QPA formats - and <em>only</em> the QPA formats - supported by this WIP tool: <a href="https://luchak.github.io/defy/">https://luchak.github.io/defy/</a> . Then drag and drop the resulting file on the running cart and play away. This will also save the new sample to the cart using <code>cstore()</code> so you can make new carts for new instruments!</del> That URL has been updated with some format improvements, so it won't work here any more until I update these carts.</p> <p>Usual caveats apply about PCM output not always working well on the web.</p> <p>For this use case there's basically no reason not to use 3.2 bit QPA, the highest QPA quality, since the file will be truncated at ~32k samples (about 6 seconds) due to table indexing limits and my laziness, and at that length 3.2 bits fits easily in a cart. But for future use cases perhaps the 2.3 bit format will have a purpose, and maybe even the 1.1 bit...</p> <p>Yes, the WIP conversion tool is a fork of <a href="https://www.lexaloffle.com/bbs/?uid=49583"> @bikibird</a>'s Defy. There's a PR out to merge it into the official version!</p> <p>And if you're curious, QPA is closely derived from <a href="https://qoaformat.org/">https://qoaformat.org/</a> and currently stands for Questionable PICO-8 Audio. If you think it might stand for something else let me know, you could be correct.</p> https://www.lexaloffle.com/bbs/?tid=53646 https://www.lexaloffle.com/bbs/?tid=53646 Sat, 05 Aug 2023 05:43:31 UTC bbforth <p>This cart is a barebones Forth implementation in 279 tokens. It could be smaller, it could be more usable, but I haven't really touched it in weeks, so here you go. Probably full of bugs.</p> <h3>Features</h3> <ul> <li><code>eval()</code> function</li> <li>Interpret and compile modes - extend syntax in Forth!</li> <li><code>if</code>/<code>then</code> (implemented in Forth)</li> <li>A few useful comments, including some commented-out utility functions and basic smoke test.</li> </ul> <h3>Non-features / shortcomings</h3> <ul> <li><code>else</code> (you can add this in Forth)</li> <li>Arithmetic (you can add this for 11 tokens per binop, or 9 tokens if you're willing to use valid Lua identifiers as operator names)</li> <li>Looping (you can probably add this for the cost of a few arithmetic/comparison operators, then add the syntax in Forth for no token cost)</li> <li>Passing args or returning values when calling Lua functions from Forth (depending on how fancy you want to get, probably takes 20-50 tokens for a reasonable wrapper)</li> <li><code>[</code> and <code>]</code> are spelled <code>_lb</code> and <code>_rb</code> for some reason.</li> <li>The inner interpreter works in a non-standard way that will make it hard to call arbitrary Forth from Lua outside of <code>eval</code> - this is probably worth rethinking.</li> </ul> <h3>A few ways to make it smaller</h3> <ul> <li>A few functions could probably be moved from Lua to the initial Forth definitions as-is, although since <code>:</code> and <code>;</code> depend on them this will be a bit messy.</li> <li>With the right Forth/Lua binding util, even more functions could probably be moved.</li> <li>With some internal representation changes, even <code>_interp</code> and <code>eval</code> could probably be (mostly) boostrapped.</li> </ul> <p>Anyway, enjoy. This may not be useful for very much, but the question of how little material you need to bootstrap a reasonable eval is still kind of a fun question.</p> <p> <table><tr><td> <a href="/bbs/?pid=131852#p"> <img src="/bbs/thumbs/pico8_bbforth-1.png" style="height:256px"></a> </td><td width=10></td><td valign=top> <a href="/bbs/?pid=131852#p"> bbforth</a><br><br> by <a href="/bbs/?uid=24137"> luchak</a> <br><br><br> <a href="/bbs/?pid=131852#p"> [Click to Play]</a> </td></tr></table> </p> https://www.lexaloffle.com/bbs/?tid=53375 https://www.lexaloffle.com/bbs/?tid=53375 Tue, 11 Jul 2023 21:17:31 UTC