Pattern Bombers are also refered to as stones, as part of the stone - scissors - paper analogy. Scissors, which includes vampires and scanners, are bigger than stones and therefore tend to get beat up by them more often. Paper, also known as a replicator, is a program that makes copies of itself throughout the core faster than a pattern bomber can destroy all of them. Stones are thus ineffectual against paper, or at least they were until W. Mintardjo stuck a two-pass core-clear on one of his stones.
Name: Dwarf Author: A.K.Dewdney Speed: 33.33% of c Size: 4 Durability: Weak Effectiveness: Average Score: bomb DAT #0 dwarf ADD #4, bomb MOV bomb, @bomb JMP dwarf end dwarfDwarf bombs every fourth instruction with DAT instructions in hopes that enemy processes will execute this code and die. Since 4 divides coresize, Dwarf will never drop a bomb on itself. Because Dwarf only hits every fourth instruction, it is a mod-4 bomber.
HOW IT WORKS: Before anything executes, core looks like this:
DAT #0, #0 ;bomb ADD #4, -1 (1) MOV -2, @-2 JMP -2, 0Then process (1) adds 4 to the B-field of bomb:
DAT #0, #4 ;bomb ADD #4, -1 MOV -2, @-2 (2) JMP -2, 0Process (2) moves bomb 4 instuctions forward, where the B-field of bomb points to:
DAT #0, #4 ;bomb ADD #4, -1 MOV -2, @-2 JMP -2, 0 (3) DAT #0, #4Process (3) simply makes the program loop back to the beginning.
DAT #0, #4 ;bomb ADD #4, -1 (4) MOV -2, @-2 JMP -2, 0 DAT #0, #4Process (4) adds 4 to the B-field of bomb:
DAT #0, #8 ;bomb ADD #4, -1 MOV -2, @-2 (5) JMP -2, 0 DAT #0, #4Process (5) drops the next bomb where the B-field of bomb is pointing.
DAT #0, #8 ADD #4, -1 MOV -2, @-2 JMP -2, 0 (6) DAT #0, #4 DAT #0, #8Process (6) loops back, and bomb after bomb are dropped forward through core.
Name: Stone Author: Matthew Householder Speed: 33.34% of c Size: 4 Durability: Weak Effectiveness: Average Score: start MOV <2, 3 ADD d1, start JMP start DAT #0 d1 DAT #-5084, #5084 end startStone is a mod-4 bomber like Dwarf, but with two important improvements. First, the step-size has been increased somewhat for better distribution of bombs against larger opponents. Second, Stone decrements other adresses while it bombs. Decrementing opponent's code may wound it so that DAT bombs can destroy it later.
HOW IT WORKS: Pre-decrement indirect addressing can be tricky, so we shall use the intuitive approach, even though it yields wrong results for weird instructions like MOV <0,<1. See "tutorial.2" or the ICWS '94 standard for precise details.
When Stone is loaded, core looks like this below. The DAT #0,#0 instruction is used only as a spacer between the executable code and the other DAT statement, as we shall shortly see.
MOV <2, 3 (1) ADD 3, -1 JMP -2, 0 DAT #0, #0 DAT #-5084, #5084The B-field of the JMP instruction (pointed to by the A-field of the MOV instruction) is decremented, so that it now points to the ADD instruction. This ADD instruction is now moved to the DAT #0,#0 instruction (pointed to by the B-field of the MOV instruction). Core now looks like this:
MOV <2, 3 ADD 3, -1 (2) JMP -2, -1 ;this has been decremented ADD 3, -1 ;this has been copied DAT #-5084, #5084This last sequence may be a little misleading, because it looks like we are dropping ADD 3,-1 bombs throughout core. We shall see this is not usually the case.
We now come to the ADD 3,-1 instruction. Since this ADD is not immediate, as it was in Dwarf, the A-operand of the DAT instruction is added to the A-operand of the MOV instruction and the B-operand of the DAT instruction is added to the B-operand of the MOV instruction:
MOV <-5082, 5087 ADD 3, -1 JMP -2, -1 (3) ADD 3, -1 DAT #-5084, #5084The executing process now jumps back (the -1 in the B-field is ignored).
MOV <-5082, 5087 (4) ADD 3, -1 JMP -2, -1 ADD 3, -1 DAT #-5084, #5084Process (4) drops another bomb: the location -5082 behind the MOV instruction is decremented and whatever it points to is moved 5087 in front of the MOV instruction. The pattern continues until someone is killed or time runs out.
Stone, then, doesn't really drop bombs as such, but rather moves instructions around core in a pseudo-random fashion. But since core is initialized to DAT 0,0, most of the instructions it moves are deadly DAT statements. This process is called transposition in the literature.
Name: Armadillo Author: Stefan Strack Speed: 32.86% of c Size: 5 Durability: Strong Effectiveness: Average Score: bomb SPL 0 loop ADD #3039, ptr ptr MOV bomb, 81 JMP loop MOV 1, <-1 end bombArmadillo drops SPL 0 bombs throughout core to stun the enemy, and then lays down a DAT carpet (also called a core-clear) to kill the enemy. This is one of the earliest bombers that used a core-clear to erase all of memory. It scores 100% wins against Wait (program 1, chapter 1) where Dwarf and Stone only score 25% wins and 75% ties. In my experience, SPL bombs are the most effective single-instruction bomb a warrior can drop. However, SPL bombs don't kill many programs cleanly, don't allow you to simultaneously bomb the rest of the core with decrements, and don't paralyze the opponent as well as the multi-instruction bombs that scanners drop.
Another innovation in Armadillo is the use of a SPL 0 instruction inside the warrior. If any of the other instuctions are hit with DAT bombs, the program may not operate correctly, but the bomb doesn't kill all of the processes. Additionally, this self-splitting code generates enough processes that imps cannot kill Armadillo by themselves.
HOW IT WORKS: When Armadillo is loaded into core, it looks like this:
SPL 0, 0 (1) ADD #3039, 1 MOV -2, 81 JMP -2, 0 MOV 1, <-1Process (1) splits into processes (2) and (3).
SPL 0, 0 (3) ADD #3039, 1 (2) MOV -2, 81 JMP -2, 0 MOV 1, <-1Process (2) executes and process (3) splits.
SPL 0, 0 (6) ADD #3039, 1 (5) MOV -2, 3120 (4) JMP -2, 0 MOV 1, <-1Process (4) drops a split bomb, process (5) changes the bombing location, and process (6) splits.
SPL 0, 0 (10) ADD #3039, 1 (9) MOV -2, -1841 (8) JMP -2, 0 (7) MOV 1, <-1Process (7) jumps back in order to conserve processes, (8) bombs, (9) changes the bombing location, and (10) splits.
SPL 0, 0 (15) ADD #3039, 1 (14) (11) MOV -2, 1198 (13) JMP -2, 0 (12) MOV 1, <-1And so the process continues. The ever-lengthening string of processes executes the code (backwards!) that drops the SPL bombs. Eventually, a SPL 0,0 gets dropped on the JMP statement:
SPL 0, 0 ADD #3039, 1 MOV -2, 1 SPL 0, 0 (1) MOV 1, <-1The loop is broken, and all of the processes fall through to this second SPL instruction eventually. We examine this last bit of code as if there were only one process running at the SPL instruction, since the program doesn't depend on process order from this point on. Process (1) splits:
SPL 0, 0 ADD #3039, 1 MOV -2, 1 SPL 0, 0 (3) MOV 1, <-1 (2)Process (2) decrements the B-field of the SPL instruction (which the SPL instruction doesn't need) and moves the blank (DAT 0,0) instruction to where the SPL instruction points:
SPL 0, 0 ADD #3039, 1 SPL 0, -1 (3) MOV 1, <-1 (4)Process (3) splits:
SPL 0, 0 ADD #3039, 1 SPL 0, -1 (6) MOV 1, <-1 (5) (4)Now process (4) executes an illegal instruction and dies, (5) decrements the SPL instruction again and bombs the next instruction backwards, and (6) splits:
SPL 0, 0 SPL 0, -2 (9) MOV 1, <-1 (8) (7)This pattern repeats until eventually the core clear wraps around and erases itself. Just before this erasure occurs, core looks like this:
SPL 0, 2 (23997) MOV 1, <-1 (23996) (23995)Process (23995) dies as usual, but this time, when process (23996) bombs, it erases the bombing instruction:
SPL 0, 2 (23997) (23998)Now, if we ignore all of the dying processes, we see that this SPL command keeps splitting processes to itself, keeping the warrior alive.
Name: Cannonade Stone Speed: 24.51% of c Size: 5 Durability: Average Effectiveness: Good Score: MOV <6, 1 start SPL -1, <5144 ADD 3, -2 DJN -2, <5142 DAT #0, #0 MOV 190, <-190 end startCannonade Stone takes the idea of self-splitting code to another level. Altough it bombs somewhat slower than other bombers, it splits off processes so quickly that a stun attack on other components of the warrior will not halt the execution of the stone. The bombing run hits every fifth instruction, with a transposition at every tenth position and a decrement between each transposition. Additionally, a DJN-stream is laid through memory, giving another form of attack without increasing the size or speed of the program. At the end of the bombing run, Cannonade Stone converts into a core-clear and partial imp-gate.
HOW IT WORKS: When Cannonade Stone is first loaded into memory, it looks like this:
MOV <6, 1 SPL -1, <5144 (1) ADD 3, -2 DJN -2, <5142 DAT #0, #0 MOV 190, <-190Process (1) splits:
MOV <6, 1 (3) SPL -1, <5144 ADD 3, -2 (2) DJN -2, <5142 DAT #0, #0 MOV 190, <-190Now processes (2) and (3) execute, adding and then bombing like every other stone.
MOV <196, -189 SPL -1, <5144 (5) ADD 3, -2 DJN -2, <5142 (4) DAT #0, #0 MOV 190, <-190Process (4) usually jumps back to the SPL instruction (more on this in a moment), and the pattern repeats: each process at the SPL command splits into two processes, which add and bomb in rapid succession.
At the end of the bomb run, the bomber mutates itself into a core-clear. The SPL -1,<5144 instruction is overwritten with the MOV 190,<-190 instruction. The executng portion of code then looks like this:
MOV 190, <-190 ADD 3, -2 DJN -2, <5142The first instruction performs the core-clear, the second instruction does nothing of strategic worth, and the third instruction loops processes back to the first instruction. Additionally, the decrement in the MOV command sets up a partial (33%) imp-gate 190 instructions before it, and the decrement in the DJN instruction sets up a second partial (33%) imp gate 2666 instructions before the first one. Since 2667 is the magic number for 3-point imps, these instructions defend the bomber against 3-point imps at roughly 67% efficiency.
Let us examine in more detail how the DJN -2,<5142 instruction works. When it is executed, the predecrement in the B-field decrements the instruction 5142 after the DJN intstruction, which is probably a DAT 0,0 command:
DJN -2, <5142 ... DAT 0, -1The DJN instruction now decrements the instruction before that, which probably doesn't have a B-value of 1, so the executing process jumps back to the beginning of the loop:
DJN -2, <5142 ... DAT 0, -1 ;this was decremented by the DJN DAT 0, -1 ;this was decremented by the <The next time the DJN instruction is executed, the B-field 5142 after the instruction is decremented, and so is the instruction pointed by that B-field (2 before it):
DJN -2, <5142 ... DAT 0, -1 ;this was decremented by the DJN DAT 0, -1 DAT 0, -2 ;this was decremented by the <As the DJN instruction is repeatedly executed, a carpet of decrements is laid down backwards through core.
This is not exactly the pattern that is laid down in core, because the SPL -1,<5144 command decrements the same B-field as the DJN instruction does. This adds gaps in the DJN-stream, making it more spread out and liable to hit the enemy program. Additionally, it turns the B-field into a better partial imp-gate.
We have made two assumptions: First, that the instruction 5142 after the DJN instruction is DAT 0,0; second, that the instruction pointed to by that instruction does not have a B-field of 1. If the first assumption fails, the worst that can happen is a non-zero B-field, in which case the DJN stream is laid somewhere else. If the second assumption fails, then the executing process does not jump back and proceeds instead to an illegal instruction. Fortunately, this is just one of many processes, so the bombing loop is not seriously affected. This result may be compunded, however, if the enemy has lots of B-fields with value 1.
Name: Night Crawler Stone Author: Wayne Sheppard Speed: 32.86% of c Size: 4 Durability: Strong Effectiveness: Good Score: start SPL 0, <-1001 MOV <21, 1+2234 SUB 1, -1 DJN -2, <-2234 end startNight Crawler Stone is a self-splitting mod-2 bomber with a DJN-stream. When it finishes its bombing run, it turns into code that performs an addition core-clear.
HOW IT WORKS: Night Crawler Stone bombs memory similarly to Stone, with the obvious improvements that Night Crawler Stone bombs in a tighter mod-2 pattern, is self-splitting, uses a DJN-stream, and embeds the bombing step size in the executing code, making it one instruction smaller.
After the SPL 0,<-1001 instruction has split off about 144 processes into the main loop, it is bombed, making the effective size of Night Crawler Stone only 3 instructions long. Just before the bomber hits the bombing loop, the SUB 1,-1 instruction is decremented, starting an addition core-clear.
Unlike traditional core-clears, the addition core-clear doesn't overwrite core with DAT statements. Instead, it modifies the A- and B-fields of the instructions to mess up the enemy's control structures. For example, a SPL 0 that survived the bombing run becomes a SPL 2 which will not hold processes by itself. An addition core-clear is only slightly less effective than a traditional core-clear, and requires no additional instructions to run.
Just before the addition core-clear takes effect, Night Crawler Stone looks like this:
DAT 0, -1 MOV <1, 3895 (12938) (12941) ... SUB 1, -1 (12940) (12943) ... DJN -2, <-2234 (12939) (12942) ...Process (*38) executes, decrementing the SUB instruction and doing a copy:
DAT 0, -1 MOV <1, 3895 (12941) ... SUB 1, -2 (12940) (12943) ... DJN -2, <-2234 (12939) (12942) ...Process (*39) executes, laying down another decrement in the DJN stream. Process (*40) then executes, changing the A- and B-operands of the DAT statement:
DAT 2, 2233 MOV <1, 3895 (12941) ... SUB 1, -2 (12943) ... DJN -2, <-2234 (12942) ...Process (*41) executes, decrementing the SUB instruction again, and then (*43) modifies the operands of the next instruction back:
DAT 2, 2234 DAT 2, 2233 MOV <1, 3895 SUB 1, -3 DJN -2, <-2234So goes the core-clear, until at the end the DJN instruction is hit and turns into DJN 0,<0, where all of the processes go and execute repeatedly, laying down a DJN stream until time expires.
Name: Keystone Stone Speed: 32.86% of c Size: 5 Durability: Strong Effectiveness: Good Score: step equ 2517 emerald SPL 0, <-25 MOV <-step+1, 92 SUB 2, -1 DJN -2, <2002 JMP step, <-step wait DJN 0, <-12 paper DJN 0, <-12 boot MOV emerald+4, paper-step MOV emerald+3, <boot MOV emerald+2, <boot MOV emerald+1, <boot MOV emerald, <boot MOV wait, paper+3053 JMP @boot end bootAfter initialization, Keystone Stone bombs with a mod-1 pattern which approximates mod-4. If paper is detected, processes are split to the label "paper," where some code can be inserted to withstand paper attacks. When the bombimg run is over, Keystone Stone turns itself into an imp gate. (P.Kline's Keystone uses this gate as a backup strategy. Under normal operation, an external core-clear erases this stone.)
HOW IT WORKS: To set things up, the imp-gate (labelled "wait") needs to be copied away from the main block of code. Rather than adding an instruction to the main block to do this, the boot-strapping code (imaginatively labelled "boot") copies the stone and the imp-gate away from itself.
This has two advantages when fighting warriors that search through memory for the enemy. First, the copied code containing the executing stone is kept small, making it more difficult to locate. Second, the original code acts as a decoy for the enemy. In fact, many programs pad the block of original code with nonsense instructions to make a larger decoy for the enemy to grapple with. Almost all modern stones use boot-strapping and decoys to slow down the enemy.
When the initialization is finished, the stone starts a typical bombing run. If a process executing the DJN instruction finds a B-operand of 1, it falls out of the loop, executes the JMP instruction, and ends up at the label "paper," where some paper-stomping code should be inserted. The rationale behind this is that typically only paper has a B-operand of 1.
The bombing run ends with the DJN -2,<2002 instruction being hit, but not with a typical DAT bomb. Because of clever planning, the imp-gate instruction overwrites the DJN -2 instruction. The bomber now looks like this:
SPL 0, <-25 MOV <-step+1, 2 SUB 2, -1 DJN 0, <-12 JMP 2517, <-2517Nearly all of the processes in the stone end up executing the DJN 0 instruction, forming an imp-gate. Along with killing imps, this imp-gate lays down a DJN-stream for extra program mangling. And processes falling through the DJN instruction don't matter much, because the SPL 0 instruction slowly generates new processes.
Name: Winter Werewolf Author: W. Mintardjo Speed: 25% of c Size: 7 Durability: Weak Effectiveness: Excellent Score: step equ 153 init equ 152 n equ ((12*8)-2) data DAT <-4-n, #0 split SPL 0, <-3-step-n main MOV jump, @3 MOV split, <2 ADD #step, 1 JMP main, init MOV @-4, <n jump JMP -1, 1 boot MOV main+5, -500+5 MOV main+4, <boot MOV main+3, <boot MOV main+2, <boot MOV main+1, <boot MOV main, <boot MOV main-1, <boot MOV data, boot-500-3-n JMP boot-500 end bootWinter Werewolf is a mod-8 bomber more in the spirit of Armadillo than Stone -- it drops specialized bombs througout core to stun the enemy, and then kills the enemy with a core-clear. It outscores Armadillo in three major aspects: It drops a more effective SPL/JMP bomb, it uses a two-pass core-clear, and it degrades into a perfect imp-gate to mop up any stray imps. The first pass of the core-clear lays down a SPL 0 stream to make sure the enemy is Really Stunned, and the second pass of the core-clear lays down DAT statements that kill the enemy. Winter Werewolf was one of the first modern programs that could compete against imp-rings.
HOW IT WORKS: After the boot-strapping routine, Winter Werewolf looks like this:
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 (1) MOV -2, <2 ADD #153, 1 JMP -3, 152 MOV @-4, <94 JMP -1, 1The next two instructions drop a SPL/JMP bomb. Fist the JMP -1,1 instruction is copied:
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 (2) ADD #153, 1 JMP -3, 152 MOV @-4, <94 JMP -1, 1 ... JMP -1, 1The next instruction decrements the bomb pointer and copies the SPL 0,<-250 instruction to the new location:
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 ADD #153, 1 (3) JMP -3, 151 MOV @-4, <94 JMP -1, 1 ... SPL 0, <-250 JMP -1, 1The next instruction changes the bomb pointer in preparation for dropping the next bomb.
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 ADD #153, 1 JMP -3, 302 (4) MOV @-4, <94 JMP -1, 1 ... SPL 0, <-250 JMP -1, 1Finally, the JMP instrction loops to bomb the next location. The B-operand of the JMP instruction is ignored, allowing it to be used as the bomb pointer. The bombing run hits every eighth locoation with one of these bombs. The big trick at this point is to have the program bomb itself without getting trapped in a SPL/JMP loop itselfÈthe bombing run is over, the program looks like this (if we reset the process counter):
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 (1) MOV -2, <2 ADD #153, 1 JMP -3, 0 MOV @-4, <94 JMP -1, 1When this first instruction is executed, the bomb pointer is bombed with the JMP -1, 1 instruction.
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 (2) ADD #153, 1 JMP -1,1 MOV @-4, <94 JMP -1, 1But, since the B-field of the bomb pointer just got changed to 1, the next bomb hits the bomb pointer, too. Remember, first the pointer is decremented...
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 ADD #153, 1 JMP -1,0 MOV @-4, <94 JMP -1, 1...and then the SPL bomb is dropped.
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 ADD #153, 1 (3) SPL 0, <-250 MOV @-4, <94 JMP -1, 1Now the most subtle command of the whole program executes: The B-field of the new SPL 0,<-250 command is altered. We shall see later why this is important.
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 MOV -2, <2 ADD #153, 1 SPL 0, <-97 (4) MOV @-4, <94 JMP -1, 1Now the core-clear begins. The SPL 0,<-97 instruction splits off processes and the JMP -1,1 instruction speeds up the core-clear, but it is the MOV @-4,<94 command that does the actual core-clear, and this deserves further comment.
The A-field of the MOV @-4,<94 instruction points to the MOV 5,@3 command that points to the SPL 0,<-97 instruction. Since the A-field uses indirect addressing, we are carpteing the core with SPL 0,<97 instructions for now. If the B-field of the MOV @-4,<94 instruction pointed to an instruction with zero B-field, this would yeild a very short (93 instruction) core-clear before the MOV command erased itself. But because of the bombing run, the B-field points to a SPL 0,<-250 command. So the pointers look like this:
<--------------------------------- ... | DAT <-98, #0 | ... | SPL 0, <-250 | MOV 5, @3 ---- <---- | MOV -2, <2 | | | ADD #153, 1 | | | SPL 0, <-97 <---- | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-250 <--------------------- ---After the first process executes the MOV @-4,<94 instruction, the pointers look like this:
SPL 0, <-97 <-------------------------- ... | DAT <-98, #0 | ... | SPL 0, <-250 | MOV 5, @3 ---- <---- | MOV -2, <2 | | | ADD #153, 1 | | | SPL 0, <-97 <---- | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-251 <--------------------- ---And after the second process executes this instruction, the pointers look like this:
SPL 0, <-97 <-------------------------- SPL 0, <-97 | ... | DAT <-98, #0 | ... | SPL 0, <-250 | MOV 5, @3 ---- <---- | MOV -2, <2 | | | ADD #153, 1 | | | SPL 0, <-97 <---- | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-251 <--------------------- ---And so the core-clear goes, filling the entire core with SPL 0,<-97 commands, until the B-field pointer gets overwritten:
DAT <-98, #0 ... SPL 0, <-250 MOV 5, @3 ---- <---- MOV -2, <2 | | <------------- ADD #153, 1 | | | SPL 0, <-97 <---- | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-97 <---------------------- ---This is where the ADD instruction mentioned above becomes so vital. If this pointer were overwritten by a SPL 0,<-250 command, the SPL core-clear would repeat, and the program would never get around to killing off the opponent.
Note that the new pointer value skips over most of the core-clearing code, allowing the program to start a second core-clear. And the next iteration of MOV @-4,<94 does even more pointer magic, overwriting the A-field pointer with the SPL 0,<-97 instruction:
DAT <-98, #0 <--- ... | SPL 0, <-250 | SPL 0, <-97 ---- <---- <------------- MOV -2, <2 | | ADD #153, 1 | | SPL 0, <-97 | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-98 <---------------------- ---Because the A-field pointer now points to the DAT <-98,#0 command, this bomb is dropped next:
DAT <-98, #0 <--- ... | DAT <-98, #0 | <-------------------- SPL 0, <-97 ---- <---- | MOV -2, <2 | | ADD #153, 1 | | SPL 0, <-97 | | MOV @-4, <94 A-field--- B-field--- | JMP -1, 1 | | ... | | SPL 0, <-98 <---------------------- ---This starts the second core-clear, using DAT statements to finally kill the enemy processes. Like the first core-clear, this one continues until it wraps around and overwrites the B-field pointer again:
DAT <-98, #0 <--- ... | SPL 0, <-97 ---- <---- MOV -2, <2 | ADD #153, 1 | SPL 0, <-97 | MOV @-4, <94 A-field--- B-field--- JMP -1, 1 | ... | DAT <-98, #0 <---------------------But this time, the new pointer does not cause the coreclear to skip the code. The core-clear continues until the MOV @-4,<94 instruction overwrites itself with the DAT <-98,#0 instruction:
SPL 0, <-97 MOV -2, <2 ADD #153, 1 SPL 0, <-97 DAT <-98, #0 DAT <-98, #0The SPL 0,<-97 instruction keeps splitting processes to itself, keeping the program alive. The only other instruction that executes is the next one, which kills off all of the processes that execute it. Both of these instructions decrement the same instruction before the executing code, forming an imp-gate to kill off any remaining imp-spirals the enemy might have.
Ideally, it ought to hit every location in 8000 bombs, every other location in 8000/2=4000 bombs, every third location in 8000/3=2667 bombs, etc. Unfortunately, this is impossible, especially with a single step size, but it suggests a basic strategy -- go for the biggest programs first and then fill in the gaps.
One way of rating the efficiency of a step size is to find the length of the largest unbombed section of code after each bomb is dropped. By adding up all of these lengths, we get a number that tells us how big an average gap is. (Indeed, by dividing this number by the number of bombs dropped, we get the average gap size.) If we minimize this number over all step sizes, we get the "Optima Numbers." For a coresize of 8000, these optima numbers are:
mod-1 3359/3039 under-100 -> 73 mod-2 3094/2234 under-100 -> 98 mod-4 3364/3044 under-100 -> 76 mod-5 3315/2365 under-100 -> 95 mod-8 2936/2376 mod-10 2930/2430The constant for Night Crawler Stone, for instance, is taken from this table.
Another common rating is how closely to in half the new bomb subdivides the old gap when it is dropped. By taking the differences between where the bombs fall and the middle of each gap and adding these distances up, we get an alternate method for testing efficiency.
Both of these methods are useful for finding general-purpose step sizes. But suppose you wanted to find a step size optimized for killing other stones. Since stones usually have four or five instructions, you would want a step size that would bomb every 4th and 5th instruction quickly, regardless of how it does in general.
Fortunately, there is a program in the public domain that calculates all of these things quicky. Corestep by Jay Han can be found as misc/corestep.c, and calcutates optima numbers and optimal step sizes. You will need a C compiler to use it, but it is otherwise self-contained. For more infromation, FTP a copy and read through it. The classic formula calculates optima numbers, the alternate formula calculates the sum of the distances between bombs and midpoints, and find-X calculates optimal step sizes against a specific program length.
If you don't have access to a C compiler or want this for some other reason, P. Kline has compiled a list of all 8000 step-sizes with their mod, find-4, find-5, find-10, and find-13 numbers, along with imp-killing constants and imp-numbers. This table is designed for use in spreadsheets or databases. It is available in the misc/ directory under the name num8000.txt with documentation in num8000.doc. He used this on Keystone Stone to come up with a mod-1 constant with a low find-4 score, so that it would act like a mod-4 bomber but interfere with enemy scans (more about this in the next chapter).
Here is a list of successful stones. All of these can be found in warrior10.tar in the 88 directory, except for SJ-4A and Keystone t21, which are buried deep within the file feb94.txt.Z (in the newsgroup directory last time I checked.) Everything here by P.Kline has an anti-vamp component, which will be talked about in a later chapter.
"Leprechaun 1b" by Anders Ivner (leprechaun)
"Emerald 2" by P.Kline
(emerald2)
"ExtraExtra 2" by P.Kline (extra2)
"Keystone t21" by
P.Kline
"SJ-4A" by J.Layland
"Moonstone 1" by Dan Nabutovsky
(moonstone)
Self-splitting stones with imp-rings can be very effective. Here is a list of imp-stone combos that are worth investigating. All of them except Cannonade can be found in warrior10.tar, and Cannonade can be found in the feb94.txt.Z file.
"Cannonade" by P.Kline
"Imprimis 6" by P.Kline (imprimis6)
"Night
Crawler III" by Wayne Sheppard (nightcrawl)
"Sphinx 2.8" by W. Mintardjo
(sphinx)
Program 2, Stone, was taken from the ICWS 1990 corewar tournament. It bears a remarkable resemblance to Rock by Scott Nelson, which was posted to the net a couple of months before the tournament. Strange, eh?
Program 4, Cannonade Stone was extracted from P.Kline's Cannonade.
Program 5, Night Crawler Stone without the SPL 0 was submitted as "No Ties Allowed," and confused the experts as to how something so deadly could fit into 3 lines.
Program 6, Keystone Stone, was stolen from P.Kline's "Keystone t21." The bootstrapping code in the example differs somewhat from the bootstrapping code used in Keystone.
Program 7, Winter Werewolf, originally did not copy the stone away from a decoy. I am led to speculate that the code as it exists here with a bigger decoy resembles Winter Werewolf 3, a program that was very successful on the hill.