Well, I did it last year, sort of. Is not that I don’t like writing this type of post, because it is good to reflect on what happened in the year that ends, but I always have that feeling that I’m missing a lot of things. Anyway, let’s do some highlights at least.
Let’s start with gamedev: two games this year! Not bad for feeling a bit uninspired. Part of the success is perhaps because my 7 years old son want us –which is me– to make more games.
Alien Intruder (DOS 16-bit), that feels a bit too close to the end of the year, but to be fair it was almost finished for a while but I run out of steam in the last 10%.
Why I have been uninspired? I don’t know for sure. We have passed the 10 years mark of me releasing consistently at least one game every year (sometimes more!). I suspect I’m getting tired because it is a lot of effort. Not that much the actual work of programming, drawing the graphics, and writing the music; but the focus required to actually finish a more or less polished game, and release it –two different things, by the way–.
If found myself wondering if it was worth it, pushing forward for months to reach my own high standards, when it is a free game after all. Even if I do things for myself –and my sons, of course–, I believe it is human wanting to know that what you do kind of matters. There so much stuff out there, and could I be using my free time for other things? Absolutely.
I can hear you say: take a break! And you may be right, but I don’t want to go back to “not finishing things”. I suspect there is a risk of that if I stop.
As I mentioned recently, I would like to write more Haskell in 2025, and that would mean a bit less time for retro-gamedev. So that could be it: not that much a break as to do different things.
And that goes in hand with my old mini PC from 2015 (an Intel Core i3-5010U), dying on me. So I got a new one, this time a bit more powerful. Still not what you would consider a gaming rig, but is clearly not a potato like the old one, so… I can play modern-ish games now!
I suppose we can add that to the list of things I could be doing instead of spending all my nights working on games, isn’t it? Anyway, not a big factor, but I have enjoyed finishing Pyre –after I completed Transistor, I’m a big fan of Supergiant Games–. It is a beautiful game, with a great story, music and sound design. Very inspiring, even if in a way that I’m sure I can’t make games like that.
Because Lutris is awesome, we have been trying other games, but our big hit is Torchlight 2 “family co-op” in the local network. It is almost not age appropriate for the boys, but the kind of isometric perspective graphics with the fantasy theme makes it alright. We have completed it once, and we are having another go with different character classes. We are still playing some Minecraft, of course, but I prefer Torchlight 2!
The third big activity that has eaten a lot of my free time has been reading books. I’m still finishing about two books a month, although this year I’ve been reading authors that write very long books. Because I had a “not very good” book I wanted to read because it was the end of a series, and a “slow starter”; I didn’t manage to read 24 books this year. But 23 is still pretty good!
I have realised that finding what to read next is a big problem when reading two books a month. So when I discover a new author that I like, I tend to try other books by them, hoping that it would be as good. I did that last year, and it did work well, even if that meant reading a lot of Brandon Mull. This year I’ve been disappointed a few times, because it wasn’t as good –but it wasn’t bad either, so it is still a win–.
My highlights are the “Mistborn” series by Brandon Sanderson; probably the best fantasy I have read in many years to the point that I’m considering re-reading it in 2025. Another highlight, also by Sanderson, was “The Sunlit Man”. Only one book with almost no padding! Very recommended.
And that’s all. Would you believe that I’ve only been making games, playing some games, and reading books? Exactly, that is the feeling I was talking about!
So I finally released Alien Intruder yesterday, on Christmas day. Perhaps it should have happened a few weeks ago, but I was tired and couldn’t focus on writing the music. But that’s OK, it has been like a Christmas present –although I don’t think is the best date to release a game!–.
Although I sent a couple of emails to the usual people –what I call the press release–, the main “announcement” has been this post in Mastodon, and a few people said they liked the game, so it is being played. But you know, it is very unlikely that the news have reached to everybody that could be interested in an new DOS game in 2024. Any help spreading the word would be appreciated!
Checking some of the websites that cover DOS games beyond being a site to download “free” games, and at least check new titles, I was visiting DOS haven, and I liked they answer to why would someone make game for DOS nowadays?, which is essentially three reasons:
Because it’s fun, if you are into that sort of thing. Indeed.
Because it’s retro, and retro is cool! Perhaps I would say nostalgia plays a role here. Although I didn’t look at the DOS era like that, some people do.
Because thanks to DOSBox, you can run DOS programs almost anywhere: Windows, Linux, MacOS, smartphones, tablets, Raspberry Pi, you name it. Yes, this is a nice plus: games are as portable as DOSBox.
I’m not encouraging people to make DOS games as a for-profit adventure –there are plenty of limitations–, but playing the game on my RG35xx H is very nice and I didn’t have to port the game.
Anyway, I wanted to make a 16-bit DOS game, in the spirit of the shareware boom of early 90s. Although it isn’t shareware –and you don’t need to register the game!–, I was inspired by the feel of those titles –and I read Shareware heroes this year as well, an enjoyable book–. I know I have already released other DOS games, targeting both the age before and after, but I guess it doesn’t matter because it is 2024 anyway.
Making a game for the 286 with VGA graphics wasn’t perhaps the best idea, and I mentioned already how dificult was to write my own sound driver, but I enjoyed the process (it is fun, remember?). I may or may not make more DOS games, but after this experience I think that would be 32-bit –meaning 386 or better–. It is probably what makes more sense now that I have been there, done that.
As always, there’s a free download in the game’s website. I was approached by a publisher asking if I would like to make a big box release of the game. I’m not sure about this, but never say never. For now, just go an blast some cute aliens!
I streamed some sessions of game development with Haskell and I got to make a platform game with some reasonable functional code. I didn’t release a game because there wasn’t a game there, really; but despite that, I think I implemented enough pieces to be confident that I can finish a game in Haskell.
However, time flies when you are having fun, and my free time this year has gone mostly to The Heart of Salamanderland –released in May–, and my current project for DOS 16-bit “Alien Intruder” that should be ready in the next two weeks, if I manage to finish writing the music –seriously, it is hard to write decent enough tunes–.
And I haven’t written any more Haskell, despite revisiting the codebase a couple of times and missing writing Haskell. Why I miss it? I don’t know for sure, but besides being fun, I like how it makes me feel happy about writing code.
I don’t have much choice of languages when I’m making games for classic systems. If is a Z80 based machine, it is going to be assembler and C. If it is for DOS, it will be assembler and C. See the pattern?
For my tools I don’t think I’m going to stop using Python, because I am very productive with it. Being my tools, I don’t need to worry about packaging, distribution, or performance. So I started writing Python in 2010 and it still makes me happy in 2024, even if I’m not interested in using it everywhere like perhaps 10 years ago.
Then I write a lot of Scala professionally every day. Sometimes I even write it for fun, and I love it. But then, it runs on the JVM and I wanted something different, perhaps also functional like Scala, for my personal projects. Likely a compiled language that generates native binaries.
I put a significant amount of time in Go two years ago, and even dabbled a bit into Ebiten. I liked that after a while the language just disappeared, and what was left was the problem I was trying to solve. But I also found out that the language was very uninspiring, and some of Google’s practices put me off. So I moved on.
I refreshed my Javascript game a bit –but that wasn’t really what I was looking for–, I wrote some C with SDL2, and it was fun, but also too similar to what I’m doing when I’m making games for classic systems.
I even tried a Scheme, again, very inspired and motivated to like it. And it didn’t happen. Again. I’m just going to accept the fact that lisps aren’t for me and live a normal life.
But with Haskell was different. Writing the code made me happy. Evidently, it is a shame I didn’t have gameplay to support the code I wrote, but that is only my fault. “Alien Intruder” could be a PC game, and that would have been a nice project to write in Haskell. I know, like making a game is not challenge enough!
So that is one thing for next year: write more Haskell –hopefully with a PC game release!–.
More than a month since the last update? What happened!?
Well, “Alien Intruder” is progressing nicely: I have 25 stages ready –including playtesting–, and everything else is more or less in place.
The current TODO list is reasonable:
At least another tileset (two ideally), so we can get to 50 stages with some variability to the eye.
Likely another enemy, to make the stages 30 to 40 more interesting.
High score table, because this is an arcade game and chasing score is part of it.
The music, at least: the menu tune, in game, game over, and stage clear. I could also change the tune every 10 stages, but it depends on inspiration writing!
Some more testing!
So is not that much, is it? The engine has been finished for the last month and I have fixed a handful of bugs that necessarily would appear when designing and playtesting the stages, although the “problem” are usually those bugs that don’t present themselves as part of this process, and that’s why we need testing.
I still need to learn to use the Reality AdLib Tracker v2, although I went down a rabbit hole with the music player and I ended implementing a DRO player, so In theory I can use anything that generates AdLib audio and can be captured by DosBox.
There was also a lot of polishing, from improving the asset management –that looks like I almost reimplemented the WAD format–, better memory allocation and far pointer support –ah, isn’t DOS 16-bit fun?–, and small things like saving the joystick calibration to disk.
Anyway, I keep posting updates in my mastodon account –that, by the way, I have moved from SDF; but that is probably a story for a different day–.
My estimation on how long would it take me to complete my current project “Alien Intruder” was a bit off, and it is because I subestimated –or more like I forgot– that I had to write my own sound driver.
In Gold Mine Run! I used the excellent libmikmod, that gives you everything you need, supporting mixing of music and sound effects. Unfortunately for DOS it only supports DJGPP, which is 32-bit, and I’m targeting 16-bit with the IA-16 GCC port.
So I decided I would write my own driver, how hard could it be?
The plan was supporting:
FM for music, via AdLib compatibility.
1 channel 8-bit PCM sound via Sound Blaster.
And I have all that now, but it took me longer than expected because the docs I found were missing small details, and my own incompetence, of course.
Playing an IMF file, like the ones used in “Wolfenstein 3D” for example, is very easy. They are just an array of triplets (register, value and time) that you send to the sound card at an specific frequency, which was my first problem: I didn’t change the Programmable Interval Timer (PIT) chip from the default 18.2065Hz in Gold Mine Run!, and that was fine for MikMod, but not good enough for the FM playback.
So I had to do that, and it took me a bit to get it right because I had something else getting in the way and I though I was reprogramming the PIT wrong.
Which goes back to the default (1193182 / 18.2065 gives us a TIMER_SPEED of 65535).
With that out of the way, we use the interrupt handler to send data to the sound card (those “register and value” from the triplets I mentioned before). And that’s all, other that I need to learn to use a tracker that can export to this format. It will be OK, I’m sure of that.
Programming the Sound Blaster for PCM sound is not too complicated, but it has two parts: setup the DMA (the hard bits), and writing to the DSP (easy and well documented, for example Programming the Sound Blaster DSP).
To keep things simple I’m focusing on “plain” Sound Blaster, without using the new features of later revisions (e.g. the Sound Blaster 16 has other ways of setting up the DMA), and it should work fine with most configurations, but I kind of expect things to be a bit standard via the BLASTER environment variable. For most people playing the game with DosBox it will be fine, and I don’t need to support all the weird configurations out there.
There were two gotchas in this part:
the memory used by the DMA can’t cross a 64k page physical boundary.
for whatever reason, when I used a large buffer for the DMA(say 48K), I got pops and weird noise in some samples.
My first big mistake, that took me some time to fix, was that I’m using DOS’ int 21h service 48h to allocate memory, and that returns a segment. So if you ask for 64k, that won’t cross a page boundary, right? Well, that is not correct, the page you get is virtual.
The way to calculate and find the page boundary is using a 20-bit memory address. If the end of our buffer is on a different segment on that 20-bit address, we are crossing a boundary.
A simple way of dealing with this is allocating double the memory you need (for example 32K for a 16K buffer), and if the end of the first 16K are on a different segment, just use the second 16K that are guaranteed then to fit on a 64k page.
/* falloc is my own "far alloc" as I implemented far pointers */ dma_buffer = falloc(MAX_SAMPLE_LEN *2);
if (!dma_buffer)
return0;
dma_start = dma_buffer;
/* 20-bit address */ dma_linear = (dma_buffer >>16) <<4;
/* avoid crossing a page boundary */if ((dma_linear >>16) != ((dma_linear + MAX_SAMPLE_LEN) >>16))
{
dma_linear += MAX_SAMPLE_LEN;
dma_start += MAX_SAMPLE_LEN;
}
/* these won't change */ dma_page = (dma_linear >>16) &0xff;
dma_offs = dma_linear &0xffff;
I keep dma_buffer unaltered because I need it to free the memory with DOS later on, but my working buffer is really pointed by dma_start. Also the DMA works with the 20-bit address (also called linear), so I pre-calculate the values here.
After this is just matter of copying the sample I want to play (in my case MAX_SAMPLE_LEN is 16K), setup the DMA to use the buffer and the right size (depending on the sample), and ask the DSP so play it.
I implemented a simple priority based playback, like I do in my 8-bit games, so a sample plays only if there is nothing playing already, or the sample you want to play has higher priority than what is currently playing.
I track what I’m playing by setting a variable when I start playing a sample, and clearing it when the IRQ interrupt is triggered by the sound card when it has finished playing a sample.
With this priority table, the action sounds fine with only one channel and saving myself from doing mixing of multiple channels:
Initially I thought I could load all my samples in a 64K page –being careful with the size of the samples–, and play them from there; but for whatever reason, even if the memory is not crossing a page boundary, the playback isn’t right and there are unexpected pops or noises in real hardware (emultted) but not in DosBox (that is more permissive). I don’t know what is this, and I can only test with the excellent 86Box, so I decided to use a smaller buffer and just copy there the sample I want to play. After that, all plays and sounds as expected.
I’m assuming this has a performance hit, but so far the game plays nice on a 286 –as long as the VGA card is decent–, so I’m not concerned.
Anyway, I didn’t want to make this a tutorial, and is not; but I explained a bit what I have been doing. The game is progressing nicely, but I’m not going to suggest any release date this time!
I mentioned recently a couple of DOS game jams, and I’m working on a game that it was a good match for the first one because I’m targeting 8086 and VGA, but I couldn’t finish it on time. That’s OK, I thought, because I can submit it to the next jam.
Unfortunately, I don’t think I’m going to get to that one either –there are 4 days left until the deadline and the game is like 40% done–. I will finish the game anyway and release it, as I did with Gold Mine Run! last year. The game it even entered a DOS Game Jam compilation CD, although it was never submitted to the jam. Which is awesome, although there aren’t that many new DOS games released every year, so if your game is not a total disaster, you should get some attention –and players!–.
Anyway, let me introduce you to my current project: Alien Intruder.
There are still some unknowns, but I can already say the following:
For MS/DOS or compatible
Requires IBM PC/AT 8086 or later (a 286 is recommended), with VGA
Controlled with keyboard or 2 button joystick (2nd button for jump)
AdLib and/or SoundBlaster (undecided yet)
I am exploring some ideas, but it will be a “jump and shoot” type of arcade. I know this will fuel those saying that I am making too many arcade games, but they are obviously wrong :P or haven’t played my latest game: The Heart of Salamanderland, that is not an arcade.
The code is using all I learned making “Gold Mine Run!”, but also The Return of Traxtor. This is because my library for DJGPP requires a 386 and protected mode, but in this game I’m using 16-bit and the IA-16 GCC port, so the code I wrote for “Traxtor” is very relevant.
It took a bit of work, but I managed to simplify and port the VGA code to run in 8086 using a “small” memory model: independent segments for code and data, which means 64K of memory for each, and allocating 64000 extra bytes via DOS so I can have a back-buffer.
And the result is better than I expected! It even runs OK on very underpowered machines if we limit the number of entities. I tried for fun emulating a Sinclair PC200 using 86Box because it was the first PC my cousin had, and it was easier to boot with VGAcompared to other IBM PCs of the time –part lack of skills on my part, and part because I couldn’t find the right MS/DOS version to boot–.
Obviously the VGA is kind of expensive because each pixel is a byte, so I knew I couldn’t target those early models and make the type of game I was planing to make, but it was a fun test even if at the end a 286 is a more reasonable requirement. For most people I assume this won’t matter because it will run just fine in any modern machine with DosBox.
I can’t say for sure when it will be finished, but I am guessing early September is perfectly possible.
Last year I released two new DOS games (Gold mine run! and The Return of Traxtor), both made in the context of a game jam, even if I don’t really do game jams anymore.
Anyway, it was a lot of fun and it pushed me to get up to speed with the DOS platform, remembering in a way what I knew back in the late 90s, and going beyond that. And there are more DOS jams in 2024!
There could be other jams this year, like the DOS Games Jam –that had its first edition in 2020–, but these two jams tick a couple of boxes for me:
The target platform must be DOS (in other game jams is more about the feel, but I prefer going a bit more retro and make a game that runs on the actual hardware).
There are some interesting limitations, being 8086code in one of them (meaning: Intel 16-bit), and a COM file the other one (all the game must fit in 64K).
Since I released The Heart of Salamanderland for the Amstrad CPC last month I’ve been going through some ideas for potential games, and I was suffering a bit of choice paralysis and not starting anything (neither continuing one of the on hold projects).
When I released Gold Mine Run I put together a DOS library to make games with DJGPP, which is 32-bit and not useful for any of these two jams. But because I made Traxtor targeting the IBM PC/TX, I have some interesting code that I can reuse.
I’m toying with the idea of making something for the first jam, although I’m not sure what I’m going to target yet. I don’t feel like doing CGA again –too soon, and it is hard to draw nice things I guess–, EGA is such a pain to program, and VGA may not be a great match for the jam’s limitations.
Fusion Retro Books, besides of publishing very nice retro-computing inspired books, has been reviving some of the old magazines that were popular in the United Kingdom in the 80s. That is Crash for the ZX Spectrum, Zzap!64 for the Commodore 64 and Amtix for the Amstrad CPC.
If I’m not mistaken, it all started with annuals for the speccy and the Commodore, that were (are?) like a slightly larger magazine on a nice hardback edition that covers games released on the last year –with exceptions, they also cover new games that were released before the annuals were a thing–, and because we were living a retro-boom and the annuals were successful, Fusion got the rights for the magazines and started publishing them in a small A5 format with 60 pages.
Looks like, of the three, the Amstrad CPC is the smaller –or less active– community of users, so when Fusion was touching base trying to find if there was enough interest to sustain a Amtix revival, I was supportive of the idea –and excited!–. But as I recall it, the response from the community was lukewarm at best.
There were different reasons. For example, some people didn’t like the original Amtix, so I thought it was not going to happen. That’s why we can’t have nice things, etc. But it finally happened, and Fusion decided to give it a go with 12 numbers, and depending on the support, perhaps keep going.
We just got the 12th and last issue of Amtix CPC, and I’m not surprised.
I do think that the Amstrad community is sometimes weird, and not that different from other retro-communities in reality. We see that with new games, when people are very quick to ask for physical releases of the games, but very often the sales of those editions don’t justify the effort –and I’m not talking about profits but to just break even–, or even with people looking forward to play the game when is in development, but when the game is released… silence.
I find this frustrating, but what can be done? The ones wishing there were more things happening around the Amstrad CPC are not always the same that won’t support those initiatives, and being a smaller community perhaps than the ones behind the ZX Spectrum or the Commodore 64 (or the MSX!), means that we have less options (physical editions, magazines).
Although all this is true, in the case of Amtix, I think there are other reasons at play.
In my opinion the magazine was a bit hit and miss, and this is probably not unique to Amtix. I was subscribed to both Amtix and Crash, and by Amtix issue 7, I decided to cancel my subscription because I wasn’t enjoying reading the magazines.
What didn’t work for me? I guess it was a mix of different things: layout problems, some of the texts were very amateur, reviews of games that shouldn’t be there –a free game that is not good, why waste pages on it when there are more games to review?–, and in general it felt like they wanted to revive exactly what those magazines were in the 80s without realising that we live in a different time.
Yes, I know. But I said it already, the Amstrad community is sometimes weird, and I’m part of it.
Perhaps it could have been better, or different, but I’m glad we had Amtix back even if it wasn’t perfect. You can still get the magazine from Fusion Retro Books pages on Amtix CPC.
I announced here last year in December that I was working on a new Amstrad CPC game, although I had started the project in November. I’m still quite hesitant to announce a new game until I know I will finish it, despite doing this for almost ten years now.
The game is essentially what I planned: whip action fighting enemies, a good sized dungeon to navigate, and levers and keys to implement some basic puzzles. Which is always a bit of a refreshing surprise because very often the finished game is not quite what I had in mind when I started the project.
I took some risks on this one, by using an engine I wrote for performance, but that requires a lot of memory for a 64K game –essentially 16K for the hardware back buffer–, and that can always affect the size of the resulting game.
I put a lot of time into the encoding of the screens, using a new idea I had never implemented before based on meta-tiles of variable size –a bit more advanced than what I described here–, and at the end I managed to cram in 6 different enemy types with their own behaviour, a final boss, 55 screens and 4 music tunes –menu, in-game, game over and boss fight–.
In reality is not that you have 64K, it was in this case 32000 bytes of usable memory, and I finished the game having 162 bytes left. It was a tense when I had to fix a couple of last minutes bugs!
The game is framed in the world of fantasy of my 7 years old son, mixed a little bit with the universe of the Fablehaven novels by Brandon Mull, that happens to be mostly “compatible”.
You could argue that is not that important, because we all know that what it was written on the inlay of the 8-bit games from the 80s was just some filler that may or may not fit the actual game, but in this case it was an important source of inspiration when designing the enemies and the mood of the game. I don’t know if I succeeded, but my son is happy with the result, and that’s the gold standard for Salamanderland!
I made some decision regarding gameplay that I knew that could be controversial: tight jumps –I implemented “coyote time”, so it shouldn’t be that hard–, and classic “3 lives and game over”. It is an 8-bit game, these shouldn’t be a problem, isn’t it?
My hope is that players will persevere and get to enjoy the game for what it is. Of course there will be people that will get frustrated and quit, but that’s something that can always happen, no matter what game you make. If you can’t make everybody happy, be sure it is you that is satisfied with the result.
The game can be downloaded and played for free here: The Heart of Salamanderland, and a physical edition by Poly Play is planned for later this year –more information about that soon!–.
There is a delusion that “maintained” open source software comes with entitlements – an expectation that your questions, bug reports, and feature requests will be attended to in some fashion.
I couldn’t agree more, as I experienced that myself a few times. Including some people being a bit too pushy for my personal taste, so that I accepted patches and features because my project was upstream for them, and it is obviously better if I maintain those changes instead of them keeping track of my project as a soft fork.
Some of those changes, I accepted them. I recall for example scp support in a sftp proxy for OpenStack Object Storage, despite scp not being part of sftp really. Although I knew I was not going to use that feature, and I was very sure I didn’t want to maintain it.
The consequence is that it makes the maintainer less happy and, although I was employed by a company for which I was maintaining that open source project, it didn’t feel right and made me double-think if I really wanted to release more open source projects.
I recall reading a blog –can’t find the link– where the author said that they would never release anything that is useful, so they don’t get bullied into maintaining it after that. It sounds a bit extreme, but makes a lot of sense.
Matt includes in his post what he calls The Open Source Maintainer’s Manifesto:
I wrote the software in this repo for my own benefit – to solve the problems I had, when I had them. While I could have kept the software to myself, I instead released it publicly, under the terms of an open licence, with the hope that it might be useful to others, but with no guarantees of any kind. Thanks to the generosity of others, it costs me literally nothing for you to use, modify, and redistribute this project, so have at it!
Perhaps I don’t fully share the tone in which he wrote the whole piece, but that’s the gist of it: when I release something as open source, I see it as a gift to everybody, and the licence makes it crystal clear what anyone can expect from it. Yet, it is difficult.
I wasn’t happy with some of the contributions to my ubox MSX lib project, and didn’t accept them –which ended in a fork, that’s OK!–. Other changes, I accepted them, and I’m still unhappy about it because it wasn’t really why I released that code as open source after a lot of work to document and prepare the whole thing.
So it may seem I’m making things more complicated to get contributions now that I’m self-hosting the project out of a forge –it was in GitHub, then in GitLab, and now I self-host it–, but in reality I’m just keeping in the open something I made for myself and I’m sharing with the world.
But is not really a product, and I don’t want it to be. Thanks!