Ultimate Amiga
Games Corner => The Crypt of Bloodwych => Bloodwych Editors and Modifications => Topic started by: Hungry Horace on June 03, 2007, 07:02:31 PM
-
just a little place to write any hacking findings.... i guess largly for myself, but maybe someone else will help find some more.
some of the basics are at: http://www.djcresswell.com/bloodwych
but of course, everything here will eventually be part of the main site, under the hacking resource
-
some memory locations.. from my real notepad... just here as a result of doing the avatars;
BW:
player 1 current postion ... EE99
XX .. YY .. ZZ
(x coord, 00, y coord, 00, direction coord)
player 1 object held: EEA9
player 2 object held: EF0B
xx .. yy
(object number, 00 , quantity)
EB2A - Bloodwyn Stats (2 lines) [$EBD0]
ED2A - Bloodwyn Pocket contents
Monsters; ~ 1700E
note, whilst shape "$65/66" represents a Summon/fat-bastard... to make him an "illusion" set Level=$80 or above.
BEXT:
F586 - Bloodwyn Stats - 4 lines in BEXT, followed by other 15 champions or recruited monster.
~153B6 - monsters unpacked (first created summon if no enemies killed)
-
Using an action replay 3 the keyboard routine starts at $5D6
The game reads the keyboard and stores the result at memory location $5D1 where it is read in later
I'm still checking so stick with me ;)
-
Keyboard routines
thanks for this info to Bip.....
<BippyM> it loads $bfe001 into reg a0
<BippyM> it then does a move.b ($c00,a0),d0
<BippyM> which basically does move.b $c00+$bfe001,d0
<BippyM> which is move.b $bfec01,d0
<BippyM> which moves the raw keycode into d0 ;)
<BippyM> sneaky eh!
i.e. searching $bf$eo$01 can help
-
these more recent tests have been done with the _102 IPF of bloodwych.
the keycodes (http://www.whdload.de/docs/en/rawkey.html) which are checked
$2ec in the binary for _102 the ingame controls
$2C6 is the loction on the modified binary i have
these can be found manually by searching:
$5F 46 4E 4F 4D 4C
$12 10 22 20 21 11
which are at $0656 in the _102 ipf when loaded.
for those who understand ASM, this is apparently the keyboard routine at $5E8
<BippyM> lea $656,a0 ; Copy address into a0
<BippyM> moveq #b,d1 ; For loop, checking 12 things
<BippyM> cmp.b (a0)+,d0 ; check to see if the contents of mem location a0 are same as d0 and increment the address at a0 by 1 byte (first time a0 would change to $657)
<BippyM> beq $063a ; If they match lets branch
<BippyM> dbf d1,$630 ; decrease d1 by 1 and loop
<BippyM> rts ; return from routine
-
Here you go master Zendick ;D
Serpent Tower pad/trigger details.
-
Nice one skate
doesnt lok like the switches investigation is really needed... most of it seems quite obvious (although i want to check a few)
http://bloodwych.condor.serverpro3.com/tp-downloads/list_of_switches.txt
moon tower is needed for the traps however....
http://bloodwych.condor.serverpro3.com/tp-downloads/list_of_triggers.txt
-
Here ya go.
-
I see you did most of them :P . Couldn't bare the thought of more questions aye? ;)
-
feel free to do the switches list :)
-
skate, try this:
[EDIT] LINK REMOVED
keys here:
In all:
F1-F8 to select floor
F10 to change edit mode
< to move down a tower / dungeon
> to move up a tower / dungeon
space-bar to enter the edit options
cursors up/down to change option
cursors left/right to change option value
S to save the game binary (screen will flash red)
in map editor:
backspace to clear space
c to copy space to clipboard
v to paste clipboard values to map
(edit options only work for items 2,3,4 and 5 - wood walls, misc, stairs and large doors)
In level layout editor only:
[ to switch toggle floor-below preview
] to switch toggle floor-above preview
-
this thread should become redundant as the "confirmed" info is transferred to the main hacking thread:
http://www.ultimateamiga.co.uk/index.php/topic,8908.0.html
-
Hi there,
I'm currently reverse engineering the bloodwych source and found your findings very handy!
I loved this game back when it came out and am currently re-living the days using STEEM and an Atari ST disk image.
So, I've been using Visual Studio and parsing out the "CODE" source file. Although the addresses are different, the data follows the same structure and byte patterns as you describe.
I've managed to create a small app to extract and render the map data, which I'm refining as I play through the game. It's definitely not as advanced as your editor, but I'm having fun digging into how it all works.
Now I've got the map data, my next steps are the character and object data, then trying to rip the images from the source, so I can convert to PNG or something similar.
More than happy to share c# code snippets to explain my parsing if that is handy for you!
e.g. using your ab cd format for map cells...
Metal Door Cell : d==5
OpenState = (b%2 == 0x0) ? EnumOpenState.Open : EnumOpenState.Closed;
Type = (b%4 <= 0x1) ? EnumType.Solid : EnumType.Grating;
Alignment = (b%8 <= 0x3) ? EnumAlignment.EastWest : EnumAlignment.NorthSouth;
Cheers,
Murlock
-
hi murlock,
really glad to see this is being found useful!!
I have a whole load of info which is in my editor which I havent yet put into the main threads. If there is any in particular you want me to add, just say - I'm far more likely to find the motivation to add the info if I think someone else is using it!!
I had looked at the ST version before, and was even considering making my Amiga editor work with both the ST and PC versions (it's already capable of spitting out .obj and .map files for the PC version) because all of the core data is identical, it's just a matter of finding it!!
I would also like to transfer the maps from the ST demo over to the amiga some time (just for kicks!)
Perhaps when i've finished it, you'd like to transfer my "book of skulls" level-set over to the ST? Or maybe just allow your editor to import/export the same files mine can spit out?
-
Thanks Horace :)
My nostalgia was sparked a couple of months ago to play Bloodwych again, and, given my roots as a developer, thought it would be interesting to see what the "inner workings" were! ;)
I did some misc googling and found some info, but found your thread the most relevant by far!
I really am in the early days (given this is a hobby), but it is kinda fun to see what techniques and tricks had to be used back then to create games. I probably only spend a couple hours a week at most tinkering around.
I am working with the original ST version - i.e. image of the original disk, not a cracked version. I also have an original image for the data disk, which I hope to delve into some day.
I intend on understanding what the source files contain and what they do. With a (very distant) goal of digging into the assembler and understand the game logic, so that I may re-create the game in C#.Net.
For the time being, my app will just be a data extractor and renderer, rather than an editor, per se. But I don't think it would be hard to extend it to pass modifications back into the source and play via STEEM.
I am more than happy to share any of my findings as this is purely for my own interest, and it's good to keep Bloodwych alive :)
I suppose my biggest question at the moment is how/where all the images and sounds are stored. Is there a specific format used?
Cheers
-
the images are in some horrible ST raw format (even on the amiga) , and I'm afraid I know little about ripping them, because I've never had any success. The 4/5 amiga sounds are also just a raw format, and pretty easy to find in the data if you just load the game binary into an amiga sample player or tracker.
I think you will find much of even the 68000k asm code is the same between the ST & Amiga versions.
It would certainly be amazing to have a fully commented source code of the original game engine, as then I (or others) could consider remakes / sequels on either the original platform, or others, based on that code. (i for example, would like to do a full sequel on the Amiga, with new spells, weapons, monsters etc, but without losing the original RPG engine)
In 5-6 weeks time, I will finish year one of my college course, and i'll get a summer break - I try add upload some more data info to the main thread for you then (plus, it's a project I'd quite like to get into again)
in the meantime, please feel free to post anything here you'd like to contribute that you find!!
edit:
just for fun i updated : Hacking: Wall Switches (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39953.html#msg39953) to the kind of standard I want to maintain the info at, because this was one of the things that really bugged me for many years!! I was so very happy when i finally cracked this one (along with Pad Triggers) because it really opened up the game in terms of editing.
I'll try and do pad triggers in the next few days, and then maybe update the Map data-type definitions. (Spaces and Misc should be quite short!)
-
If I recall the horrible ST raw format uses some interesting 5bits/colour per pixel or something equally as horrid. I'll look into it and see what I can find...
I guess the ST may have used some kind of ST chip synth instead of raw sound samples, because it all sounds so clanky.
The 68000 asm decompiling will probably be a large task, I have found some website in the past that might assist with this, so I might have to go searching to see if there's any 68000 decompiler to make this task easier.
Yeah, my thought was to recreate the engine on the PC using .NET, if only because it'll be heaps easier to debug and extend. Appreciate this is a mammoth task though (but I can dream...)
Thanks for the wall switch info. I guess that you've probably already discovered the items that I have developed, although if my c# code can help describe the logic, do give me a shout! :)
-
i started to add some cross-reference links to the main contents of the hacking thread (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39941.html#msg39941)
I added again the object lists:
BW Object Lookup Table (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39958.html#msg39958)
BEXT Object Lookup Table (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39959.html#msg39959)
which had somehow been lost. This is actually the very first data I started to edit when I began hacking Bloodwych back in the 1990s! it was nice to be able to stuff the champions pockets full of goodies right at the start!!
I also began to update the Pad / Floor Traps & Triggers (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39952.html#msg39952) information, although map data (6) (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39950.html#msg39950) needs updating to go with this (just as map data (1) (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39945.html#msg39945) does for the switches) , and I havent put any descriptions yet for some of the more wierd and wonderful action that are performed by a few of these!!
Let me know if the format i'm adding this info in is of any help, and if there is anything you'd like to know about first (e.g. map data, pocket layouts etc etc)
edit... i also need to make links easier to see on this Crypt forum theme at some point!!
-
That's great :)
The button info confirms some of the hunches I had so far, and the remaining new info will save me a lot of time trying to figure things out :)
What I plan to do eventually is create a spreadsheet to map out the areas of the "CODE" file to determine the areas of data and areas of suspected code/content.
The Atari ST source also consists of a few other source files - the auto bootloader, a resource file and some empty saved games, so I'll probably investigate those too at some point.
One thing I notice is that the colour of buttons seem to be random, and some banners (out side of specific Green Serpent/Blue Moon/Red Dragon/Yellow Chaos) also seem to be random colours. Is this determined by X,Y or something else?
If you'd like me to post my C# Logic for determining values then I'm happy to push it up into this thread. :)
-
ATARI ST Monster Data:
[$17474] = 12 bytes of Monster Data - counts of monster per tower - i.e. 6x2 bytes
[$17480] The Keep
[$17780] The Serpent Tower
[$17a80] The Moon Tower
[$17d80] The Dragon Tower
[$18080] The Chaos Tower
[$18380] Zendik's Tower
For each tower, assume we are looping through the monster data, 6 bytes at a time. ID is just the counter.
This method inits a monster data, so we can then store in a list and render out at the relevant F/X,Y later.
(One thing i did find here is that the floor is 1-based, whereas I generally stored as 0-based, hence needed to subtract 1.)
public Monster(int id, List<byte> data)
{
if (data == null || data.Count != 6)
{
throw new ArgumentException(string.Format("Incorrect Monster Data size {0}",
data == null ? "null" : data.Count.ToString()));
}
ID = id;
Type = EnumType.None;
/* AB CD EF GH IJ KL
A = Type of Monster
B - The number of the tower floor on which the monster starts. (1-based)
CD - The X-coordinate postiioning of the tower floor on which the monster starts.
However, if between two and four monsters are to start in the same location (teamed) this number is left void and given a value of FF on the second to fourth team-members.
EF - The Y-coordinate positioning of the tower floor on which the monster starts.
GH - The level of the monster. This determines the monsters strength, speed, available spells, and for non-humanoid monsters also the colour of the monster.
IJ - This is the form / look of the monster. This number represents those given in the monster lookup table.
KL - This is the rather unusual "team-data" number. In the majority of cases this is left blank with the value of FF.
However, when two monsters are to begin the game "teamed" with the same starting postion, this value simple increases from 00 and beyond, for every teamed-monster.
*/
int ab = data[0];
int cd = data[1];
int ef = data[2];
int gh = data[3];
int ij = data[4];
int kl = data[5];
int type = ab >> 4;
int f = ab % 0x10;
F = f-1; //-1 to make 0-based
X = cd;
Y = ef;
Level = gh;
AppearanceID = ij;
TeamPosition = kl;
switch (type)
{
case 0x00:
Type = EnumType.Normal;
break;
case 0x01:
Type = EnumType.Spellcaster;
break;
case 0x03:
Type = EnumType.None;
break;
case 0x04:
Type = EnumType.ArcBolt;
break;
default:
Type = EnumType.None;
break;
}
}
Other notes:
- The monster type I found 4 to be the ArcBolt in the Dragon Tower, not 3 as in your notes. I also found only 1 monster of type 3 who lives in the serpent tower - still to investigate what's so special about it.
- The Behemoth in Zendicks tower starts at X=0x89, when he should be at X=0x09. I'm not sure if there is some kind of mod that should be applied to the X co-ordinate or if there's something special that "teleports" him in to the centre of the last floor.
- The team data seems to work by setting a unique number per monster in a team. The first monster (let's call him the leader) has an X co-ord as you state, the others in the same team have 0xFF. For a monster "leader" with team number N, the other monsters have N+1, N+2, N+3. So if you loop through the monsters in order with team data, you can look at the previous team member and do the following:
//now we need to fix up the teaming
List<Monster> teamMonsters = Monsters.Where(m => m.TeamPosition != 0xFF).ToList();
foreach (Monster m in teamMonsters)
{
if (m.X == 0xFF)
{
Monster prevTeamMember = teamMonsters.Where(t => t.X != 0xFF && t.TeamPosition == m.TeamPosition-1).FirstOrDefault();
if (prevTeamMember != null)
{
Monsters.Single(z => z.ID == m.ID).X = prevTeamMember.X;
Monsters.Single(z => z.ID == m.ID).Y = prevTeamMember.Y;
}
}
}
Hope this helps! :)
-
actually, i think you may be right, in that monster 4 is the arc bolt machine - i need to check my editor code to confirm... but i seem to recall that it should be something like this:
This can be one of three values;
0 - Normal Fighting Monster
1 - Normal Spell-casting Monster
2 - Fighting Drone
3 - Spell-casting Drone
4 - Arc Bolt Machine
The "drones" dont seek out the players, and act in the same way as a Summon spell.... they simply walk along until they hit a wall and turn. Obviously, they do also attack players that get in their way in the process!!
I've used a number of these in Book of Skulls, but you are right, in the original BW there is only one in the serpent tower (he is a fat summon character who walks back and forth between 3 spaces)
You are also spot-on with the team data stuff ... I will update my post to include everything you've written! (including the ST addresses) Unfortunately, i did not update my notes when I was working on the editor!
This teaming also puts a restriction on the number of monsters you can have in a tower... because if too many "auto" team up when you start a tower, then the team-counter goes over a set value and causes the game to crash (not good!!)
thanks for this :)
-
Good stuff - I found that drone Summon in the Serpent Tower last night, as I was playing through. Haven't played Bloodwych in so many years it's good to relive the old memories. And my map generator is definitely helping! :)
I guess that limitation on the number of teams you mention explains why you often see "couples", so they can't team up with other "couples".
I'll start posting the rest of my investigations here... a lot of this is based on your posts anyway, so probably nothing new.
ATARI ST Floor and object data -
[$0EE3C] The Keep
[$1023E] The Serpent Tower
[$11640] The Moon Tower
[$12A42] The Dragon Tower
[$13E44] The Chaos Tower
[$15246] Zendik's Tower
/*
* FLOOR DATA
($1402 total bytes per block)
[$00000] - (8 bytes) Width values of the respective Floors of the the Tower, starting with the lowest floor.
[$00008] - (8 bytes) height of each of the floors, starting with the lowest floor.
[$00010] - ($28 bytes) ?other header? not sure what this is for
[$00038] - ($FC8 bytes) Map Data
[$01000] - ($402 bytes) Object data - first 2 bytes signify amount of data used for object data
*/
Getting item stacks for a tower - assume we are parsing the $400 bytes of object data (from $01002)
where objectDataSize = the value parse from the first 2 bytes
int objOffset = 0;
ItemStacks = new List<ItemStack>();
while (objOffset < objectDataSize)
{
//what we care about here is how many bytes are used for this ItemStack
//This is a minimum of 5 bytes, with an additional 2 bytes added for each extra item in this location.
//Format: AB CD EF GH IJ .. KL MN .. OP QR .. etc..
int ef = DataObjects[objOffset + 2];
//EF = The number of items in the given location, minus 1.
int bytesToTake = (2*ef) + 5;
List<byte> bytesForThisItemStack = DataObjects.Skip(objOffset).Take(bytesToTake).ToList();
ItemStack itemStack = new ItemStack(bytesForThisItemStack);
ItemStacks.Add(itemStack);
//increment to the next ItemStack
objOffset += bytesToTake;
}
public ItemStack(List<byte> data)
{
Position = EnumPositon.None;
//this should be a new object
//This is a minimum of 5 bytes, with an additional 2 bytes added for each extra item in this location.
//Format: AB CD EF GH IJ .. KL MN .. OP QR .. etc..
int ab = data[0];
int cd = data[1];
int ef = data[2];
//A = The position of the items in the location (4 positions for a standard floor, and 2 positions for a shelf)
//00 - Position 1 - North West Space, North Facing Bottom Shelf, West Facing Top Shelf
//04 - Position 2 - North East Space, East Facing Bottom Shelf, North Facing Top Shelf
//08 - Position 3 - South West Space, West Facing Bottom Shelf, South Facing Top Shelf
//0C - Position 4 - South East Space, South Facing Bottom Shelf, East Facing Top Shelf
int a = ab >> 4;
switch(a)
{
case 0x00:
Position = EnumPositon.NorthWest;
break;
case 0x04:
Position = EnumPositon.NorthEast;
break;
case 0x08:
Position = EnumPositon.SouthWest;
break;
case 0x0C:
Position = EnumPositon.SouthEast;
break;
}
//BCD = The index number for the location.
int b = ab % 0x10;
int bcd = (b << 8) + cd;
IndexNumber = bcd;
//EF = The number of items in the given location, minus 1.
//go through each this count to extract object data
Items = new List<Item>();
for (int oi = 0; oi <= ef; oi++)
{
//lets call each pair GH IJ for simplicity
//GH - object code for the item in the location. based on the object lookup table
int gh = data[3 + (2 * oi)];
//IJ - This is the number of item GH to be "stacked" at this point.
int ij = data[4 + (2 * oi)];
Item item = new Item(gh, ij);
Items.Add(item);
}
}
and an example of a map generated for the keep is attached... as you can see I have yet to ingest the champion data, only objects and monsters.
-
ATARI ST Button Data:
[$0589E] The Keep
[$058DE] Serpent Tower
[$0591E] Moon Tower
[$0595E] Dragon Tower
[$0599E] Chaos Tower
[$059DE] Zendick's Tower
64 bytes ($40 bytes) per tower
ATARI ST Pad/Trigger Data:
[$06db4] The Keep
[$06e34] Serpent Tower
[$06eb4] Moon Tower
[$06f34] Dragon Tower
[$06fb4] Chaos Tower
[$07034] Zendick's Tower
128 bytes ($80 bytes) per tower
-
cool bananas :D
I've attached a .zip file of all the stuff which can be inputted/outputted from my editor. A couple of the files (.map .ob) are the swappable with the PC game files. I've used the original game data, so you can easily find this same data in the ST version, and see if it's identical (i think it will be)
I *think* these cover everything in the game, so if you can make these interchangeable in/out of your map program, we could be onto a real winner with swapping data between ST and Amiga versions (had you considered adding support for the amiga binary files?)
This list should also give you an idea of all the things that are easily editable relating to gameplay itself... there are a few others i'd like to look into (such as the menu-screen layout) but those may differ between versions... i've yet to confirm that.
files contained are:
scrolls.block 3380 ----rwed Today 00:59:42
champions.pockets 256 ----rwed Today 00:59:42
champions.stats 512 ----rwed Today 00:59:42
woodtrap2.location 4 ----rwed Today 00:59:42
woodtrap1.location 4 ----rwed Today 00:59:42
gem-blu.locations 24 ----rwed Today 00:59:42
gem-tan.locations 24 ----rwed Today 00:59:42
dungeon.entrances 24 ----rwed Today 00:59:40
keep.entrances 26 ----rwed Today 00:59:40
zendik.triggers 128 ----rwed Today 00:59:40
chaos.triggers 128 ----rwed Today 00:59:40
drag.triggers 128 ----rwed Today 00:59:40
moon.triggers 128 ----rwed Today 00:59:40
serp.triggers 128 ----rwed Today 00:59:40
mod0.triggers 128 ----rwed Today 00:59:40
zendik.switches 64 ----rwed Today 00:59:40
chaos.switches 64 ----rwed Today 00:59:40
drag.switches 64 ----rwed Today 00:59:40
moon.switches 64 ----rwed Today 00:59:40
serp.switches 64 ----rwed Today 00:59:40
mod0.switches 64 ----rwed Today 00:59:40
zendik.monsters 768 ----rwed Today 00:59:38
chaos.monsters 768 ----rwed Today 00:59:38
drag.monsters 768 ----rwed Today 00:59:38
moon.monsters 768 ----rwed Today 00:59:38
serp.monsters 768 ----rwed Today 00:59:38
mod0.monsters 768 ----rwed Today 00:59:38
monsters.totals 12 ----rwed Today 00:59:38
zendik.ob 1026 ----rwed Today 00:59:38
chaos.ob 1026 ----rwed Today 00:59:38
drag.ob 1026 ----rwed Today 00:59:38
moon.ob 1026 ----rwed Today 00:59:38
serp.ob 1026 ----rwed Today 00:59:38
mod0.ob 1026 ----rwed Today 00:59:38
zendik.map 4096 ----rwed Today 00:59:38
chaos.map 4096 ----rwed Today 00:59:38
drag.map 4096 ----rwed Today 00:59:38
moon.map 4096 ----rwed Today 00:59:38
serp.map 4096 ----rwed Today 00:59:38
mod0.map 4096 ----rwed Today 00:59:38
edit: mod0 is the name given for the Keep in the PC version, so i've used that filename here, although I always reference The Keep in my own listings, as it makes more sense!
-
Yep - I'd just need to add another layer to allow me to choose where i'm getting the bytes from, either the ST's "CODE" file, or lots of individual files as you note below.
Subsequently I will need to add logic to output data read in to either ST or AMIGA format.
My map application doesn't have any ability to edit data at this stage (probably a fair way off yet) but I can test my approach with a book of skulls map potentially...
This might be ready in the next week or two, depending on how much free time I have... ;)
I also still need to ingest champion and scroll data, to make my system more complete.
-
Just to elaborate, what I'm thinking is:
Stage 1:
ATARI ST "CODE" -> intermediate format -> Render in my map tool
AMIGA "CODE" -> intermediate format -> Render in my map tool
PC "CODE" -> intermediate format -> Render in my map tool
intermediate format -> Render in my map tool
Where the intermediate format is the common format you have supplied above in the zip.
This will also allow me to check that the Amiga, ST and PC data is exactly the same.
If you can let me know the best Amiga source and emulator to use, I can try and extract relevant file(s) to be used on a PC with my map tool.
I think I have a PC version of bloodwych kicking around somewhere.
Once we have this complete, then I can try the following:
Stage 2:
intermediate format -> write into ATARI ST "CODE"
intermediate format -> write into AMIGA "CODE"
intermediate format -> write into PC"CODE"
This would allow us to potentially take a "Book of Skulls" map in intermediate format and try and apply to the Atari ST source.
I would guess I would have a copy of the "CODE" in my solution to use as a template to write into. Then would manually have to get this file back into the emulator. (For the ST you can map a PC folder to be treated as a "hard drive" to copy in/out of the emulator).
-
Ok, more progress.
I have named the intermediate format "BCF"... i.e. "Bloodwych Common Format".
My app now does the following:
Load ATARI ST -> BCF in memory
BCF as files (i.e. contents of your zip) -> BCF in memory
BCF in memory -> BCF as files
i.e. I can now save out in the same format as yours! :)
I am yet to ingest from the st the following:
champions.pockets
champions.stats
dungeon.entrances
gem-blu.locations
gem-tan.locations
keep.entrances
scrolls.block
woodtrap1.location
woodtrap2.location
(still work in progress to locate the bytes and ingest...)
---------------------
FINDINGS:
Using windows comp.exe on the files generated so far, nearly all files are exact matches, the only discrepancy is
mod0.map
Your export / ST export
Compare error at OFFSET 31
file1 = 15
file2 = 1F
Compare error at OFFSET 33
file1 = 15
file2 = 1F
Compare error at OFFSET 34
file1 = 0
file2 = 5
Compare error at OFFSET 35
file1 = 18
file2 = 4C
As far as I know these bytes are in the mysterious map "header" data which I'm not sure what it defines, so as yet I can't tell the difference
Compare error at OFFSET 8C8
file1 = 11
file2 = 9
Keep F3, x:15, y:13, i:1096
It's the first bronze key door in the game
I noticed this before as an anomaly in the ST version - the only bronze key door with ABCD = 0915 instead of 1115 - but it still works exactly the same
That's it!
-
i'll check my header interpretation from the editor source, but iirc one part of it is to define the X/Y "offset" of each floor (and therefore 16 "words" are avaialble for this... enoguh to cover the max of 8 floors, in each coord.
This allows you to "arrange" the way the floors stack up on top of each other.... it's a bit hard to explain but obviously this is really important in allowing you to make sure your stairs etc match-up between floors of different sizes. i'll see if i can post a few pictures that help explain.
There is also a prt of it that seems to represent a different floor in each tower/dungeon .... i think it's two values, the offset-value presenting 0,0 of the floor and the offset representing the coord for floorwidth,floorheight .... by 'offset-value' in this case, i mean the same values that are used to position objects in a level.
I've also had a look into my wall-scroll reading/editing data, and i have attached the code for that section of the editor. I'll try and write up a meaningful interpretation of it soon! (It's a bit tricky because it's very much variable-length)
these locations will be needed though:
scrollref = $1a058
scrolltext = $1a0ea
and these funtions are defined:
DEF FN FINDSCROLL = deek (scrollref+(scrollnumber*2))
DEF FN FINDSCROLLTEXT = deek (scrolltext + findscroll + LOC)
('Deek' is AMOS's way of peeking a word rather than a byte, and i cant remember what the variable LOC is for)
-
Thanks mate, this'll be a great help :)
Nearly played to the end of the serpent tower so far. Having a map handy makes life easier!
I think I get what you mean regarding the floor offsets. I was wondering how the different floors lined up, so the game would know when you went up/down where you should end up.
I've found the scroll text in the St code, so I'll attempt to ingest it in the next few days if I get a chance.
Cheers :)
-
Ok, a quick read-through of that scroll data, reminds me that we have two "sections" of the scroll information, within the scrolls.block file.
Part 1 - is a list of word-values, declaring how many bytes into the part 2 each scroll begins.
e.g. Scroll #001 will always begin at value $0000, so this is the first value.
If scroll #001 is $20 bytes long, then the next value will be $0020 , if scroll #002 is $12 bytes long, then the third value will be $0032 etc.
Part 1 is $0092 bytes long.
this is the original $92 bytes of data from BW, laid out in a meaningful way.
0000 0026 004E 008A 00B3 00CC 00DE 0127
0145 0180 019B 01B4 0207 0220 0252 0269
028A 02A9 02D3 02EB 0340 037B 03A8 03C0
03E9 040C 0444 047C 049D 04D3 0500 0525
054F 056D 0593 05C2 05FE 0642 0667 067E
06B7 06DB 0709 0731 076F 07A3 07D6 0802
0837 0858 087C 08A3 08BA 08E1 092E 0950
0987 09A6 09D0 0A0B 0A29 0A40 0A67 0AA2
0AED 0B14 0B4C 0B77 0BA2 0BCE 0BFA 0C32 0C5D
Part 2 is simply a list of the scrolls, with a specific "format", this is $CA2 bytes long.
here is the very first scoll entry from BW, also in a format to make it understandable:
FC 1E 04 54 48 45 20 4B 45 59
FC 1F 05 41 42 4F 56 45
FC 1E 06 44 4F 54 48 20 4C 49 45
FC 1F 07 42 45 4C 4F 57
FF
in-game this reads:
THE KEY
ABOVE
DOTH LIE
BELOW
- Byte 1: $FC should be seen as simply a "line declaration", so the first scroll can be seen to be 4 lines long.
the following two bytes, are effectively an X/Y coordinate for placing the scroll on the screen.
- Byte 2: X Coordinate
$1E should be seen as zero however, as any lower value than this will not be on the scroll on-screen.
In order to keep the text on screen, the length of the text on each row, determines the maximum value of the X coord. I.e. if you have a row of only 1 character long, you can have a maximum X value of 8 (therefore value $1E + $08 = $26 can be patched in) ... if you have a row of 3 characters, your maximum X value is 6 etc.
You can fit a total of 9 characters into a row only.
- Byte 3: Y Coordinate
$03 should be seen as zero however, as any lower value than this will not be on the scroll on-screen
there are 7 possible rows which can be used for each scroll, so the maxium value of this byte should be $03+$07 = $0A, for the bottom row of the scroll.
-Byte 4+: Text in ASCII format.
Standard text data, continuing until either a $FC or $FF filled byte is reached.
After all data required for text layout is done, a single byte of $FF is used to terminate the scroll data.
When deciding which scroll is referred to in the map data, a standard reference number is used (as with switches etc), however, the starting point (i.e. which scroll is referred to by Reference 1) changes with each tower. I do not recall finding how this is determined from within the game-code... it must be somewhere though!
-
Sweet :)
It's always cool to see the nifty tricks devs had to do with their data structures. Keep forgetting we are spoiled today with the tools we have.
-
And more!!
the attached three pictures should explain the floor-offset thing a bit better.
basically the white grid is the current floor we are editing (floor 4) and the dark-grey grid is the floor below (floor 3)
- when both floors are offsetted to 0,0, they are in-line, at in the top left of the editing area.
- when the offset of floor 4 is changed to 1,1 (picture 2) the stacking of the two grids changes, and the floor is no longer in-line with the top left of the editing area.
- in picture 3, i have adjusted the offset of floor 3 to 2,2... but we are still viewing from floor 4 (for the sake of reference) ... we can see here a further push away from that top corner.
Although my editor puts limitations on offsets of floors, in order to keep everything on the editable screen, I *beleive* that in-theory you could use vastly higher offset values (just as you could floor sizes, which i do not allow) , with only a very small overlap. This would mean that you could continually move further and further from your original start position.
There's no real reason to do this however, other than "because you can".
-
Hope this is all helping.
Was really interesting to read those differences in the amiga and ST files.... i havent yet exported the "common format" from any of the later versions of the Amiga BW binary though (it would require a fair bit of jigeery pokery in my editor to do that) - perhaps some of those may be the same as the ST though... who knows.
Is there any chance you could post a copy of the main ST binary file here so I can have a peek through it with my hex editor? I especially want to have a look for that woodtrap stuff because thats hard-coded into some of the 68000 asm code.
-
Header Data for The Keep
0C 15 0F 1F 13 04 00 00 <<< Widths of Floors 0-7
01 15 0F 1F 13 05 00 00 <<< Heights of Floors 0-7
00 00 << offset at which floor 0 begins
00 18 << offset at which floor 1 begins
03 8A << offset at which floor 2 begins
05 4C << offset at which floor 3 begins
0C CE << offset at which floor 4 begins
0F A0 << offset at which floor 5 begins
0F C8 << offset at which floor 6 begins (although there is no floor 6 in the Keep)
00 00 << offset at which floor 7 begins
08 05 08 00 06 0B 00 00 <<< X offsets of Floors 0-7
05 05 08 00 06 0C 00 00 <<< Y offsets of Floors 0-7
00 15 00 15 <<<< Width / Height of "special" floor (floor 1 for the keep)
00 18 <<< offset at which "special" floor begins (floor 1 for the keep)
00 05 <<< Number of Top Floor
Post: Hacking : Map-Data Structure (http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39942.html#msg39942) now updated with this info :)
edit 2: this also tells us that the ST version has the starting floor (3) as the "special" floor in the Keep, unlike the Amiga version having Floor 1 (aka The Maze), and that this and the bronze key door are our only differences!
-
Doh, just typed a reply and lost it because the forum didn't like my attachment... :(
Anyway, have now attached "CODE" as "CODE.txt" but it's really just the main binary.
Not sure what version of ST bloodwych it is, other than it's a true STX image of an original disk - not a trainer or demo.
Your offset explanation makes perfect sense - hoping to get some more time to work on my code later this week.
Is there any significance to the "special" floor, any why it would be different between ST/Amiga?
And what are the wood traps?
Below are the ST addresses for this binary:
From To Ingest Notes
0x00000 0x589D No Unknown
0x0589E 0x058DD Yes Buttons K
0x058DE 0x0591D Yes Buttons S
0x0591E 0x0595D Yes Buttons M
0x0595E 0x0599D Yes Buttons D
0x0599E 0x059DD Yes Buttons C
0x059DE 0x0601D Yes Buttons Z
0x0601E 0x06DB3 No Unknown
0x06DB4 0x06E33 Yes Triggers K
0x06E34 0x06EB3 Yes Triggers S
0x06EB4 0x06F33 Yes Triggers M
0x06F34 0x06FB3 Yes Triggers D
0x06FB4 0x07033 Yes Triggers C
0x07034 0x070B3 Yes Triggers Z
0x070B4 0x0EE3B No Unknown
0x0EE3C 0x1023D Yes Floors And Objects K ($28 bytes other header data still to account for)
0x1023E 0x1163F Yes Floors And Objects S
0x11640 0x12A41 Yes Floors And Objects M
0x12A42 0x13E43 Yes Floors And Objects D
0x13E44 0x15245 Yes Floors And Objects C
0x15246 0x16647 Yes Floors And Objects Z
0x16648 0x17474 No Unknown
0x17474 0x17476 Yes Monster Count K
0x17476 0x17478 Yes Monster Count S
0x17478 0x1747A Yes Monster Count M
0x1747A 0x1747C Yes Monster Count D
0x1747C 0x1747E Yes Monster Count C
0x1747E 0x1747F Yes Monster Count Z
0x17480 0x1777F Yes Monsters K
0x17780 0x17A7F Yes Monsters S
0x17A80 0x17D7F Yes Monsters M
0x17D80 0x1807F Yes Monsters D
0x18080 0x1837F Yes Monsters C
0x18380 0x1867F Yes Monsters Z
0x18680 0x590FA No Unknown
-
the "woodtraps" are my name for the wooden doors that are automatically closed behind you and effectively "locked" by the clever use of two sets of doors.
These only occur once, in the Keep, for the two gem-hole rooms on Floor 2. However, they are are hardcoded in 68K ASM to affect a particular X/Y coord.. it's this coord in the ASM code that is is stored and can be edited.
I've just done a quick search, and the ASM code is the same on both versions :)
the address of woodtrap 1 coord is: $7442 (ST) (values: 000D000C) and $745A (ST) (values: 00030000) for woodtrap 2 :)
i have no idea what the significance of a special floor is, or why it should differ on each version. Perhaps we should list the special floors of each Tower and then see if we can guess at what is common between them all?
I hope to have a list of all the ST addresses soon, with support added to my editor :)
-
ok, i worked out all the addresses, but i'm getting some wierd things happen.
1) all the addresses you gave me were $4 bytes out.
2) The serpent tower void-lock doors at the start are rotated the wrong direction (yet i thought you had done a comparison on the .map files, with only differences in the Keep?)
3) one of the floors (4) of the Chaos tower seems really screwed up!
4) one of the bluish-gem [positions in the Keep aligns to x0B y0D instead of x0B y0A like it should.
5) the Monster data seems to "slip out" by 1 byte between the dragon and chaos towers... see the addresses below.
I dont know if calling it .txt caused a problem on the download - the file i used was 364773 long (which seems strange in itself, to have an odd-number length file) .... perhaps you could zip the file first then upload it?
BW_ST:
LOOKUPTOWER(0)=$EE38 : Rem -- mod0.map
LOOKUPTOWER(1)=$1023A : Rem -- serp.map
LOOKUPTOWER(2)=$1163C : Rem -- moon.map
LOOKUPTOWER(3)=$12A3E : Rem -- drag.map
LOOKUPTOWER(4)=$13E40 : Rem -- chaos.map
LOOKUPTOWER(5)=$15242 : Rem -- zendik.map
LOOKUPSWITCHES(0)=$589E : Rem -- mod0.switches
LOOKUPSWITCHES(1)=$58DE : Rem -- serp.switches
LOOKUPSWITCHES(2)=$591E : Rem -- moon.switches
LOOKUPSWITCHES(3)=$595E : Rem -- drag.switches
LOOKUPSWITCHES(4)=$599E : Rem -- chaos.switches
LOOKUPSWITCHES(5)=$59DE : Rem -- zendik.switches
LOOKUPTRIGGERS(0)=$6DB4 : Rem -- mod0.triggers
LOOKUPTRIGGERS(1)=$6E34 : Rem -- serp.triggers
LOOKUPTRIGGERS(2)=$6EB4 : Rem -- moon.triggers
LOOKUPTRIGGERS(3)=$6F34 : Rem -- drag.triggers
LOOKUPTRIGGERS(4)=$6FB4 : Rem -- chaos.triggers
LOOKUPTRIGGERS(5)=$7034 : Rem -- zendik.triggers
LOOKUPOBJECTS(0)=LOOKUPTOWER(0)+$1002 : Rem -- mod0.ob +2
LOOKUPOBJECTS(1)=LOOKUPTOWER(1)+$1002 : Rem -- serp.ob +2
LOOKUPOBJECTS(2)=LOOKUPTOWER(2)+$1002 : Rem -- moon.ob +2
LOOKUPOBJECTS(3)=LOOKUPTOWER(3)+$1002 : Rem -- drag.ob +2
LOOKUPOBJECTS(4)=LOOKUPTOWER(4)+$1002 : Rem -- chaos.ob + 2
LOOKUPOBJECTS(5)=LOOKUPTOWER(5)+$1002 : Rem -- zendik.ob +2
LOOKUPHARDCODE1=$7442 : Rem -- woodtrap1.location
LOOKUPHARDCODE2=$745A : Rem -- woodtrap2.location
LOOKUPGEMTAN=$5820 : Rem -- gem-tan.locations
LOOKUPGEMBLU=$5838 : Rem -- gem-blu.locations
LOOKUPDUNGEONSTART=$7248 : Rem -- dungeon.entrances
LOOKUPKEEPFLOORSTART=$712A : Rem -- keep.entrances
LOOKUPKEEPSTART=LOOKUPKEEPFLOORSTART+$6
LOOKUPMONSTERCOUNT=$17470 : Rem -- monsters.totals
LOOKUPMONSTER(0)=$1747C : Rem -- mod0.monsters
LOOKUPMONSTER(1)=$1777C : Rem -- serp.monsters
LOOKUPMONSTER(2)=$17A7C : Rem -- moon.monsters
LOOKUPMONSTER(3)=$17D7C : Rem -- drag.monsters
LOOKUPMONSTER(4)=$1807B : Rem -- chaos.monsters
LOOKUPMONSTER(5)=$1837B : Rem -- zendik.monsters
LOOKUPCHAMPIONSTATS=$E812 : Rem -- champions.stats
LOOKUPCHAMPIONPOCKETS=$EA12 : Rem -- champions.pockets
LOOKUPWALLSCROLLREF=$1A212 : Rem -- scrolls.block
LOOKUPWALLSCROLLTEXT=LOOKUPWALLSCROLLREF+$92
Rem -- internal data, reference only.
LOOKUPSPELLTEXT=$19D84 : Rem -- spell names, armour terror etc
LOOKUPSPELLNAME=$1867B : Rem -- maryhadalittlelaaneeitwerragudd
LOOKUPNAMES=$D8F6 : Rem -- champion names (first occurance)
Return
edit: i tried a quick and dirty hack of a .ST disk image (equinox crack) and found that it did not have the same differences as the file i downloaded above:
- the bronze door at the start of the game was normal!
- the chaos tower was fine
- the monster-lists were now equally spaced. ($300 each)
i definitely think there is something up with that file!!
I am also changing the output of the woodtrap 1/2 data (to include the preceding 4 bytes, even though they are never edited) to make it easier to locate this data in different versions..... The idea being, i may create a program to read in say... 6 bytes of my normal "output" files, then search for where this data is within a disk image or other binary.... this would mean i could add additional support for images, versions, etc much quicker.
-
Yep I definitely think something went awry uploading to the forum. I'll zip it and re-upload it when I get home later. :)
I'm pretty certain that there must've been multiple versions released for the ST, as the start screen on my version shows F9/F10 for load/save, whereas my "real" ST 3.5 disk version (long lost now) never had that feature.
That bronze key difference is a weird one, but it kind of makes sense in the logic, I think it goes something like NoLock, MageLock, VoidLock, BronzeLock over a certain byte combo. My memory fails me but I'll dig the logic out later.
I like the idea of writing a "searcher" based on known byte strings. I might add this to my TODO list, as eventually I could removing the hard-coding on ingesting ST data and make it more configurable for any binary passed in.
-
Reattached CODE in ZIP, should be 364,794 bytes
Here is my code for parsing a Metal Door Cell (i.e. D = 5)
It's all based on A, except when A=0, then there are checks on B and C too.
You can see how the Bronze key fits snugly into args.B >= 0x8 && args.C == 0x1
Maybe the developer initially had another type of lock in mind? Regardless it's only the special bronze door on the Atari ST that uses this case.
switch (args.A)
{
case 0x0:
if (args.B < 0x8 && args.C == 0x0)
{
LockState = EnumLockState.NoLock;
}
else if (args.B < 0x8 && args.C == 0x1)
{
LockState = EnumLockState.MageLock;
}
else if (args.B >= 0x8 && args.C == 0x0)
{
LockState = EnumLockState.VoidLock;
}
else if (args.B >= 0x8 && args.C == 0x1)
{
LockState = EnumLockState.BronzeKey;
}
break;
case 0x1:
LockState = EnumLockState.BronzeKey;
break;
case 0x2:
LockState = EnumLockState.IronKey;
break;
case 0x3:
LockState = EnumLockState.SerpentKey;
break;
case 0x4:
LockState = EnumLockState.ChaosKey;
break;
case 0x5:
LockState = EnumLockState.DragonKey;
break;
case 0x6:
LockState = EnumLockState.MoonKey;
break;
case 0x7:
LockState = EnumLockState.ChromaticKey;
break;
}
-
i cant check right now, (at work) but i suspect on the Amiga version this door will be openined with the serpent wand or something (the next item in the item list)... i seem to remember managing to do that myself when investigating data!!
my editor presently marks it as a "void" door rather than invalid.
-
i've adapted my editor to read the (now correct!!) binary file, which is great :)
I guess i will need a copy of the ST extended levels file soon!
I've attached a ZIP file with all of the files outputed using the ST version.
The ST bronze door is definitely a void-lock on the amiga version (see attached)... so it's a "non common" rule and I wont be letting my editor create that kind of data. I guess if i'm going to play-through the ST data (to see if the special floor affects anything) i will need to manually fix this!
and here are the all-important address:
BW_ST:
LOOKUPTOWER(0)=$EE3C : Rem -- mod0.map
LOOKUPTOWER(1)=$1023E : Rem -- serp.map
LOOKUPTOWER(2)=$11640 : Rem -- moon.map
LOOKUPTOWER(3)=$12A42 : Rem -- drag.map
LOOKUPTOWER(4)=$13E44 : Rem -- chaos.map
LOOKUPTOWER(5)=$15246 : Rem -- zendik.map
LOOKUPSWITCHES(0)=$589E : Rem -- mod0.switches
LOOKUPSWITCHES(1)=$59DE : Rem -- serp.switches
LOOKUPSWITCHES(2)=$591E : Rem -- moon.switches
LOOKUPSWITCHES(3)=$595E : Rem -- drag.switches
LOOKUPSWITCHES(4)=$599E : Rem -- chaos.switches
LOOKUPSWITCHES(5)=$59DE : Rem -- zendik.switches
LOOKUPTRIGGERS(0)=$6DB4 : Rem -- mod0.triggers
LOOKUPTRIGGERS(1)=$6E34 : Rem -- serp.triggers
LOOKUPTRIGGERS(2)=$6EB4 : Rem -- moon.triggers
LOOKUPTRIGGERS(3)=$6F34 : Rem -- drag.triggers
LOOKUPTRIGGERS(4)=$6FB4 : Rem -- chaos.triggers
LOOKUPTRIGGERS(5)=$7034 : Rem -- zendik.triggers
LOOKUPOBJECTS(0)=LOOKUPTOWER(0)+$1002 : Rem -- mod0.ob +2
LOOKUPOBJECTS(1)=LOOKUPTOWER(1)+$1002 : Rem -- serp.ob +2
LOOKUPOBJECTS(2)=LOOKUPTOWER(2)+$1002 : Rem -- moon.ob +2
LOOKUPOBJECTS(3)=LOOKUPTOWER(3)+$1002 : Rem -- drag.ob +2
LOOKUPOBJECTS(4)=LOOKUPTOWER(4)+$1002 : Rem -- chaos.ob + 2
LOOKUPOBJECTS(5)=LOOKUPTOWER(5)+$1002 : Rem -- zendik.ob +2
LOOKUPMONSTERCOUNT=$17474 : Rem -- monsters.totals
LOOKUPMONSTER(0)=$17480 : Rem -- mod0.monsters
LOOKUPMONSTER(1)=$17780 : Rem -- serp.monsters
LOOKUPMONSTER(2)=$17A80 : Rem -- moon.monsters
LOOKUPMONSTER(3)=$17D80 : Rem -- drag.monsters
LOOKUPMONSTER(4)=$18080 : Rem -- chaos.monsters
LOOKUPMONSTER(5)=$18380 : Rem -- zendik.monsters
Rem ----
LOOKUPHARDCODE1=$7442 : Rem -- woodtrap1.location
LOOKUPHARDCODE2=$744C : Rem -- woodtrap2.location
LOOKUPGEMTAN=$5820 : Rem -- gem-tan.locations
LOOKUPGEMBLU=$5838 : Rem -- gem-blu.locations
LOOKUPDUNGEONSTART=$7248 : Rem -- dungeon.entrances
LOOKUPKEEPFLOORSTART=$712A : Rem -- keep.entrances
LOOKUPKEEPSTART=LOOKUPKEEPFLOORSTART+$6
LOOKUPCHAMPIONSTATS=$E816 : Rem -- champions.stats
LOOKUPCHAMPIONPOCKETS=$EA16 : Rem -- champions.pockets
LOOKUPWALLSCROLLREF=$1A218 : Rem -- scrolls.block
LOOKUPWALLSCROLLTEXT=LOOKUPWALLSCROLLREF+$92
Rem -- internal data, reference only.
LOOKUPSPELLTEXT=$18680 : Rem -- maryhadalittlelaaneeitwerragudd
LOOKUPSPELLNAME=$19D8A : Rem -- spell names, armour terror etc
LOOKUPNAMES=$D8FA : Rem -- champion names (first occurance)
Return
-
i've just had a thought.... is there any chance this Bronze Door is part of the copy protection?
The Equinox Crack on the ST doesnt have this different data (the door is the same as on the amiga) - perhaps if the copy protection is "passed" then the value of this door is changed? Have you tried making a copy of the disk , to standard DOS disk and seeing what happens? It's the perfect location for something like this - you get to "see" the game but not actually get anywhere.
Does any ST emulator let you monitor the address this is loaded to in memory, so you can see when (if) it is changed.... what is the value of this map-data in memory once the game is started?
questions and thoughts :)
-
I have the CODE file for the Extended levels too, so will zip and post that over later on. :)
Started writing a rudimentary byte searcher to find the bytes that make "The key of hue is what i'm due" (including all the bytes for offsets, line breaks, etc.), which seems to be working ok. So I might extend this next to search a given file for all the byte strings - so it is easy to identify offsets in a file.
I'm also going to break out my hard coding of addresses into a configurable file, so you can choose which set of offsets to use easily.
That's a good point about the bronze key door - when I get a chance I'll modify the bytes in my ST binary and try and run it through STEEM emulator, just to see what happens.
It might just be a quirk of an early version of Bloodwych, which was resolved in later versions. Same with the special floor.
-
I'm also going to break out my hard coding of addresses into a configurable file, so you can choose which set of offsets to use easily.
a configurable file was what i have been starting to consider.... and those files can be generated by scanning a file for all the correct addresses. I am only identifying my files by size at the moment though... perhaps later i will consider a better/proper way of identifying a file to be edited.
Whilst it would be nice to do "live" scanning when trying to edit a file, I am limited by the fact I am still writing the editor as an amiga program! (although i recommend running it on UAE with acceleration anyway)
It might just be a quirk of an early version of Bloodwych, which was resolved in later versions. Same with the special floor.
i would think if that were the case, the special floors shouldnt be in extended levels as well.... iirc, they are :S
-
configurable file was what i have been starting to consider.... and those files can be generated by scanning a file for all the correct addresses. I am only identifying my files by size at the moment though... perhaps later i will consider a better/proper way of identifying a file to be edited.
Yep, I'm thinking of basically something like: NiceName, FileName, Offsets. Given I have the luxury of .NET, either in XML or JSON. Possibly just having 1 main config file that you can add multiple definitions too, then on starting the app, you can choose a definition to work with.
This reduces the overhead of "Live" scanning a file every time, you just scan it once to get the offsets, then add it to your config for future work. Then you could feasibly export to any of these files, substituting in the bytes at the correct offsets.
i would think if that were the case, the special floors shouldnt be in extended levels as well.... iirc, they are :S
Interesting... well it might just be a case of observing what these floors are and trying to see if there's anything common about them. Or if it's just a quirk of something that was not implemented, or perhaps a copy-protection feature? Short of finding the assembly that uses these values it'll be guesswork.
-
Attached is Bloodwych Data Disk ST Binary (354,162 bytes)
I have made some progress tonight around removing my hard coded offsets.
I am in the middle of creating a method to inspect a binary and return the offsets. This will then be saved out to JSON format.
At present the output is hard-coded, but I can ingest from the JSON file, so now I read my offsets from a configurable file.
Attached text file shows my JSON config. Setup would mean inspecting a binary then adding (copy/paste) the result into this file and tweak, then you can choose to ingest from a list of known configs.
Next Steps
- finish my binary inspector to determine offsets from inspecting binary
- ingest all known binary data offsets for the BCF that I have yet to ingest
- upgrade my mapper to show other relevant data yet to ingest
- get an Amiga binary and attempt to ingest
- get the PC binary and attempt to ingest (may need upgrades to the config, since not just 1 binary...)
-
looking good!! and thanks again for the binary.
I was thinking, with regards to the PC version, i might make something that merged all the files, processed it accordingly, and then split the files back out again.
The woodtrap code is the only thing I am concerned about here, because it wont be 68k asm. I've had a quick search for the x/y coords but i cant find them - i think it would be pretty easy though to put into the generated config file (like you JSON one) something that says "woodtrap editing disabled" and simply have that turn the option off on the editor.
everything else that is not a .map/.ob file appears to be inside the main BWYCH.EXE file :D
Probably gonna have to take a break from BW for the next 2 weeks though while I complete my college work though (it's the end of term)... a shame, because I am quite getting into all this again now!!
-
Yes that does sound like a good idea to merge the PC files.
I have been tempted though to expand my configuration to allow for reading multiple files (so effectively there would be 1 ingestion process, regardless of how the files are named/laid out). I'll see how I go.
At the moment I don't worry about ingesting the woodtraps, but agree that might be something to disable if not easy to work with the PC version.
No worries if you need a break. I have a few things too that might impact my BW development time in the next few weeks. It's a fun little hobby though :)
-
Ok - got my app to inspect the Atari ST code and retrieve all offsets (except for woodtraps and gem specifics). My numbers agree with yours :)
Basically by taking the first x bytes of the known code and using that as a list of bytes to match against. So hopefully now I can point it at any BW binary and get those offsets out.
Next step there would be to allow it to work over many files... (for PC and BCF files)
However, I think next I'll get the floor offsets, scrolls, etc. that I'm not ingesting yet into my mapping tool.
-
my "searcher" program seems to be working, but it takes ages to process the file (then again, its running uncompiled on AMOS!)
i also obly checked 8 bytes, so i couldnt use the first 8 bytes (because of the triggers/switches being the same at that point mostly) but this has caused a problem with the "repetitive" pockets data being incorrectly found, so i will need to do a bit more investigation!
-
i've made and adapted my own thing to lookup offsets. It's not brilliant but it works!
attached is a zip file containing the output of a few processes... the amiga binaries and a couple of ST disk images, plus your CODE file.
The way i've set it up, means that when i have some BEXT BCF files, i'll be able to use the same program to search those binaries as well :)
-
That's excellent :)
In other news, my party of Murlock, Sethra, Sir Edward and Megrim are on their way out of the Serpent tower and heading to the Moon Tower. It's really strange being able to play Bloodwych with a map alongside. I'm also surprised at how much I still remember :)
-
Hooray! I've got my inspector reading all but the woodtrap locations (which is not a biggie for me at the moment anyway)
I've also got my mapper obeying offsets and determining the special floors. :)
Which are (Atari ST): K3, S3, M3, D7, C5, Z4. (Not determined what it means yet)
Next stop is the champion data - do you have any info on parsing these? The pockets seem straightforward, but the stats, spells and start positions look a little more tricky...
-
Scrolls
Assuming the first scroll in the list is ID=0.
Whenever you have a stone wall cell (AB CD, where D=1)
Cells with a scroll can be found using:
if ((args.AB >= 0x15) && (args.AB % 4 == 0x1))
{
DisplayState = EnumDisplayState.Message;
}
And the Scroll ID can be determined using:
ScrollID = (args.AB-0x15) >> 0x02;
Each tower then applies an offset to this scroll id to pick the correct scroll from the list.
I have figured these out and they are as follows:
The Keep = 0,
Serpent Tower = 0x15,
Moon Tower = 0x21,
Dragon Tower = 0x29,
Chaos Tower = 0x31,
Zendick's Tower = 0x3B
i.e. so scroll ID = 0 for the serpent tower would be 0x15...
-
Well I've been playing through and am now just at the start of Zendik's tower.
No idea what the significance of the special floors are, I guess without understand the source code we won't know.
Once I complete Bloodwych, I'll probably start on the data disk. At that time I'll get my mapper to extract the Extended levels too (hopefully)
-
Well, Bloodwych is complete, so now onto the data disks.
(Internally I seem to be naming the games B1 and B2, which reminds me of the Bananas In Pyjamas, but that's another story...).
So, in good news, it didn't take much effort to get my map ingestor working with the data disk. With some prior knowledge of the game (and that there are only 4 "towers"), plus some searching through the Hex source, I've found offsets for everything I needed.
(These numbers are all in decimal)
"OffsetMap": [
63920,
69042,
74164,
79286
],
"OffsetMonstersTotals": [
88180,
88182,
88184,
88186
],
"OffsetMonsters": [
88188,
88956,
89724,
90492
],
"OffsetObjects": [
68016,
73138,
78260,
83382
],
"OffsetScrollsBlock": 96358,
"OffsetSwitches": [
25832,
25896,
25960,
26024
],
"OffsetTriggers": [
31286,
31414,
31542,
31670
]
Most of the original bloodwych ingestion worked like a charm with this, but there are a few new monster types, switch types and trigger types which I haven't identified yet - must be new in BEXT. Details below.
It was bizarre to see there are only 2 wall scrolls, I'd forgotten they weren't really used in BEXT.
Monsters:
As we know, data of the form AB CD EF GH IJ KL, where A is the type.
I now have:
case 0x00:
Type = EnumType.Normal;
break;
case 0x01:
Type = EnumType.Spellcaster;
break;
case 0x02:
Type = EnumType.Drone;
break;
case 0x03:
Type = EnumType.DroneSpellcaster;
break;
case 0x04:
Type = EnumType.ArcBolt;
break;
case 0x08:
Type = EnumType.Unknown08;
break;
case 0x09:
Type = EnumType.Unknown09;
break;
I think Unknown08 and Unknown09 might relate to monsters who have to drop a specific object (e.g. a key), but not sure at this stage.
Triggers
For Triggers, some new types not see in the original...
case 0x00:
Type = EnumType.Unknown00;
break;
case 0x3E:
Type = EnumType.Unknown3E;
break;
case 0x40:
Type = EnumType.Unknown40;
break;
I'd bet one of these is to do with the "Summon Generation" trigger pads, but not sure yet.
Switches
For switches, i found quite a few new types:
case 0x00:
Type = EnumType.Unknown00;
break;
case 0x10:
Type = EnumType.Unknown10;
break;
case 0x12:
Type = EnumType.Unknown12;
break;
case 0x14:
Type = EnumType.Unknown14;
break;
case 0x16:
Type = EnumType.Unknown16;
break;
Not a clue what these do yet...
-
i think you are right about the monsters that drop stuff (i.e. keys or potions).
also look out for some strange 'types' of monsters... these would be spells (infernos) which 'launch' the moment you enter that particular floor/tower iirc.
nice to see all this progress! cant wait to know what all those extra gubbins are!
-
3/4 towers down, lots of interesting things discovered... :)
Monster types 8 and 9: as far as I can tell, 8 is normal, 9 is spellcaster. The main difference is they drop special objects when killed. This may be a key, or a specific weapon, or even a crystal (in the case of the behemoths that drop crystals)
As to what determines these objects in code, I have yet to ascertain...
I also have some findings on those unknown switches and triggers, but don't have the details at hand. The main new types are those "summon" generators and the switch walls that generate behind you. I also noticed a nice quirk in the moon tower of wood wall cells with no wood walls set - these stop the switch walls from spreading.
-
Hurrah! Data disks complete!
Summary of new types:
Monsters:
case 0x00:
Type = EnumType.Normal;
break;
case 0x01:
Type = EnumType.Spellcaster;
break;
case 0x02:
Type = EnumType.Drone;
break;
case 0x03:
Type = EnumType.DroneSpellcaster;
break;
case 0x04:
Type = EnumType.ArcBolt;
break;
case 0x08:
Type = EnumType.ObjectHolderNormal;
break;
case 0x09:
Type = EnumType.ObjectHolderSpellcaster;
break;
Triggers
case 0x3E:
Type = EnumType.GenerateSummons;
break;
case 0x40:
Type = EnumType.CreateSpecialWallWithSwitch;
break;
Switches
case 0x10:
Type = EnumType.OpenVoidLockDoor10;
break;
case 0x12:
Type = EnumType.SpecialWallWithSwitch12;
break;
case 0x14:
Type = EnumType.CreatePad14;
break;
case 0x16:
Type = EnumType.TogglePad16;
break;
-
Interesting stuff, always checking back for update on whats happening in the world of Bloodwych so good to see the work you are doing, I've just started learning VB .Net myself in the hope to make a level editor and maybe one day a remake.
Would it be possible to see your C# code as although I cant write it, it would be interesting to read though.
-
Hey There,
Since I completed the data disk, haven't had much time into refining this.
I need to refactor the code somewhat to make it a bit more understandable and modular. But then I would have no problem with posting code snippets to assist in demostrating logic.
Cheers
-
That would be good, as a newbie to programming I was interested how you read the file as a binary and how you do drawing off the map.
Also Have you thought about doing a process hook so you can read/write the data in realtime? I managed to do this using cheatengine (http://www.cheatengine.org/) but wanted to be able to make a map editor and character editor in VB
-
Hey, this site is back up :)
For reading binary, basically I read the file as a list of bytes and just accessed that list based off the known offsets.
For drawing the map, I'm just using some rudimentary system.drawing logic, not very efficient but does the trick. ;)
Unfortunately my time is incredibly limited at present, so may not be able to do much further work until near the end of year :(
-
Murlock,
Did you do any more work on this?
-
Well I've finally started work on my own stuff as I have never programmed or done 3D stuff progress is slow but what im doing is a Map Editor/Viewer which supported Bloodwych Files see below
(http://s18.postimage.org/eumnxvrmx/Blood_Map.png)
I cant work out how the orientation of the Wooden Walls/Doors is done yet but will look into that as its only a start (if anyone can help me out with this they are more than welcome). I've also started to write a Bloodwych remake engine the plan is to be able to support the Bloodwych map files and also my own type based on these I only started this a couple of days ago but progress is going alright im focusing on getting the engine to look right and have multiplayer support (local and network) so when your playing via the network its still split screen updating the window below like Bloodwych was when you played together on the same machine.
(http://s19.postimage.org/r4np56s1v/Blood_Rmk.png)
-
this is from memory, but i think the wood walls use a base-4 type scenario, and defined as N E S W
0 = no wall,
1 = plain wall
2 = open door
3 = closed door
so if we want North = plain wall, East South and West as nothing, we define that as; "1 0 0 0"
therefore in binary;
01 00 00 00
and in hex $40
i will have to check my notes / editor code, but it's definately something like that!!
-
That kind of makes sense 40,02 = West Facing Wall but there is loads which do not match the pattern i.e.
In the Keep - Level 4
Object Value: C402 X: 5 Y: 29
Object Value: 4702 X: 5 Y: 28
Object Value: 4002 X: 5 Y: 27
Object Value: 0102 X: 6 Y: 28
Object Value: 0D02 X: 8 Y: 28
-
i may have got the order wrong, in that it should be 'reversed'
bit 76 54 32 10
dir WW SS EE NN
if NN (bit 0/1) =00 then there is nothing to the north.
if SS (bit 4/5) = 11 then there is a closed door south
this code would 'type out' the type of wall (it's from my editor)
_DESC2:
LINE$(10)="WOODEN STRUCTURE "
EE=(AA*16)+BB
LINE$(11)="N: "
LINE$(12)="E: "
LINE$(13)="S: "
LINE$(14)="W: "
For COT=0 To 3
BOT=COT*2
If Btst(BOT,EE)=False and Btst(BOT+1,EE)=False : Rem %%% NOTHING
LINE$(11+COT)=LINE$(11+COT)+"NOTHING "
Else If Btst(BOT,EE)=True and Btst(BOT+1,EE)=False : Rem %%% WALL
LINE$(11+COT)=LINE$(11+COT)+"WOOD WALL "
Else If Btst(BOT,EE)=False and Btst(BOT+1,EE)=True : Rem %%% OPEN DOOR
LINE$(11+COT)=LINE$(11+COT)+"OPEN DOOR "
Else If Btst(BOT,EE)=True and Btst(BOT+1,EE)=True : Rem %%% CLOSED
LINE$(11+COT)=LINE$(11+COT)+"CLOSED DOOR "
End If
Next COT
If Btst(0,CC)=True
LINE$(15)="LOCKED"
Else
LINE$(15)="NO LOCK"
End If
For COT=10 To 15
OPTION(COT)=True
Next COT
Return
the two bytes of map-data are read-in as AA BB CC DD iirc. (again, maybe in reverse order)
-
Object Value: C402 X: 5 Y: 29
$c4 = 11 00 01 00
north = nothing
east = wall
south = nothing
west = closed door?
is this right? (i dont have the map here!)
-
Sorry you where right :)
Cheers :)
Now Doors / Banners :)
Then Objects :s
-
if i had time to verify everything, i would use what i am posting here to update the general hacking information thread.... :/ sorry about that!
here is the similar code for metal doors, although i can see already bad working practices in it!! (oh how i have learnt so much since!)
_DESC5:
LINE$(10)="METAL DOOR "
OPTION(11)=True
OPTION(12)=True
OPTION(13)=True
OPTION(14)=True
If BB mod 4=0 or BB mod 4=1
LINE$(11)="REGULAR "
Else If BB mod 4=2 or BB mod 4=3
LINE$(11)="PORTCULLIS "
End If
If BB mod 2=0
LINE$(13)="OPEN "
Else If BB mod 2=1
LINE$(13)="CLOSED "
End If
If(BB=>0 and BB<=3) or(BB=>8 and BB<=11)
LINE$(12)="NORTH / SOUTH "
Else If(BB=>4 and BB<=7) or(BB=>12 and BB<=15)
LINE$(12)="EAST / WEST "
End If
If CC mod 4=1
LINE$(14)="LOCKED"
LOCKED=1
OPTION(15)=True
Else If AA=0
LINE$(14)="UNLOCKED"
LINE$(15)=""
LOCKED=0
Else If AA>0 and(CC mod 4<>1)
LINE$(14)="TRICK LOCK"
OPTION(15)=True
End If
If AA=0 and LOCKED=1 : LINE$(15)="MAGELOCKED"
Else If AA=1 : LINE$(15)="BRONZE LOCK"
Else If AA=2 : LINE$(15)="IRON LOCK"
Else If AA=3 : LINE$(15)="SERPENT LOCK"
Else If AA=4 : LINE$(15)="CHAOS LOCK"
Else If AA=5 : LINE$(15)="DRAGON LOCK"
Else If AA=6 : LINE$(15)="MOON LOCK"
Else If AA=7 : LINE$(15)="CHROMATIC LOCK"
Else If AA>=8 : LINE$(15)="INVALID LOCK"
End If
If BB>=8
LINE$(14)="LOCKED"
LINE$(15)="VOID LOCK"
OPTION(15)=True
End If
Return
i have now seen (been reminded!) that of the two bytes for each map-date piece, it's broken down as
AA BB CC DD
with DD as the 'type' (i.e. 0-7) and AA BB being the main parameters for each type.
with this, it seems that CC (like with many types) is used to define if the door is locked, and if there is any occupancy in that space (e.g. objects, people, spells)
For example, some of this works as follows;
for BB
Bytes: % 3210
Byte 0 = open / close
Byte 1 = regular metal / portcullis
byte 2 = north&south facing / east&west facing
byte 3 = black-locked
with AA this define what type of lock is used
if AA = 0 (and regular lock type from CC) , door is magelocked
AA=0 -> 7 = bronze, iron, serpent, chaos, dragon, moon, chromatic... all else invalid locks.
-
for walls (banners etc) i am going to give you the code, but let you decipher for yourself!!
If you can please post a nice concise understanding of it here though,i will ultimately use it to update the main info thread as mentioned above :)
_DESC1:
OPTION(11)=True
LINE$(10)="STONE WALL"
If CC=>8 : Rem - has something on it
OPTION(12)=True
If CC=$8 or CC=$C : Rem - north
LINE$(11)="FACING NORTH "
Else If CC=$9 or CC=$D : Rem - east
LINE$(11)="FACING EAST "
Else If CC=$A or CC=$E : Rem - south
LINE$(11)="FACING SOUTH "
Else If CC=$B or CC=$F : Rem - west
LINE$(11)="FACING WEST "
End If
If BB mod 4=0 : Rem Shelf
LINE$(12)="SHELF "
Else If BB mod 4=1 : Rem Sign
Rem --- sign colours
LINE$(12)="SIGN"
OPTION(13)=True
If AA=0 and BB=1 : Rem - seed sign
LINE$(13)="GENERATED COLOUR"
Else If AA=0 and BB=5 : Rem - serp sign
LINE$(13)="SERPENT "
Else If AA=0 and BB=9 : Rem - dragon sign
LINE$(13)="DRAGON "
Else If AA=0 and BB=13 : Rem - moon sign
LINE$(13)="MOON "
Else If AA=1 and BB=1 : Rem - chaos sign
LINE$(13)="CHAOS "
Else If BB mod 4=1 : Rem -- remainders are wall text
LINE$(13)="SCROLL"+Str$((((AA*16)+BB)/4)-4)
End If
Else If BB mod 4=2 : Rem Switch
OPTION(13)=True
If(AA*16)+BB>=$7
LINE$(12)="SWITCH"
If BB=$2 or BB=$A : Rem switch unused
LINE$(14)="LIT"
Else If BB=$6 or BB=$E : Rem switch used
LINE$(14)="DIM"
End If
OPTION(14)=True
EE=(AA*16)+BB
SWITCH=(EE/8)-0
LINE$(13)="REFERENCE:"+Str$(SWITCH)
' now lets talk about what the switch does
SWITCHTYPE= Fn SWITCHTYPE
If SWITCHTYPE=0 : RESULT$="NO EFFECT"
Else If SWITCHTYPE=2 : RESULT$="REMOVE"
Else If SWITCHTYPE=4 : RESULT$="TOGGLE WALL"
Else If SWITCHTYPE=6 : RESULT$="OPEN METAL DOOR"
Else If SWITCHTYPE=8 : RESULT$="ROTATE WALL"
Else If SWITCHTYPE=10 : RESULT$="TOGGLE PILLAR"
Else If SWITCHTYPE=12 : RESULT$="PLACE PILLAR"
Else If SWITCHTYPE=14 : RESULT$="ROTATE WOODEN"
Else RESULT$="INVALID"
End If
LINE$(16)=RESULT$
LINE$(17)="X: "+Right$(Hex$( Fn SWITCHX+$100),2)+" Y: "+Right$(Hex$( Fn SWITCHY+$100),2)
OPTION(16)=True
OPTION(17)=True
Else : LINE$(12)="SWITCH"
LINE$(13)="EMPTY" : End If
Else If BB mod 4=3 : Rem crystal hole
LINE$(12)="SOCKET"
OPTION(13)=True
OPTION(14)=True
EE=(AA*16)+BB
If EE>=$0 and EE<=$7 : Rem -- serpent
LINE$(13)="SERPENT"
Else If EE>=$8 and EE<=$F : Rem -- chaos
LINE$(13)="CHAOS"
Else If EE>=$10 and EE<=$17 : Rem -- dragon
LINE$(13)="DRAGON"
Else If EE>=$18 and EE<=$1F : Rem -- moon
LINE$(13)="MOON"
Else If EE>=$20 and EE<=$27 : Rem -- grey
LINE$(13)="GREY"
Else If EE>=$28 and EE<=$2F : Rem -- bluish
LINE$(13)="BLUISH"
Else If EE>=$30 and EE<=$37 : Rem -- brown
LINE$(13)="BROWN"
Else If EE>=$38 and EE<=$3F : Rem -- tan
LINE$(13)="TAN"
Else If EE>=$40 and EE<=$FF : Rem -- everything else that isnt
LINE$(13)="INVALID"
End If
If BB=$3 or BB=$B : Rem --- if full
LINE$(14)="FULL"
Else If BB=$7 or BB=$F : Rem -- if empty
LINE$(14)="EMPTY"
End If
End If
Else
LINE$(11)="PLAIN"
End If
Return
p.s. hope all this helps - and have fun :)
-
Cheers Horace :)
Didnt ever know if you where still around on the forums as you have been quiet for some time.
-
Cheers Horace :)
Didnt ever know if you where still around on the forums as you have been quiet for some time.
No problem!
I have a lot going on in the "real world" atm that is preventing me from being involved in anything new, but i will always check this forum and chip-in where i can :)
edit:
thanks to my earlier post, i updated the wooden walls data definition:
http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39946.html#msg39946
-
Wood Walls and Banners/Switches now supported just need Gem Slots and to sort out the Doors now..
(http://s19.postimage.org/72asuxrb3/Blood_Map.png) (http://postimage.org/image/72asuxrb3/)
(http://s19.postimage.org/ejolnw67z/Blood_Map2.png) (http://postimage.org/image/ejolnw67z/)
-
you appear to be missing a number of banners? (i.e. those which have automatic colouring)
http://www.ultimateamiga.co.uk/HostedProjects/GamesCorner/TheCryptOfBloodwych/Maps/bw_keep_3.jpg
^^ look in th efirst main room after the recruitment area... there are various signs i have shown as 'white' which are generated at random (or rather, seeded "random"...)
here is my stone-wall code, which includes banners and gems etc.
like before, it probably uses some modulo functions where it would be better using bit-test functions... sorry about that!
to read about how switches are defined, see also here:
http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39953.html#msg39953
because the below code refers to a function which reads-in the relevant switch operation.
_DESC1:
OPTION(11)=True
LINE$(10)="STONE WALL"
If CC=>8 : Rem - has something on it
OPTION(12)=True
If CC=$8 or CC=$C : Rem - north
LINE$(11)="FACING NORTH "
Else If CC=$9 or CC=$D : Rem - east
LINE$(11)="FACING EAST "
Else If CC=$A or CC=$E : Rem - south
LINE$(11)="FACING SOUTH "
Else If CC=$B or CC=$F : Rem - west
LINE$(11)="FACING WEST "
End If
If BB mod 4=0 : Rem Shelf
LINE$(12)="SHELF "
Else If BB mod 4=1 : Rem Sign
Rem --- sign colours
LINE$(12)="SIGN"
OPTION(13)=True
If AA=0 and BB=1 : Rem - seed sign
LINE$(13)="GENERATED COLOUR"
Else If AA=0 and BB=5 : Rem - serp sign
LINE$(13)="SERPENT "
Else If AA=0 and BB=9 : Rem - dragon sign
LINE$(13)="DRAGON "
Else If AA=0 and BB=13 : Rem - moon sign
LINE$(13)="MOON "
Else If AA=1 and BB=1 : Rem - chaos sign
LINE$(13)="CHAOS "
Else If BB mod 4=1 : Rem -- remainders are wall text
LINE$(13)="SCROLL"+Str$((((AA*16)+BB)/4)-4)
End If
Else If BB mod 4=2 : Rem Switch
OPTION(13)=True
If(AA*16)+BB>=$7
LINE$(12)="SWITCH"
If BB=$2 or BB=$A : Rem switch unused
LINE$(14)="LIT"
Else If BB=$6 or BB=$E : Rem switch used
LINE$(14)="DIM"
End If
OPTION(14)=True
EE=(AA*16)+BB
SWITCH=(EE/8)-0
LINE$(13)="REFERENCE:"+Str$(SWITCH)
' now lets talk about what the switch does
SWITCHTYPE= Fn SWITCHTYPE
If SWITCHTYPE=0 : RESULT$="NO EFFECT"
Else If SWITCHTYPE=2 : RESULT$="REMOVE"
Else If SWITCHTYPE=4 : RESULT$="TOGGLE WALL"
Else If SWITCHTYPE=6 : RESULT$="OPEN METAL DOOR"
Else If SWITCHTYPE=8 : RESULT$="ROTATE WALL"
Else If SWITCHTYPE=10 : RESULT$="TOGGLE PILLAR"
Else If SWITCHTYPE=12 : RESULT$="PLACE PILLAR"
Else If SWITCHTYPE=14 : RESULT$="ROTATE WOODEN"
Else RESULT$="INVALID"
End If
LINE$(16)=RESULT$
LINE$(17)="X: "+Right$(Hex$( Fn SWITCHX+$100),2)+" Y: "+Right$(Hex$( Fn SWITCHY+$100),2)
OPTION(16)=True
OPTION(17)=True
Else : LINE$(12)="SWITCH"
LINE$(13)="EMPTY" : End If
Else If BB mod 4=3 : Rem crystal hole
LINE$(12)="SOCKET"
OPTION(13)=True
OPTION(14)=True
EE=(AA*16)+BB
If EE>=$0 and EE<=$7 : Rem -- serpent
LINE$(13)="SERPENT"
Else If EE>=$8 and EE<=$F : Rem -- chaos
LINE$(13)="CHAOS"
Else If EE>=$10 and EE<=$17 : Rem -- dragon
LINE$(13)="DRAGON"
Else If EE>=$18 and EE<=$1F : Rem -- moon
LINE$(13)="MOON"
Else If EE>=$20 and EE<=$27 : Rem -- grey
LINE$(13)="GREY"
Else If EE>=$28 and EE<=$2F : Rem -- bluish
LINE$(13)="BLUISH"
Else If EE>=$30 and EE<=$37 : Rem -- brown
LINE$(13)="BROWN"
Else If EE>=$38 and EE<=$3F : Rem -- tan
LINE$(13)="TAN"
Else If EE>=$40 and EE<=$FF : Rem -- everything else that isnt
LINE$(13)="INVALID"
End If
If BB=$3 or BB=$B : Rem --- if full
LINE$(14)="FULL"
Else If BB=$7 or BB=$F : Rem -- if empty
LINE$(14)="EMPTY"
End If
End If
Else
LINE$(11)="PLAIN"
End If
Return
-
Good spot, missed that group off..
(http://s19.postimage.org/yxtioigzn/Bloodwish.png)
Also I'm looking into a new features which will let me support real time map editing in the Amiga/ST Versions when played on a emulator, also this will allow you to have a real time map which updates the player location :)
-
Real Time Player tracking is now working for Steem, I need to clean up the code and decide if I want to read all of the ST Memory so I can update the whole map (open/closed doors etc) then once this is all working nicely I'll add in a WinUAE option
(http://s19.postimage.org/6xpvnlugz/Steem_Mapping_Bloodwych.png)
-
I got real time tracking working with winaue also.
Horace any chance you want to share the source of your editor so I can try add all the same features as I can see any reason why I cant make my version support the same + Atari/PC support also
Had to add the option for winuae to rescan the memory as sometimes it will find the data in the HDD image not the live game :)
(http://s19.postimage.org/vwiectg8z/Win_UAE_Support.png)
-
Not sure if you guys know this but the Level Data is always 5122 Bytes apart
So MOD0 + 5122 bytes = Serpent Tower Data
This is the same on Atari and Amiga
-
So live mapping is now working (read-only atm) but with the live data there is some different values have you seen these two before?
01 43
00 43
This is normally a Pillar/Bed but do you know what that 4 Means?
Also do you have any code for the different doors as I still need to fix this.
*UPDATE* Ah its cause there is items on the floor :)
-
i will upload the source for you when i get the chance :)
looking at map data as AB CD, in your example C = 4
bits are switched on/off in this value, to show what is 'occupying' the space. You will generally only see this when you look at 'live' data
this is entirely from memory, but i *think*....
bit 0 (usually C=1 ) is used for a door lock , bit 2 (C=4) you have just said is an object.... bit 3 (C=8) iirc is a person (monster or champion).... that leaves bit 1 (C=2) must be a spell in that location. e.g. a fireball that has been launched or similar.
obviously these can then also be combined if several of them are true!
what door code are you after? metal doors? i posted this already... :)
http://www.ultimateamiga.co.uk/index.php/topic,8806.msg44182.html#msg44182
-
Cheers again Horace not sure how i missed the door code :s
Doors now working :)
-
as promised, attached is the editor source code. i cant be 100% it's the latest version, but it has all of the map code (converting to both text, and AMOS style draw commands) in it.
I'm not sure there is much 'extra' to learn from it... i did try to create a "standard" input/output file format that could potentially become cross-platform (the .map and .obj files that it creats be substituted directly into the PC version for example) ... however, when i tried to put my new levels into the ST version, it didnt want to run - this may be related to the ST versions copy protection though.
to answer your other PM question MM, i have never created any remake of BW... i have some basic wall-drawing code for AMOS but i didnt write it, and it isnt very good tbh!! (for map/editing purposes your ST/UAE plugins are far better!!)
anyway, hope this helps, and sorry for the delay - kind of been busy of late!!
-
Thanks Horace, just taken a quick read though and its amazing how similar we have done things but I cant see any code about objects in the file which is what I was really after
Also did you have any of the graphics still like monster images/object images as all the ones I ripped where from the PC version so will need to redo them otherwise, any chance in your Bloodwych font saves me having to make it :)
I'll make sure your not reading any map data I've missed and then on that side we should be same, do you happen to know if the location of the Bloodwych data is different in different version of the Amiga disks/HDD version as I'll try add support to import/export as many formats as possible
-
there should be object code, because there is an object stack-editor and a way to place them.
they arent 'displayed' on the map though. i will check the most recent code for you later today if i can :)
there are definitely two different versions on the amiga, with different offsets for the data - i think they may have fixed a couple of bugs (like where some objects on the floor are displayed in front of the wrong things, like people's feet) .... not sure of the exact changes!!
i will also dig out the monster /character graphics for you, but the font was made by my brother, so i may have to simply 'draw' all of the font-characters onto an image file for you!!
-
ok, so it seems you were right - that code was quite out of date!
I have exported the code from the last version i had - you should find this more useful i hope!!
edit: all monsters / characters .gif attached also ;)
edit 2: and now the font image thing .... although you may need a 'pure' output, this is just a screen grab from OSX... let me know if you need it 'clean'
-
Real Time Player tracking is now working for Steem, I need to clean up the code and decide if I want to read all of the ST Memory so I can update the whole map (open/closed doors etc) then once this is all working nicely I'll add in a WinUAE option
Any chance you can open source your map viewer code? Don't worry about clean up, everyone's code is a mess ;) It'd be another resource for the rest of us that might want to spend time bringing Bloodwyche out of the darkness. I know I've tried disassembling it several times, but unfortunately given up.
-
Any chance you can open source your map viewer code? Don't worry about clean up, everyone's code is a mess ;) It'd be another resource for the rest of us that might want to spend time bringing Bloodwyche out of the darkness. I know I've tried disassembling it several times, but unfortunately given up.
shame - i'd love to see the Amiga Bloodwych fully disassembled and commented.... then we could start thinking about adding new functions, switches, objects, etc :D
i can dream!!
-
shame - i'd love to see the Amiga Bloodwych fully disassembled and commented.... then we could start thinking about adding new functions, switches, objects, etc :D
i can dream!!
10 grand and I'll disassemble it and provide you with portable C source code. Maybe even an Android port. Too many other things to do, otherwise.
-
10 grand and I'll disassemble it and provide you with portable C source code. Maybe even an Android port. Too many other things to do, otherwise.
if i had that kind of money lying around, i'd take the time off work and do it myself ;)
And FYI, C version, or android ports arent really what interest me.... i'd still be after creating an enhanced amiga version!
-
Hey there,
It's been a long time since I logged in, but I'm back! :)
Unfortunately I haven't updated my source code, real life smashed this project onto the backburner... :(
If it's of any use, I'll try and dig out the source files in case the logic helps, although most of my stuff came from Horace originally anyway! I may have some extra cases for the ST Data Disk though.
MadMunky - all the real time tracking stuff looks great, I never ever got that far.
My goal was originally a rewrite for windows using .NET, but my grand plans were heavily cut back to a map-viewer that I could use as a guide when playing. Once I completed Bloodwych and Data Disk I pretty much ceased development.
-
Hi Murlock,
The source code would be grate so I could try see how to clean up my code and just see how we both did bits.
Is there any chance in you still doing a .NET version I would love to see the game remade in .NET if its something I could help you with Id be more than happy to try help.
-
I cant wait to have my house decorated so i can settle-down into working on some Bloodwych maps again!
-
Yep I'll try and dig out the code soon, hopefully by this weekend.
If I recall correctly I hadn't refactored or optimised it so it was pretty rough at times.
My dream is still to port/remake the game using .NET, but I just don't have the time to devote at the moment :(
-
Here is my attached code. Also contains the Atari ST source files that the code will parse.
Not cleaned up or anything, but enough to show you how I did it.
You'll need Visual Studio .NET 2010 to be able to open/run it.
Although with any text editor you can go digging through source.
-
I suppose I should give you some instruction, given the UI is not logical... ;)
Important files of note are configs B1.json and B2.json, which contain all configurable file offsets and path settings to ingest the data from the binaries.
(where B2 is the data disk)
Currently looks for:
C:\SourceControl\Experiments\BloodwychDataTool\BloodwychDataTool\Data\B1\AtariST\
C:\SourceControl\Experiments\BloodwychDataTool\BloodwychDataTool\Data\B2\AtariST\
So you'll have to modify these JSONs if you extract elsewhere.
When you run the program, you'll see a screen like this. (1.png)
For test purposes, all you need to care about are the buttons:
Ingest from B1.config
Ingest from B2.config
These buttons will attempt to:
- load the B1/B2 assembly
- lookup byte offsets from the config
- parse the file appropriately
They work pretty much in the same way, except B2 has more cases for the extra functionality added.
Whether you chose B1 or B2, if successful you'll see a message like: InitFromConfig completed at 1/10/2012 3:08:18 PM
Then you should go to the Map tab and click "Process Map".
If this suceeds, you should see something like:
Processing World Data...
Processing Tower Data...
Processing Tower: The Keep
Processing Tower: Serpent Tower
Processing Tower: Moon Tower
Processing Tower: Dragon Tower
Processing Tower: Chaos Tower
Processing Tower: Zendick's Tower
Refreshing Dropdowns
Done
And a screenshot like 2.png
Now you can pick your floor from the dropdown and select GO. The up and down arrow buttons allow for quick floor switching.
You can click any map cell for extra status information.
Final screenshot = 3.png
If you have any questions/problems feel free to ask :)
-
Real Time Player tracking is now working for Steem, I need to clean up the code and decide if I want to read all of the ST Memory so I can update the whole map (open/closed doors etc) then once this is all working nicely I'll add in a WinUAE option
Any chance you can open source your map viewer code? Don't worry about clean up, everyone's code is a mess ;) It'd be another resource for the rest of us that might want to spend time bringing Bloodwyche out of the darkness. I know I've tried disassembling it several times, but unfortunately given up.
I'll continue work on this hopefully soon and try clean some stuff up and soon as its a little nicer and cleaner i'll be happy to release the code.
-
Hi All,
Though id pop by and say happy new year and let everyone know im still working on bits for Bloodwych.
Currently (Yesterday) I started working on making my map editor/viewer again this time in Java, as I've got to learn Java at the same time its slow progress as I'm still a newbie programming with but its starting to take shape.
I'm hoping to add in the Real Time emulator tracking and also full editing but one step at a time least now its multi-platform friendly thanks to Java..
(http://s14.postimage.org/kpljy76pd/Blood_Mapper.png)
-
Horrace: You still around?
1. I just wanted to say thanks all the hacking information on this site makes it so much easier to remember how things are structured in Bloodwych
2. You still run a Mac? If so do you ever run any ST emulators (can you let me know the name of them) also whats the Amiga emulator you run?
-
Horrace: You still around?
i am!! still reading, just not always got time to post!
1. I just wanted to say thanks all the hacking information on this site makes it so much easier to remember how things are structured in Bloodwych
no problems - i like being asked about stuff i havent yet typed up, or added to the site, because it encourages me to finally update the "missing" info on here!
2. You still run a Mac? If so do you ever run any ST emulators (can you let me know the name of them) also whats the Amiga emulator you run?
I do still run a couple of macs!
For Amiga, I am msotly using FS-UAE these days on both Mac and PC (i.e. when at work) because of the cross-over. Very occassionly i revert back to e-uae, but as fs-uae improves, that happens less and less. .... plus you can play fs-uae online against a friend which is perfect for Bloodwych.
For the ST emulator, i have run NoSTalgia and Hitari i think... but neither for very long, so i cant really recommend one!!
Do ful edits work on the ST version of Bloodwych? I tried importing my map data into the ST version once (i think i patched a disk image?) but it didnt work... I was wondering if the ST had a WHDload style HD installer, that would negate any copy-protection and therefore make edits easier?
-
I'll take a look see if I can add some Mac Real-Time support.
As for the ST I've only currently read the data in real time which is via Steem's memory and that was backwards so I wrote a function to convert it then back again.
Not really got as far as any editing but once I have things a little further with my app i'll look into editing the actual files, im sure I've look with a file editor at the ST Files and there where pretty similar to the Amiga/ST but never looked into the copy protection side of things, I know when I'm reading the data from memory its all the same :)
-
Horrace:
I've had a quick play and it seems for the ST the best way of doing it is to modify the save files as these contain all the data, I can send you one if you dont have one, I've tried it out and it lets you make changes to the map and then they are there when loaded.
If you have the map you where trying to apply send me a copy and ill try put it in the ST save file.
On the PC I use the following tools
Steem - Atari ST emulator
Medway Boys 38 Disk - This is hacked and the save function works where as it does not in the automation disk
MSA Converter 2.1 - This lets you load up disk images and change files etc...
Also before you have said you have to modify some of the 68000 code to get the maps to work, what parts are you changing and why?
-
Horrace:
Bit of a Request please...
Can you post up a line of HEX (say 8 bytes) of the following info so I can use it to find the locations..
Bloodwych Characters
DataDisk Characters
DataDisk Level Start Data (I guess the tower sizes etc...)
If you have any more details on the DataDisk like the towers ripped that would be handy :)
-
I have attached all my unmodified datafiles (attached in ZIP) - you will want the champions.stats and champions.pockets files.... these are files exported from my editor
The ones that include some asm code is the wood-trap ... these are the two wooden doors that close behind you where the gems-holes are in The Keep... designed to limit the area to one player only i guess, but the locations where they close another door is hard-coded into the asm. (they refer to specific x/y coords on the map) ... sometimes i think it would be easier to simply not use this function!
As for the datadisk... eek! now you're asking! I will have to dig through some old code, but it's stuff that is easiest to find using a live editor (i.e. Action Replay III) and the trainer functions!
i have definitely extracted data disk info before in order to create my maps....
http://www.ultimateamiga.co.uk/index.php/topic,8815.0.html
-
Very nice :)
Can your editor not export clean files like this for the Data Disk?
Problem I've got is I dont seem to be able to run the DD on Steem (I find Steem quicker and easier than the WinUAE :)) I'll keep playing away...
I've attached a memory dump of Bloodwych on the ST if you can find the HEX Values for the woodtrap stuff we can have a look see if it matches on the ST
-
the version of my editor which has the export files function doesnt have support for extended levels files unfortunately, i am trying to find the old version that i used to create the maps, but i'm not having much luck at the mo!
-
found it, but i seem to have hard-coded the floor sizes etc.
this is the portion of the code that specifies the offsets for the tower (dungeon) data, and i have attached the BEXT binary file where you use these offsets:
Rem --- 63312 from bext
Rem --- 1358 from bextsave
Rem ---- offset -61954
BXT_MAX:
Data 7,5,8,8
BXT_SERPENT:
Data 7,5,63312
Data 17,17,63382
Data 17,17,63960
Data 17,17,64538
Data 23,23,65116
Data 19,19,66174
Data 15,15,66896
Rem -- 1088 between
BXT_CHAOS:
Data 21,21,68434
Data 21,21,69316
Data 21,21,70198
Data 21,21,71080
Data 15,15,71962
BXT_MOON:
Data 7,9,73556
Data 17,17,73682
Data 17,17,74260
Data 17,17,74838
Data 17,17,75416
Data 17,17,75994
Data 15,17,76572
Data 15,17,77082
BXT_DRAGON:
Data 21,21,78678
Data 17,17,79560
Data 15,15,80138
Data 15,15,80588
Data 12,12,81038
Data 12,12,81326
Data 12,12,81614
Data 12,12,81902
Rem -- 17x17 (578) -- 21x21 (882) -- 15x15 (450)
Rem -- 19x19 (722) -- 23x23(1058) -- 15x17 (510)
-
Nice! Just did a quick hack to rip out the maps from memory and they loaded strait up in my editor.
-
Horrace:
I have the following configured to search for the level, where as in BW all the levels where 4096 in the DataDisk there are different lengths
private int [] ExtSerpTower = {7,17,17,17,23,19,15,0,4104};
private int [] ExtChaosTower = {21,21,21,21,15,0,0,4103};
private int [] ExtMoonTower = {7,17,17,17,17,17,15,15,4096};
private int [] ExtDragonTower = {21,17,15,15,12,12,12,4103};
I use this data, the first 7 bytes to work out the offsets of the maps then read in the amount of bytes needed for that level, I've also been coming across a new hex code see below
990A
-
Made it so my Viewer can dump memory and images to files now, here the Datadisk from my save game :)
(http://s19.postimage.org/wkorwnce7/Ext_Chaos_Tower_Level_0.png) (http://postimage.org/image/wkorwnce7/)
(http://s19.postimage.org/97qqe4wan/Ext_Chaos_Tower_Level_1.png) (http://postimage.org/image/97qqe4wan/)
(http://s19.postimage.org/3wgvqjgjj/Ext_Chaos_Tower_Level_2.png) (http://postimage.org/image/3wgvqjgjj/)
(http://s19.postimage.org/ohvnify4f/Ext_Chaos_Tower_Level_3.png) (http://postimage.org/image/ohvnify4f/)
(http://s19.postimage.org/5e2c23la7/Ext_Chaos_Tower_Level_4.png) (http://postimage.org/image/5e2c23la7/)
(http://s19.postimage.org/bt1cyrrzz/Ext_Dragon_Tower_Level_0.png) (http://postimage.org/image/bt1cyrrzz/)
(http://s19.postimage.org/4ec16e64f/Ext_Dragon_Tower_Level_1.png) (http://postimage.org/image/4ec16e64f/)
(http://s19.postimage.org/nxgmfr4vz/Ext_Dragon_Tower_Level_2.png) (http://postimage.org/image/nxgmfr4vz/)
(http://s19.postimage.org/qg2bgfqm7/Ext_Dragon_Tower_Level_3.png) (http://postimage.org/image/qg2bgfqm7/)
(http://s19.postimage.org/cnnwkszun/Ext_Dragon_Tower_Level_4.png) (http://postimage.org/image/cnnwkszun/)
(http://s19.postimage.org/noj1pts3j/Ext_Dragon_Tower_Level_5.png) (http://postimage.org/image/noj1pts3j/)
(http://s19.postimage.org/d2z6dtlrz/Ext_Dragon_Tower_Level_6.png) (http://postimage.org/image/d2z6dtlrz/)
(http://s19.postimage.org/u4wltnq0v/Ext_Dragon_Tower_Level_7.png) (http://postimage.org/image/u4wltnq0v/)
(http://s19.postimage.org/8704zvb0f/Ext_Moon_Tower_Level_0.png) (http://postimage.org/)
(http://s19.postimage.org/lp71c5n5r/Ext_Moon_Tower_Level_1.png) (http://postimage.org/image/lp71c5n5r/)
(http://s19.postimage.org/wdasazx4v/Ext_Moon_Tower_Level_2.png) (http://postimage.org/image/wdasazx4v/)
(http://s19.postimage.org/wrc4alh8f/Ext_Moon_Tower_Level_3.png) (http://postimage.org/image/wrc4alh8f/)
(http://s19.postimage.org/rtyjphf9b/Ext_Moon_Tower_Level_4.png) (http://postimage.org/image/rtyjphf9b/)
(http://s19.postimage.org/c9r5yy54v/Ext_Moon_Tower_Level_5.png) (http://postimage.org/image/c9r5yy54v/)
(http://s19.postimage.org/cnshyjp8f/Ext_Moon_Tower_Level_6.png) (http://postimage.org/image/cnshyjp8f/)
(http://s19.postimage.org/70w512mpr/Ext_Moon_Tower_Level_7.png) (http://postimage.org/image/70w512mpr/)
(http://s19.postimage.org/g9ybb6vlr/Ext_Serpent_Tower_Level_0.png) (http://postimage.org/)
(http://s19.postimage.org/mp1xllrpb/Ext_Serpent_Tower_Level_1.png) (http://postimage.org/image/mp1xllrpb/)
(http://s19.postimage.org/hrod0hpq7/Ext_Serpent_Tower_Level_2.png) (http://postimage.org/image/hrod0hpq7/)
(http://s19.postimage.org/meuf29d33/Ext_Serpent_Tower_Level_3.png) (http://postimage.org/image/meuf29d33/)
(http://s19.postimage.org/n5n581fgf/Ext_Serpent_Tower_Level_4.png) (http://postimage.org/image/n5n581fgf/)
(http://s19.postimage.org/lf446jxxb/Ext_Serpent_Tower_Level_5.png) (http://postimage.org/image/lf446jxxb/)
(http://s19.postimage.org/3qcdexm67/Ext_Serpent_Tower_Level_6.png) (http://postimage.org/image/3qcdexm67/)
-
Here's all the Bloodwych Maps..
http://postimage.org/gallery/66emnftc/fd62c12a/
And again the DataDisk so that you can see what levels what..
http://postimage.org/gallery/6e0zwpqo/
-
Just popping by to say Hi - I do read these forums from time to time :)
Looks like you're making some great progress.
The code I attached a few months back might help with some of the data disk, as I recall having to add some extra cases to cover some of the new things introduced.
Also I think I stored a number of byte patterns as constants for initially looking up the data.
-
Thanks Murlock I'll take a look I've not really looked much at your code as yet..
I've made mine as your see in the above code to search for the level widths as this is the same in all versions so I just do + and - from Mod0 to work out where the data is I need to change.
Thinking about it I may need to change this so when people are creating there own maps it still works but that is a easy change later on.
One thing which is making my life harder is that when Using Steem everything is held in memory backwards so I have to keep converting my array backwards and forwards did you manage any way round this?
Also I know bloodwyn is the first character in the dataset but does anyone have a list of which order the other characters come in (just me being lazy :))
EDIT : Im pretty sure this is the right order for the characters...
public enum Characters {
Blodwyn_StoneMaiden, Murlock_Darkheart, Elanore_Of_Avalon, Rosane_Swifthand,
Astroth_Slaemworth, Zothen_Runecaster, Baldrick_The_Dung, Elfric_Falendor,
Sir_Edward_Lion, Megrim_of_Moonwych, Sethra_Bhoaghail, Mr_Flay_Sepulcrast,
Ulrich_Sternaxe, Zastaph_Mantric, Hengist_Meldanash, Thai_Chang_of_Yinn
}
-
Horrace: Fancy updating the Champion Data
Hacking : Bloodwych Champion Data Structure
Each Character is 32 Bytes below is an example for Blodwyn (data is shown in Decimal)
A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z,AA,BB,CC,DD,EE,FF
1,35,17,13,13,35,35,31,31,6,9,5,128,0,0,0
199,255,0,0,0,0,7,9,2,0,3,0,0,0,0,0
Data in HEX
01 23 11 0d 0d 23 23 1f 1f 06 09 05 80 00 00 00
c7 ff 00 00 00 00 07 09 02 00 03 00 00 00 00 00
A = 1 (LEVEL)
B = 35 (STRENGH)
C= 17 (AGILITY)
D = 13 (INTELLIGENCE)
E = 13 (CHARISMA)
F = 35 (HIT POINTS)
G = 35 (HIT POINTS)
etc...
I know what the values are up to 'L' What's stored in the bytes after?
-
A,B,C,D, E,F,G,H
I,J,K,L, M,N,O,P
Q,R,S,T, U,V,W,X
Y,Z,AA,BB, CC,DD,EE,FF
1,35,17,13, 13,35,35,31
31,6,9,5, 128,0,0,0
199,255,0,0, 0,0,7,9
2,0,3,0, 0,0,0,0
Data in HEX
01 23 11 0d 0d 23 23 1f
1f 06 09 05 80 00 00 00
c7 ff 00 00 00 00 07 09
02 00 03 00 00 00 00 00
Full break down as far as i remember….
A = 1 (LEVEL)
B = 35 (STRENGTH)
C= 17 (AGILITY)
D = 13 (INTELLIGENCE)
E = 13 (CHARISMA)
F = 35 (HIT POINTS)
G = 35 (MAX HIT POINTS)
H = 35 (VITALITY)
I = 35 (MAX VITALITY)
J = 06 (CURRENT SPELL POINTS - INGAME…. DIFFERENT PRE-GAME?)
K= 09 (MAX SPELL POINTS - INGAME…. DIFFERENT PRE-GAME?)
L = 05 ('NATURAL' ARMOUR LEVEL)
M,N,O,P = (SPELL BOOK CONTENTS …. H= PAGE 1, I = PAGE 2 … ETC)
USE BINARY … %1000 = TOP SPELL ONLY, %1111 = ALL SPELLS ON PAGE
Q = FOOD LEVEL (0-$C7)
R = $FF ?
S = 00 - CURRENTLY WORN GLOVES (OBJECT VALUE)
T = 00 ?
U = 00 ?
V = 00 ?
W = 07 - START X-COORD
X = 09 - START Y-COORD
Y = 02 - START DIRECTION (0-3)
Z = 00 - START TOWER (DO NOT CHANGE!!)
AA = 03 - START FLOOR
BB = 00 ?
CC = 00 ?
DD = 00 ?
EE = 00 ?
FF = 00 ?
I am 99% sure one of the unknown values changes depending on levelling up pending, and/or spells waiting to be purchased… so XP is probably in there also, if not part of the same thing.
I will definitely use this to update the information in this board, but i'd be interested to know if you can work out any blanks by looking at the "live" data for us!
-
R = When you have a protect spell this is the amount of time it last 05 = 5 Second FF = 255 Seconds
Z = Timer Between hits (Gets Set to EE, as you level this will decrease so you attack faster. When set to 00 you will kill almost anything instantly as you do so many hits so quickly)
BB = Hit Cooldown
CC = XP till next Level (Change this to FF and you will gain a level when you sleep)
DD = XP Countdown Goes up to FF when if you do a hit of 20 it will reduce down by 20 then wrap round again when at 0
EE = Spells waiting to buy from the fairy
-
R = When you have a protect spell this is the amount of time it last 05 = 5 Second FF = 255 Seconds
nice find!
Z = Timer Between hits (Gets Set to EE, as you level this will decrease so you attack faster. When set to 00 you will kill almost anything instantly as you do so many hits so quickly)
definately Z ? if you set this to a different value before game-start does it work? is it different before / after game-start? (i.e. part of the code sets it to EE)
BB = Hit Cooldown
?? could you explain this one please! Amount of time after being hit until you do something?
CC = XP till next Level (Change this to FF and you will gain a level when you sleep)
so does this count downwards, getting set to a new (higher) value when you level-up, decreasing to -1 ($FF) for a level-up?
DD = XP Countdown: Goes up to FF when if you do a hit of 20 it will reduce down by 20 then wrap round again when at 0
err... bit confused on this one also!
EE = Spells waiting to buy from the fairy
a fixed number? i.e. changes to 01 for 1 spell to buy, 2 for 2 spells... ?
-
Z = As soon as the game starts the value gets cleared and set to EE so if I set it to 10 it will just get cleared, the same thing happens for all characters, when you move off the level it seem to only effect your party characters. (Level 01 = E9, Level 02 = D9, Level 04 = C9, Level 06 = A9, Level 08 = 99, Level 10 = 89)
BB = When you attack this will then increases to 07 and counts back down to 00 before you can attack again (this gets overridden though when you change Z) (7 seconds it seems between attack maybe this changes as you lvl)
CC = Correct this counts downwards FF when you lvl as soon as you sleep and it says you leveled and then go to some like 30 you will then have to decrease this value back to FF and you will level again and the next time the value will be something like 35 and repeats. (Ill see if I can work out the pattern) (Pattern is 11,24,40,50,53,60)
DD = Starts at 255 then if you do 30 Damage to a monster this will reduce to 225 (You need to get it down to 0) I guess this is linked to CC
EE = Correct so if you put this value to 09 you will be able to buy 9 spells from the fairy
-
OK this is a little odd anyone know about this...
My logic for finding the maps is find mod0's first 8 bytes then 57 bytes in is normally where all the map data is this works fine on all emulators but in Stew or WinUAE once you start the game a load of extra bytes are inserted so the bytes which where at 57 in are now at 312 below is all the data from mod0 to where 02 04 is for the first set of stairs for level 0, any ideas...
0c 15 0f 1f 13 04 00 00 01 15 0f 1f 13 05 00 00
00 00 00 18 03 8a 05 4c 0c ce 0f a0 0f c8 00 00
08 05 08 00 06 0b 00 00 05 05 08 00 06 0c 00 00
00 1f 00 1f 00 18 05 4c 00 00 f0 1e 00 00 00 b3
00 5d 00 00 00 27 06 18 ff ff 00 00 00 07 00 08
00 00 ff ff 10 0e 05 03 00 0c 00 16 00 00 00 00
00 00 00 0e 05 03 00 00 00 00 00 00 ff ff ff ff
ff ff 00 00 00 00 82 23 00 ff 00 00 ff ff ff ff
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ff
00 7f 00 ff 00 03 ff ff ff ff ff ff ff ff 01 00
00 00 00 00 00 00 00 60 0f 00 00 00 00 00 00 09
00 0c 00 00 ff ff ff ff ff ff 00 00 00 00 00 00
00 00 00 00 ff ff ff ff 00 00 00 00 00 00 ff ff
ff ff ff ff 00 00 00 00 00 00 00 ff 00 00 ff ff
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
00 ff 00 00 00 00 00 00 ff ff ff ff ff ff ff ff
0c 15 0f 1f 13 04 00 00 01 15 0f 1f 13 05 00 00
00 00 00 18 03 8a 05 4c 0c ce 0f a0 0f c8 00 00
08 05 08 00 06 0b 00 00 05 05 08 00 06 0c 00 00
00 15 00 15 00 18 00 05
I wish I looked before I posted, this seems to be all the data about player location, mouse location etc...
-
check the gem locstions and dungeon entrance files in the exported data i sent you.... some of it looks like it might be that!
would love to know more abiut mouse input and how icons are "trigggered" asthis would be useful for patching the WHDload slave for CD32 pad control.
great to see new investigation going on. i must confess i am mostly interested in editing the game binary rather than memory data though!
-
You should check out Cheat Engine with something like winuae it allows you to track and edit memory in real time which makes it pretty easy to find out what's going on it also has features to show what the asm code is doing and what writes and reads memory locations maybe it will help with the binary data
I'll try map out most of the data I posted its all stuff like what screens displayed which direction the player is facing what character portrait is displayed
-
Horace: Did you ever try and recreate bloodwych?
-
Horace, you have all the item graphics?
-
re-create Bloodwych?
I have a very basic walls/pillars only version of Bloodwych that runs from AMOS, but that's about it... I also figured i would only try to re-code the game in AMOS myself (for potential expansion) if i had a documented re-sourced version of the original code, so that i could ensure the gameplay alogarithms were identical!!
http://www.youtube.com/watch?v=6y6xfmCrwck
As for gfx... i probably have all the grafhics in IFF (amiga) format, but mostly as screen-grabs from my Action Replay... i think i do also have some of the "untouched" (ie. not coloured by the game) somewhere, but it might take me a while to dig that out!!
-
I have most the graphics in there raw format from the ST version (same as you un-coloured)
http://postimage.org/gallery/bx2luikw/dd77bd0b/
I was more after them in a format 01.png = coinage etc.. Its no big deal I'm going to try make a massive sprite sheet pre-colored of all the images from the ST version
As for recreating the game, I was just interested how you do all the drawing in 2D I've only ever done work in 3D which does all the hard work for you. Even if we don't have 100% source for bloodwych if you have ever played RTC (Return to chaos) on the PC which is a Dungeon Master clone it was done as best guess to emulate the feel of the original and you could never tell they didnt have the source.
So far I've getting some good progress with my editor, im working on the character editor at the moment hence why i need all the items all the code works I've just got to do the GUI (which makes it nice and easy grabbing images when you can change them as the emulator is running :))
I'll post up the image sheet I create once its done.
-
i must have them all in an AMOS bank format.... (from my editor) .... i could probably export them to small IFF/ILBM files for you?
-
If you can without too much effort it would be good to compare
here's all the bloodwych items..
http://postimage.org/gallery/84gtz5qo/50cefc4e/
-
Quick little peek at my Mapper so far.. (ignore the gfx glitch in Steem think I've had it running too long)
http://youtu.be/G5Wd3s3KhDA
So far it supports all Formats PC/Amiga/ST (Real Time on Amiga/ST)
Pocket Editor
Displays Monsters/Non Party Characters
To Do...
------
Fix Character editor to add full Character Stat/Spell editing via GUI (Code is done)
Detect if game has started or not (Extra bytes to compensate for)
Fix locations of characters in the editor to match Bloodwych
Display Party location on the map
Mapper
--------
Add switches/scripts
Add Objects
Add Monster Details
Allow Editing (Write changes back to memory and make GUI)
Extended Level fixes and editing
Saving to file
Realtime updates (Timer)
Speed Fixes
Look into Mac/Linux support for realtime editing, currently I use Windows ReadProcessMemory and WriteProcessMemory so I need to find alternatives for other OS's
-
Even if we don't have 100% source for bloodwych if you have ever played RTC (Return to chaos) on the PC which is a Dungeon Master clone it was done as best guess to emulate the feel of the original and you could never tell they didnt have the source.
I'm not so sure. I'm pretty sure I looked over the source of RTC back when it came out, and it was a disassembled. You could tell this by the labels that were still present, e.g. lbC00012FC or whatever.
I think a good first step to a Bloodwych resourcing, would be to simply fix up all the absolute addressing and make an executable. Don't worry about understanding the game. Once there's an executable, then anyone who comes along can work with a version that can easily be reassembled and debugged and the like.
-
I dont know enough about assembly to be able to do that, copse you seem to know what your talking about so is there any chance you would do this as it would be grate to have this.. :(
As for RTC I didnt think there was ever the source code released?
http://www.ragingmole.com/RTC/
5. Do you have the DM / CSB source-code...?
...usually accompanied by "...and will you please send me a copy too?".
No...and no. As mentioned before, I have written RTC myself entirely from scratch without reference to the source-code. Please don't ask me for a copy as I don't have one!
-
I dont know enough about assembly to be able to do that, copse you seem to know what your talking about so is there any chance you would do this as it would be grate to have this.. :(
No time. You'd be best to ask at EAB.
As for RTC I didnt think there was ever the source code released?
Ah, maybe I have it confused for another remake which had the source code. Still, I'd be surprised if the guy really wrote it from scratch.
-
Horace any chance in a write up on how to parse though the scroll data,
If see it goes 0000,0026,004E
So the text will start at 00 then 26 bytes is the next scroll etc..
After the last one at 222 (00DE) What does the next one mean (01,27) Is this next the next level or tower?
Also wanted to ask about switches, how do you work out what switch is linked to switch action?
-
Horace any chance in a write up on how to parse though the scroll data,
If see it goes 0000,0026,004E
So the text will start at 00 then 26 bytes is the next scroll etc..
After the last one at 222 (00DE) What does the next one mean (01,27) Is this next the next level or tower?
Also wanted to ask about switches, how do you work out what switch is linked to switch action?
I will try and answer some of this tomorrow!!
-
After the last one at 222 (00DE) What does the next one mean (01,27) Is this next the next level or tower?
you will see from my other post, this is answered now - you just need to read 0127 as a 'word' not a byte.
-
cool new discovery :)
I was investigating how the keyboard routines worked in-game, and discovered that each applicable keycode for movement is translated back into a final figure, which is then plugged into an address.
That address varies between players; using SPS439 disk (sorry, i changed SPS because of the WHDload patching) these addresses are at;
Player 1 … $EED2
Player 2 … $EF34
The value fed into these addresses are the same for each player, and are as follows;
Move Forward = $A
Move Backward = B$
Move Left = $C
Move Right = $D
Rotate Left = $E
Rotate Right = $F
So at any stage in the code, you can just "plug" that number into the address, and it will attempt to perform that action! (when the action is complete, or fails, the number at that address reverts back to zero)
More interesting still, is that by plugging a few "other" numbers into this, i was able to make the players perform other actions.... as such, i will be monitoring the address closely to see what other actions can be performed , and ultimately mapped to a separate key or button-press. :)
-
Some additional "Actions" i've found
$1 --- show stats
$2 ---- icon action (e.g. spell , door etc)
£3 ---- inventory
$4 --- attack
$5 --- attack (defend?)
action 2 is a bit useless because it doesnt update the "live" graphic - e.g. if you prepare a spell in the spell-book and use this action, the spell is still shown as "prepared" even though the action has been performed.
the "attack" action is excellent, and i think i will map this to the green button or something.
strangely, actually performing these actions "normally" does not cause the value to change, so this appears to be entirely for the purpose of keyboard control (!!)
-
Horace: Regarding the Wall Switches and Floor Pads (as that's now what i'm trying to recreate).
I see you have documented what they do http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39953.html#msg39953 but how does the map object get linked to the right action?
-
The short answer, is that when you define a wall /switch in the map with the appropriate word, (. . . 1) part of the first byte will relate to its "reference number" , and this maps directly to the switch actions which are listed in the "list of applicable actions for this tower" block mentioned in the link above. (i.e. reference number 1, is action no. 1 in the list)
You will have already come across this when allocating wall scrolls or gem slots to the correct colour, or mapping a sign to a particular walk scroll text... The main difference is that some bits tells us it is a switch, and other bits give us the reference number.
I will have a look at my editor code this morning and see if I can type up the definition for stone walls for you at least, and maybe floor pads (. . . 6) too!
-
I like it when you ask me these questions MadMunky, as it inspires me to write-up my findings about specific data elements :)
I have updated the section on standard walls:
http://www.ultimateamiga.co.uk/index.php/topic,8908.msg39945.html#msg39945
I will look to do a similar update on the floor switches section also, and maybe in-time the other data definitions!
Hope it helps.
edit:
Have now also updated the sections on Doors, Stairs and Floor Triggers etc.
-
A question if I may though.... Are you using the "pure data" file blocks I posted here before?
I would really like to establish them as a "standard" for cross-format pollination of Bloodwych levels etc, and this project has got me thinking more about finishing my own maps (which will genuinely be an option in summer once my studies are finished!) and I would love it if they worked on this!
-
Thanks horace, at the moment the moment it uses the pc map files but it will work with any file that follows that standard as thats how i got the extended levels working
So as long as it starts with, map x/y then offset etc it will work fine i can take any rip from the amiga or st and load them in so if you want to rip out your map you did ill load it up and show you
-
I dont know if this is any used to you, but when i was making the maps for Book of Skulls (which i will try to continue with again soon i think) i had real trouble keeping track of what objects and monsters i had used.
Without knowing what you've used, it is really hard to get a sense of progression through the game, and you never want to repeat-use objects like wands, rings or special weapons!!
To get around this, i made myself an excel spreadsheet, with VBA macros in it, and yesterday spent some time adapting it a little more. Basically it runs through the .ob files, and lists all the items used in each tower. "Stacks" of objects are highlighted in alternating colours (grey/white) in order to see what objects are bunched together... if the .map file is also available, it will also give you an x/y/f coordinate of where the item is located on the map. I will be looking to do something similar with lists of Monsters in the future.
Because i had a habit of just placing "1 gold coin" in locations where i wanted to later include weapons and stuff, but didnt know what to put there, i have also made it highlight the use of 1 gold coin in Red.
Other features include a complete list of the objects, and how often they occur in the .ob files that have been checked, so that you can check that items are not used too much or too little. I also gave this list a set of 'categories' so you can filter the list and just look at "all blades" for example.
Hope it is of some use! (file attached)
----
edit: I've also attached the 'clean' data i was talking about. I think you will need this, as although the PC version has the .map and .ob files (both of which i have re-created in my ripping of data here) you will need also a lot of other sections of data which are only stored in the main EXE even on the PC version.... this is why i think it is best for any editors / readers etc to "rip" the data in this form, and then patch it back in later.
You'll probably find this data set useful for your online version :)
-
Thanks Horace it all help, I was going to just use the PC Bloodwych.exe and find the locations for the stuff thats not in the MAP files but this makes it easier.
Have you got a export of your custom map at all as it would be good to try it and make sure it loads...
-
sure, data is attached.
Maps are complete up to the Moon tower - objects in the MOon tower still need work (thus why i built the Excel sheet) .
Maps in the Dragon tower are a work in progress... everything beyond that should appear blank.
I look forward to see it online!
-
Right loaded and seems to work fine, load up the page and hit "l" till you see it load "horace_mod0" I have loaded the other towers also but I need to know the correct X/Y for the players start position and then ill put in the right start positions as you change tower
-
Horace,
You found this but do you know where this data is store in the file?
http://www.ultimateamiga.co.uk/index.php/topic,9735.msg46344.html#msg46344
-
hi MM
Sorry for the slow reply... I will dig this out for you soon, and give you the 'data' (code) so you can do a search for it. I even use the same code syntax in my BoS patches, so when you know how to read it once, you could read it again for each applicable character / object.
I just brought a new MacBook Pro and I'm still in the process of transferring everything, and setting myself up for my paying work is taking a bit of a priority atm ;)
I am looking forward to trying this machine out with FS-UAE online :D
-
Thanks Horace, no real rush :) did you ever look any more at the amour palettes I posted if not the I'm planning to over the weekend
Sent from my iPad using Tapatalk
-
So i've located some more data in BW....
Two tables relating to objects (so far)
objecticons.block @ $E13E in the BW439 file , $1B9 bytes long, terminated with a $FF.
this goes through each object in turn (including obect '0' - an empty slot) and defines its appearance (an icon number), it's colour (if re-coloured) and its name (2 bytes, referring to words in the word block)
The word block is contained shortly above, and begins 1 byte before the word "Empty" (the byte before contains the number '5' to indicate that 5 characters are used in the word)
The second table i have discovered, is immediately after table 1, and defines how the object looks on the floor. This is $6D bytes long, and starts with the coinage.
That is all for now! An example edit is attached :)
-
ironically , the armour palettes mentioned above by MM, are the only thing stopping this from being "correct":