Skip to main content

Homebrew game - the worst bug ever


Exhibit A. (6502 assembly code)

Have you seen it yet?  This took me two sessions to find, God knows how many hours. 

The frustrating problem with this bug is that it caused memory to be corrupted elsewhere, memory that happened to be a part of the program. So the symptoms made no sense. 

The game would crash after losing the first life, during setup for the next life. Once I'd found the place where the crash was happening, it was a routine that had already worked properly once. What was going wrong when it ran the second time?

After pausing the emulator just before the crash and opening the monitor to examine the memory, I could see that my code at that particular spot was now corrupt. But that corruption could have happened at any time after initialising the game. No clue as to the culprit. 

I eventually started returning from the game at various points and using the monitor to check whether the corruption had happened yet. Eventually a pattern in the corruption itself gave a clue as to where it might be coming from. Even then, that loop looked fine. Often you see what you know should be there, rather than what actually is there.

Even if you've not spotted the problem, you may be asking why I needed to decrement X until beyond zero. Decrementing X or Y followed by a bne or beq (branch if equal to zero or not zero) is a very neat way to write a loop.  This loop needs X to count down to and including zero. I guess it's possible to rearrange things a couple of different ways so that you can still use X=0 and not need a compare. But this is the dirty way that I wrote it at the time because I tend to just bash it out willy nilly and tidy up later* 

I've now altered that loop so that it doesn't do that compare at all. 

In case you haven't spotted it, cmp is a 'compare A' not a 'compare X'.

The loop copied a block of data into a table using X as an index. So with the bug in place, the loop wasn't finishing and after 0 it was writing spurious data into memory 255 bytes away, which happened to be in the middle of real game code. 

Lessons learned: As well as taking the time to write more elegant code, I'll also be rearranging things so that all variables and tables are at the very end of my code. If a similar corruption or overflow happens, then it's likely to go into empty memory or into game data or tables. That will leave the game running and may yield more clues as to what's going wrong.

The game is going great now that it's back on track, thanks for asking.





* probably never tidy it up. 

Comments

Popular posts from this blog

ZX81 reversible internal 16k upgrade

T his post is an upvote for Tynemouth Software's  ZX81 reversible Internal 16K RAM upgrade . Their instructions are easy enough for even me to follow and don't involve cutting tracks. This is the ZX81 I've had out on display and used whenever I wanted to. It's an issue 1 and was probably a kit judging by some very untidy assembly. It has a ZX8-CCB  composite video mod and an external keyboard fitted. On board it has two 1k x 4-bit chips.  The ZX81 originally came with 1k on board. Thanks to a trick with compressing the display in ram, that was enough to type and run a small program but you soon felt the limitations. Back in the early 80s, the solution was a 16k ram pack which plugged into the back[1] and this is the way I've been using this particular machine. These ram packs are notorious for 'ram pack wobble'. Even if fastened into place, you can still randomly find your work disappearing. This is a very reliable solution using a more modern 32k chip (half

Driving NeoPixels with Z80

I 've long been thinking about a version two   RC2014 LED matrix module . I've had a matrix with a MAX 7219 on a module. It's a nice enhancement. But there's only so much you can do with a single-colour LED array right? Wouldn't it be cool to have RGB LEDs?  At Liverpool MakeFest I saw a wall-sized ping-pong ball NeoPixel display and picked up some NeoPixels with the intention of making one. Possibly driven by my RC2014.  I enjoy learning about protocols and have had some SPI devices working with the RC2014 - bit-banging SPI works really well because it doesn't care about timing. NeoPixels really do care about timing though. From Adafruit's web page about their 8x8  NeoPixel matrix: If there's one thing I want to get across in this blog post, it's don't just accept what you're told . Question everything. Learn about what's going on and find out why you're being told something isn't possible. Get creative with workarounds. I'

Making new ROMs for the Vic20 / Vicky Twenty

M y Vicky Twenty is very nearly complete.  As things stand, the board and every single component is new*. The processor and VIAs are newly-manufactured (W65C02 and W65C22).  Obviously the Vic1 chip isn't manufactured today, but there is 'new old' stock about. I have been able to buy a Vic 1, date code 1987 (which seems very late). It obviously hasn't been in a computer before, passes the acetone test and works. The same goes for two of the ROMs - character and BASIC. But I haven't been able to buy a new-old Kernal ROM (901486-07). I am able to borrow one - all of the boards I have, have this particular ROM socketed. I don't know whether all of this indicates that the Kernal has proved less reliable than the other two. I recently bought a TL866 for another project. Of all the retro-computing hardware things I've had to learn to do, making ROMs has been one of the simplest. So far, everything has been very easy and worked first time.  I'm not sure that it&