One of the most dangerous things a script can do is attempt to make a critter wield a weapon when the critter doesn't have the necessary animation (pistol, rifle, throw, etc). A critter will often either just disappear or the game will crash if this is attempted. Having changed what almost every critter in the game wields, I learned it is very easy to make mistakes with weapons in the hands of characters that can't use them. So I've been pondering how to do two things:
1) A check to see if any given critter can use any weapon
2) A way to randomize what weapons critters use
I already solved the second part, I have 5 new script commands that select random items, some are for random loot, some are for random store inventory, and there already is facility to handle random weapon generation. The first problem is more difficult to solve.
What I've been thinking for a week is this: it only requires one bit of storage to specify if a critter can use any particular weapon or attack. So there is no need to have a gazillion different variables for this. And this is really interesting: did you know that you can store many different values in a SINGLE global variable? Yes, its possible with bitwise operations, and this is why you see the bwand (bitwise and) and bwor (bitwise or) opcodes among the scripting commands. So I figured out a solution for this problem, one global variable for each unique appearance (FRM) stores multiple bits (0 or 1) and each bit represents an attack type. Here's how it's done:
First definition of the bits, if you look in define.h, you'll see that they already defined names for bit_1 through bit_32, example:
#define bit_1 ( 1 )
#define bit_2 ( 2 )
#define bit_3 ( 4 )
#define bit_4 ( 8 )
#define bit_5 ( 16 )
#define bit_6 ( 32 )
#define bit_7 ( 64 )
#define bit_8 ( 128 )
#define bit_9 ( 256 )
These are of course, powers of 2 because we're talking about binary bits.
So what I can do is define bits that represent attacks, like this:
#define ATTACK_PISTOL bit_1
#define ATTACK_RIFLE bit_2
#define ATTACK_BIGGUN bit_3
So how can I make one global variable store ALL of the attacks for a given character? It can be done using bitwise or (bwor) to set a bit and bitwise and (bwand) to check if a bit is set. For example:
set_global_var(GVAR_NAME, GVAR_NAME bwor bit_1)
This will set the first bit in that global variable, and to check if its set:
x = (global_var(GVAR_NAME) bwand bit_1)
If x equals the value of the bit (1,2,4,8,16,32...) in this case 1 ... then the bit is set.
Now, how to avoid a huge chain of if-then-else to check these bits? Well, each FRM has a unique identifier called FID (File ID) and they start at number 16777217, and ascend following the order of frm names in \art\critters.lst. What I need is to make an frm name map to a global variable, then store within the bits of the global which attacks are available. The game already uses global variables up to number 695. So what I figured was this: the first critter will use 701, and they ascend in the global variable index in the same order as they do in critters.lst. This way I can use the art fid number to access the global variable by index. So I define one simple formula:
#define GVAR_ARTFID_INDEX obj_art_fid(self_obj) - 16776516
This gets the art fid number from the system, and converts it into a gvar index number. Take for example the first critter in critters.lst, its art fid number is 16777217. Plugging that into the above formula we get 701, exactly where I want to start having GVARs for attack lookups. So this formula will automatically work for any critter, take the HMWARR character that you start as, his fid number is 16777278, and using the formula this makes him GVAR 762. I'll use HMWARR as an example of how this lookup works.
I define GVARs for all of the critters in \data\vault13.gam:
GVAR_ATTACKS_hapowr :=0; // (701)
GVAR_ATTACKS_harobe :=0; // (702)
GVAR_ATTACKS_hfcmbt :=0; // (703)
GVAR_ATTACKS_hfjmps :=0; // (704)
...
GVAR_ATTACKS_hmwarr :=0; // (762)
Note that these exactly follow the order of critters in critters.lst, because this order is the same order as art fid number returned by the script command obj_art_fid(self_obj) .. and subtracting 16776516 yields the index in the GVAR list. So now I know where to set or get bits, and I don't need to use names, I can use variable numbers, and since art fid (critter art file number) comes from the system, I can make one command work with any critter.
Next I define the bits for all of the attacks:
#define ATTACK_PISTOL bit_1
#define ATTACK_SMG bit_2
#define ATTACK_RIFLE bit_3
#define ATTACK_BIGGUN bit_4
#define ATTACK_FLAMER bit_5
#define ATTACK_MINIGUN bit_6
#define ATTACK_ROCKET bit_7
#define ATTACK_KNIFE bit_8
#define ATTACK_CLUB bit_9
#define ATTACK_SLEDGE bit_10
#define ATTACK_SPEAR bit_11
#define ATTACK_PUNCH bit_12
#define ATTACK_KICK bit_13
#define ATTACK_THROW bit_14
Those are all of the attacks, so 14 bits are needed for each critter to store their available attack. This means: if a bit is set they can use that type of weapon or attack.
I already have a spreadsheet that lists all of the attacks for each critter, this is a bit of work, but I can set the bits for critters at the beginning of the game, for example in the artemple map script, the first script to run. All attacks get loaded into GVARs and are therefore available to be checked by any script, anywhere in the game. Here's an example of setting the attacks for the primitive warrior (HMWARR) that the player starts as:
I need a mapping of FRM name to GVAR index for convenience:
#define GVAR_ARTFID_HMWARR (762)
And I define two macros, one for setting bits, and one for getting bits:
#define SET_ATTACK(frm,type) set_global_var(frm, global_var(frm) bwor type)
#define GET_ATTACK(frm,type) (global_var(frm) bwand type)
Then I proceed to set the attack bits for HMWARR character:
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_SPEAR);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_PUNCH);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_KICK);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_THROW);
Now I can check if ANY character can use ANY WEAPON like this:
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_SPEAR)) == ATTACK_SPEAR) then display_msg("Spear");
Note how it automatically gets the artfid from a macro that finds the right location of the global variable. The GET_ATTACK command will return the same bit value for spear (bit_11 which is 1024 or 2 ^ 10 within a 32-bit address) IF that bit is set, otherwise zero. I had to check to make sure we can address up to 32-bits for GVARs and yes it works. This means that every GVAR can store 32 on/off states. And if you think about permutations/combinations of those states, a single GVAR can represent a very large number of unique events or states. If you can project where I'm going with this, all I need is a lookup of the bit number by item PID for weapons, and I can distill it down to one simple command:
if (weapon_check(PID_WEAPON))
and I can combine that with my selectRandomWeapon command:
while not (weapon_check(PID_WEAPON)) do begin
weapon := selectRandomWeapon();
end
I have different variants of the selectRandomWeapon command, it can be totally random (any weapon type) or it can be randomized within a type (energy rifles, miniguns, thrown weapons etc). The above command will keep picking weapons until it finds one that the critter has an attack animation for. Now we have random loot, random store inventories, and random weapons on critters.
Putting together the peices in a small example (2 weapon types and one critter):
#define ATTACK_PISTOL bit_1
#define ATTACK_SMG bit_2
#define GVAR_ARTFID_INDEX obj_art_fid(self_obj) - 16776516
#define GVAR_ARTFID_HAPOWR (701)
#define SET_ATTACK(frm,type) set_global_var(frm, global_var(frm) bwor type)
#define GET_ATTACK(frm,type) (global_var(frm) bwand type)
procedure start;
procedure talk_p_proc;
procedure start begin
SET_ATTACK(GVAR_ARTFID_HAPOWR, ATTACK_PISTOL);
SET_ATTACK(GVAR_ARTFID_HAPOWR, ATTACK_SMG);
end
procedure talk_p_proc begin
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_PISTOL)) == ATTACK_PISTOL) then display_msg("PISTOL");
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_SMG)) == ATTACK_SMG) then display_msg("SMG");
end
The #define parts will be placed in a header file so any script can include them. The SET_ATTACK commands will be done in the first script that runs (artemple map script) so the attack bits are available to any script in the game.
One of the things that annoys me the most about FO scripting is lack of arrays. It makes it difficult to do lookups or operations on sets. However, the bitwise operations in conjunction with GVARs provides a way to do this very efficiently. As you can see, GVARs are very powerful.
1) A check to see if any given critter can use any weapon
2) A way to randomize what weapons critters use
I already solved the second part, I have 5 new script commands that select random items, some are for random loot, some are for random store inventory, and there already is facility to handle random weapon generation. The first problem is more difficult to solve.
What I've been thinking for a week is this: it only requires one bit of storage to specify if a critter can use any particular weapon or attack. So there is no need to have a gazillion different variables for this. And this is really interesting: did you know that you can store many different values in a SINGLE global variable? Yes, its possible with bitwise operations, and this is why you see the bwand (bitwise and) and bwor (bitwise or) opcodes among the scripting commands. So I figured out a solution for this problem, one global variable for each unique appearance (FRM) stores multiple bits (0 or 1) and each bit represents an attack type. Here's how it's done:
First definition of the bits, if you look in define.h, you'll see that they already defined names for bit_1 through bit_32, example:
#define bit_1 ( 1 )
#define bit_2 ( 2 )
#define bit_3 ( 4 )
#define bit_4 ( 8 )
#define bit_5 ( 16 )
#define bit_6 ( 32 )
#define bit_7 ( 64 )
#define bit_8 ( 128 )
#define bit_9 ( 256 )
These are of course, powers of 2 because we're talking about binary bits.
So what I can do is define bits that represent attacks, like this:
#define ATTACK_PISTOL bit_1
#define ATTACK_RIFLE bit_2
#define ATTACK_BIGGUN bit_3
So how can I make one global variable store ALL of the attacks for a given character? It can be done using bitwise or (bwor) to set a bit and bitwise and (bwand) to check if a bit is set. For example:
set_global_var(GVAR_NAME, GVAR_NAME bwor bit_1)
This will set the first bit in that global variable, and to check if its set:
x = (global_var(GVAR_NAME) bwand bit_1)
If x equals the value of the bit (1,2,4,8,16,32...) in this case 1 ... then the bit is set.
Now, how to avoid a huge chain of if-then-else to check these bits? Well, each FRM has a unique identifier called FID (File ID) and they start at number 16777217, and ascend following the order of frm names in \art\critters.lst. What I need is to make an frm name map to a global variable, then store within the bits of the global which attacks are available. The game already uses global variables up to number 695. So what I figured was this: the first critter will use 701, and they ascend in the global variable index in the same order as they do in critters.lst. This way I can use the art fid number to access the global variable by index. So I define one simple formula:
#define GVAR_ARTFID_INDEX obj_art_fid(self_obj) - 16776516
This gets the art fid number from the system, and converts it into a gvar index number. Take for example the first critter in critters.lst, its art fid number is 16777217. Plugging that into the above formula we get 701, exactly where I want to start having GVARs for attack lookups. So this formula will automatically work for any critter, take the HMWARR character that you start as, his fid number is 16777278, and using the formula this makes him GVAR 762. I'll use HMWARR as an example of how this lookup works.
I define GVARs for all of the critters in \data\vault13.gam:
GVAR_ATTACKS_hapowr :=0; // (701)
GVAR_ATTACKS_harobe :=0; // (702)
GVAR_ATTACKS_hfcmbt :=0; // (703)
GVAR_ATTACKS_hfjmps :=0; // (704)
...
GVAR_ATTACKS_hmwarr :=0; // (762)
Note that these exactly follow the order of critters in critters.lst, because this order is the same order as art fid number returned by the script command obj_art_fid(self_obj) .. and subtracting 16776516 yields the index in the GVAR list. So now I know where to set or get bits, and I don't need to use names, I can use variable numbers, and since art fid (critter art file number) comes from the system, I can make one command work with any critter.
Next I define the bits for all of the attacks:
#define ATTACK_PISTOL bit_1
#define ATTACK_SMG bit_2
#define ATTACK_RIFLE bit_3
#define ATTACK_BIGGUN bit_4
#define ATTACK_FLAMER bit_5
#define ATTACK_MINIGUN bit_6
#define ATTACK_ROCKET bit_7
#define ATTACK_KNIFE bit_8
#define ATTACK_CLUB bit_9
#define ATTACK_SLEDGE bit_10
#define ATTACK_SPEAR bit_11
#define ATTACK_PUNCH bit_12
#define ATTACK_KICK bit_13
#define ATTACK_THROW bit_14
Those are all of the attacks, so 14 bits are needed for each critter to store their available attack. This means: if a bit is set they can use that type of weapon or attack.
I already have a spreadsheet that lists all of the attacks for each critter, this is a bit of work, but I can set the bits for critters at the beginning of the game, for example in the artemple map script, the first script to run. All attacks get loaded into GVARs and are therefore available to be checked by any script, anywhere in the game. Here's an example of setting the attacks for the primitive warrior (HMWARR) that the player starts as:
I need a mapping of FRM name to GVAR index for convenience:
#define GVAR_ARTFID_HMWARR (762)
And I define two macros, one for setting bits, and one for getting bits:
#define SET_ATTACK(frm,type) set_global_var(frm, global_var(frm) bwor type)
#define GET_ATTACK(frm,type) (global_var(frm) bwand type)
Then I proceed to set the attack bits for HMWARR character:
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_SPEAR);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_PUNCH);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_KICK);
SET_ATTACK(GVAR_ARTFID_HMWARR, ATTACK_THROW);
Now I can check if ANY character can use ANY WEAPON like this:
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_SPEAR)) == ATTACK_SPEAR) then display_msg("Spear");
Note how it automatically gets the artfid from a macro that finds the right location of the global variable. The GET_ATTACK command will return the same bit value for spear (bit_11 which is 1024 or 2 ^ 10 within a 32-bit address) IF that bit is set, otherwise zero. I had to check to make sure we can address up to 32-bits for GVARs and yes it works. This means that every GVAR can store 32 on/off states. And if you think about permutations/combinations of those states, a single GVAR can represent a very large number of unique events or states. If you can project where I'm going with this, all I need is a lookup of the bit number by item PID for weapons, and I can distill it down to one simple command:
if (weapon_check(PID_WEAPON))
and I can combine that with my selectRandomWeapon command:
while not (weapon_check(PID_WEAPON)) do begin
weapon := selectRandomWeapon();
end
I have different variants of the selectRandomWeapon command, it can be totally random (any weapon type) or it can be randomized within a type (energy rifles, miniguns, thrown weapons etc). The above command will keep picking weapons until it finds one that the critter has an attack animation for. Now we have random loot, random store inventories, and random weapons on critters.
Putting together the peices in a small example (2 weapon types and one critter):
#define ATTACK_PISTOL bit_1
#define ATTACK_SMG bit_2
#define GVAR_ARTFID_INDEX obj_art_fid(self_obj) - 16776516
#define GVAR_ARTFID_HAPOWR (701)
#define SET_ATTACK(frm,type) set_global_var(frm, global_var(frm) bwor type)
#define GET_ATTACK(frm,type) (global_var(frm) bwand type)
procedure start;
procedure talk_p_proc;
procedure start begin
SET_ATTACK(GVAR_ARTFID_HAPOWR, ATTACK_PISTOL);
SET_ATTACK(GVAR_ARTFID_HAPOWR, ATTACK_SMG);
end
procedure talk_p_proc begin
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_PISTOL)) == ATTACK_PISTOL) then display_msg("PISTOL");
if ((GET_ATTACK(GVAR_ARTFID_INDEX, ATTACK_SMG)) == ATTACK_SMG) then display_msg("SMG");
end
The #define parts will be placed in a header file so any script can include them. The SET_ATTACK commands will be done in the first script that runs (artemple map script) so the attack bits are available to any script in the game.
One of the things that annoys me the most about FO scripting is lack of arrays. It makes it difficult to do lookups or operations on sets. However, the bitwise operations in conjunction with GVARs provides a way to do this very efficiently. As you can see, GVARs are very powerful.