wormy 0.97.105
[wormy.git] / wormy.z80
index e0bb2b21affa566f98efa39a2fd6ba19de19d30a..22131ff968e97df3212c725349b5b56528e9a10a 100644 (file)
--- a/wormy.z80
+++ b/wormy.z80
@@ -1,58 +1,86 @@
-; Title                      : Wormy
-; Version                    : 95%
-; Release Date               : summer 2001
-; Filename                   : wormy.86p (5kb)
-; Author(s)                  : Shiar
-; Email Address              : shiar0@hotmail.com
-; ICQ UIN                    : #43840958
-; Web Page                   : www.shiar.org
-; Description                : ruling Nibbles-like game 1-4 players
-; Where to get this game     : www.shiar.org (home of Wormy)
-; Other games by author(s)   : Nemesis beta
-; Additional Credits to      : Matthew Shepcar (wrote original Peaworm, end'98)
-;                              Jonah Cohen (helped writing worm)
-
-;-----------------------------
-;----------- TO-DO -----------
-;-----------------------------
-
-; 95% = DONE
-
-;    [* internal levels         ]
-;    [  * check levels/gametype ]
-;    [  * enough hiscore saves! ]
-;    [* complete readme         ]
-;  1% * misc (pollish, bugs, &&&)
+;___����________________________________________________________����____________
+;  ����                                                         �����
+; ���                                                            ����     ����
+;���      �����          ��������       �����     ����  ����      ���     �����
+;���     �������       �����������     �������   �������������     ��    ��� ���
+;��     ���   ���     �����    ����   ����  ��� ���  ����  ���     ���  ���� ���
+;��     ��     ��     ��  �      ��� ����      ���  ������  ���    ���  ���   ��
+;��     ��     ��     �� ��      ��� ���       ��� ���� ��� ���    ��� ����   ��
+;��     ���   ���     �� ��       �� ���       �������   ��� ��     �������   ��
+;��     ���   ���    ��� ��       �� ��        �� ���    ��� ���     �����   ���
+;���     ��� ���     ��� ��       �� ���       �� ����  ���� ���             ���
+;���      �����     ���  ���     ��� ���       ��  ��������  ���            ���
+; ���      ���      ���  ���     ��� ���       ���  ������   ���            ���
+;  ����  �������  ����    ���   ���  ����      ���            �����       ����
+;   �������� ��������     ���������   ���      ���             ���������������
+;    �����     �����       �������     ��       ���              �����������
+;_______________________________________________________________________________
+;
+; Version                  :                                       97% (0.97.15)
+; Release Date             :                                     2002 January 15
+; Author(s)                :                                               Shiar
+; Email Address            :                                     wormy@shiar.org
+; Web Page                 :                                       www.shiar.org
+; Description              :                ruling Nibbles-like game 1-4 players
+; Where to get this game   :                       www.shiar.org (home of Wormy)
+; Other games by author(s) :                                        Nemesis beta
+; Additional Credits to    : Matthew Shepcar  :   wrote original Peaworm, end'98
+;                            Jonah Cohen      :              helped writing worm
+; Files                    : wormy.z80 (89kB) : 515ad14b922572bdc8a96e780b8b24ca
+;                            wormy.86p (6532) : efef32a8c541b4585087f55deb31f51d
+;_______________________________________________________________________________
+
+;NOTES:
+;* Use for LEARNING practises ONLY!
+;   Don't _ever_ release altered code w/o permission!
+;* Code was originally by Scabby (Matthew Shepcar),
+;   a few lines by Jonah Cohen,
+;    and everything else by me - Shiar (Mischa Poslawsky).
+;* Although I've commented & labeled quite some stuff, I can't guarantee
+;   everybody will understand it. This is probably not the best source
+;    to learn z80 from. w00t the Shyer Way (tm).
+;* Don't scroll down if you get scared easily.
+
+;_______________________________________________________________________________
+; _______  _____        ______   _____
+;    |    |     |       |     \ |     |
+;    |    |_____| _____ |_____/ |_____|
+;_______________________________________________________________________________
+
+; 97% = DONE
+
+;  2% * misc (bugs, &&&, pollish, &&, &)
+;     X score >9 dispay [15]
+;     * mem at worm #4 (still 12 bytes or so.. or more? or less?)
+;     * look at init-z in line routine again
+;     X sp modes not configurable [15]
+;     * team score
+;     * random pixel still occurs
+;     * levels: Wormage | Jonah? | 2 | 3 | 4
 ;   * LINK
-;     * fix first packet loss
-;  1% * transmit game/level data
-;;;1% * get g/o signal (l&l) working
-;  2% * send new peas' positions
+;  1% * fix transmit game/level data (?)
+;     * >Somehow do a lot of testing with 2 calcs< *sigh*
 
 ;100% = bugs fixed + levels done
 
-;-----------------------------
-;-----------------------------
-;---------  W O R M  ---------
-;-----------------------------
-;-----------------------------
+;     * arrow to offscreen peas
+;     * multiple peas in sp level
+;>110%:        sound, startpos (?)
+;>120%:        coop (DON'T COUNT ON IT)
 
-#define buffer      ;use display buffer (otherwise write directly to screen)
-#define readymask   ;"greys" out the field before starting a level
-#define coolzgfx    ;nice graphics for game over screen
-
-;#define readytext  ;displays "prepare" before level starts
-;#define invincible ;worms cannot die =)
-;#define optdie     ;in race games worms don't die when they run into each other
-
-#define cal call
-#define psh push
-#define dnz djnz
-;#define halt nop
+;_______________________________________________________________________________
+;  _/_/_____ __   _ _______        _     _ ______  _______
+; _/_/   |   | \  | |       |      |     | |     \ |______
+; / /  __|__ |  \_| |_____  |_____ |_____| |_____/ |______
+;_______________________________________________________________________________
 
 #include "asm86.h"
 #include "ti86asm.inc"
 
+ #define cal call
+ #define psh push
+ #define dnz djnz
+
 _SHRACC            = $4383 ;4x srl a
 _SHLACC            = $438B ;4x sll a
 _divHLby10         = $4044 ;hl=hl/10
@@ -83,13 +111,24 @@ _load_ram_ahl      = $462F ;ahl->page+hl
 _writeb_inc_ahl    = $5567 ;ld (ahl++),c
 _jforce            = $409C ;TI-OS stack restored
 _EXLP              = $4493 ;swap (hl),(de) b times
-_GETB_AHL         = $46C3 ;a=(ahl) \ hl=ahl
+_GETB_AHL          = $46C3 ;a=(ahl) \ hl=ahl
+
+ #define buffer     ;use display buffer (otherwise write directly to screen)
+ #define readymask  ;"greys" out the field before starting a level
+ #define coolzgfx   ;nice graphics for game over screen
+ #define spprotect  ;options not changable for singleplayer modes
+;#define intlevels  ;-internal levels
+;#define readytext  ;-displays "prepare" before level starts
+;#define invincible ;-worms cannot die =)
+;#define optdie     ;-in race games worms dont die when they run into each other
 
-;-----------------------------
-;------- data  storage -------
-;-----------------------------
+;_______________________________________________________________________________
+; _______ _______  _____   ______ _______  ______ _______
+; |______    |    |     | |_____/ |_____| |  ____ |______
+; ______|    |    |_____| |    \_ |     | |_____| |______
+;_______________________________________________________________________________
 
-;--- permanent
+;--- permanent ---
 
 ScrBuffer   = $8820 ;-A7FF (20*FF=1FE0)
 worm2p      = $A800 ;-ABFF (400)      %10101O00
@@ -99,70 +138,84 @@ worm3       = $AC3E ;-AC59 (1F)
 worm4       = $AC5D ;-AC77 (1F)
 balls       = $AC7C ;-ACFF (3x43d)
 SinCosTable = $AD00 ;-AE00 (4*40)
-        ;free $AE01 ;-AFFF (1FF)
+turn10      = $AE01 ;-AE01 (1) (counter)
+peaspos     = $AE02 ;-AE05 (4+) (peas)
+;free       = $AE06 ;-AFFF (1FA)
 worm1p      = $B000 ;-B7FF (800)      %1011O000
 worm3p      = $B800 ;-BBFF (400)      %10111O00
-        ;free $BF91 ;-BFFF (6F)
-     ;program $D748 ;-EFFF (186D+4A)
+;free       = $BF91 ;-BFFF (6F)
+;program    = $D748 ;-EFFF (186D+4A)           ;6327
 worm4p      = $F000 ;-F3FF (400)      %11110O00
 leveldata   = $F400 ;-FA70 (<=671)
+       ;can you believe i actually left $269 bytes of memory unused?!?
 
-turn10      = $AE01 ;      (1) (counter)
-peaspos     = $AE02 ;-AE05 (4) (peas)
+;MEM|8---9---A---B---C---D---E---F---| ;there's something wrong
+;   |..[------]||[]|......[-----]|[].| ;if you still understand
+;   |   SCREEN 2* 13      PRGM   4LV | ;after looking at this
 
-;MEM|8---9---A---B---C---D---E---F---|
-;   |..[------]||[]|......[-----]|[].|
-;   |   SCREEN 2* 13      PRGM   4LV |
-
-;--- temporary
+;--- temporary ---
 
 namelength  = $BC00 ;(1)         @menu
+datalink    = $BC00 ;(8)        @init
 #ifdef buffer
 DispBuffer  = $BC00 ;(10x57d)    @game
 #else
 DispBuffer  = $FC70
 #endif
-
-;-----------------------------
-;------- program start -------
-;-----------------------------
-
-.org _asm_exec_ram
-
-wormVhost   = 093
-wormVclient = 193
-
-start:
-  nop
-  jp Start
-  .dw 1
-  .dw WormTxt
-  .dw WormIcon
-
-WormTxt:
-  .db "WORMY by SHIAR -- beta 95%",0
+lefttotalb  = $AFFF
+drawctfpea1 = peaspos+4
+drawctfpea2 = peaspos+6
+
+;_______________________________________________________________________________
+;  _____   ______  _____   ______       _______ _______ _______  ______ _______
+; |_____] |_____/ |     | |  ____       |______    |    |_____| |_____/    |
+; |       |    \_ |_____| |_____| _____ ______|    |    |     | |    \_    |
+;_______________________________________________________________________________
+
+ .org %1101011101001000
+
+start:         ;turn back NOW!
+  nop      ;and so it begins...
+  jp  Start
+  ld  bc,$5242
+  rst 10h
+ .dw  WormIcon
+  ld  d,a
+  ld  c,a
+  ld  d,d
+  ld  c,l
+  ld  e,c
+  jr  nz,$+100
+  ld  a,c
+  jr  nz,$+$55
+  ld  c,b
+ .dw  16713
+ .db  %1010010
+  jr  nz,$+47
+  dec l
+ .db  " 97% .15",0
 WormIcon:
 .db 8,2
 .db %00000000,%00111100
 .db %00000000,%01010010
 .db %00000000,%01100001
 .db %01100011,%10011001
 .db %10010100,%01101001
 .db %10011001,%00011001
 .db %01000010,%11000001
 .db %00111100,%00111110
.db  8,2
.db  %00000000,%00111100
.db  %00000000,%01010010
.db  %00000000,%01100001
.db  %01100011,%10011001
.db  %10010100,%01101001
.db  %10011001,%00011001
.db  %01000010,%11000001
.db  %00111100,%00111110
 
 levelhead  = 'w'
 levelhead2 = 95 ;wormy levels header = "95"
 
 int_handler:
-  ex af,af'
-  in a,($03)
+  ex  af,af'
+  in  a,($03)
   bit 3,a
-  jp z,$0039
+  jp  z,$0039
   res 0,a
   out ($03),a
-  jp $0039
+  jp  $0039
 int_end:
 
 Start:
@@ -188,7 +241,7 @@ Start:
   ldir
   inc a   ;ld a,$D4
   ld  i,a
-  im  2
+  im  2 ;...it *does* work ;)
 
   cal _RAM_PAGE_7
   ld  hl,$BFFF ;VAT start
@@ -243,7 +296,7 @@ stringfound:
   inc ix
 _searchnext:
   cal _RAM_PAGE_7
-  jr searchnext
+  jr  searchnext
 
 loadgametype:
   psh de
@@ -261,12 +314,26 @@ loadgametype:
   jr  nz,loadgametype
   ret
 
+;_______________________________________________________________________________
+;        _______ _    _       _______ _______        _______ _______ _______
+; |      |______  \  /        |______ |______ |      |______ |          |
+; |_____ |______   \/   _____ ______| |______ |_____ |______ |_____     |
+;_______________________________________________________________________________
+
 searchcomplete:
   ld  a,255
   ld  (ix),a ;end mark
 
+#ifdef intlevels
   ld  ix,templevels-6
-levelselectmenu:
+#else
+  ld  ix,templevels-3
+#endif
+  cp  (ix+3) ;1st=255
+  jp  z,ExitNoStats ;->no lvls
+  cp  (ix+6) ;2nd=255
+  jr  z,loadlevel1 ;->1 lvl
+levelselectmenu: ;load next page
   psh ix ;offset
   ld  a,-2
   ld  (availevels),a
@@ -276,123 +343,149 @@ levelselectmenu:
   ld  hl,txtLevsel
   cal _vputs ;"< SELECT LEVELS >"
   ld  hl,$FC00+(2*16)
-  cal hr
-  ld  hl,$FC00+(10*16)
-  cal hr
+  ld  b,16*9
+  cal menuinvloop
   ld  hl,$FC00+(56*16)
   cal hr
-  ld  hl,$0C01 ;x=1
+  ld  hl,$0601 ;x=1
   ld  (_penCol),hl
 dispnextlevel:
   ld  de,3
   add ix,de
   ld  a,(ix)
+  ld  b,a
   inc a ;cp 255
   jr  z,__levselect
-  dec a
 displevel:
-  ld  h,(ix+1)
-  ld  l,(ix+2) ;ahl=(ix)
-  cal _load_ram_ahl ;hl=ahl
-  ld  b,(hl) ;b=title size
-  cal _vputs
   ld  hl,_penCol
   ld  (hl),$01 ;x=1
   inc hl
   ld  a,(hl)
   add a,6
   ld  (hl),a ;y+6
-  ld  hl,availevels
-  inc (hl)
   cp  49 ;bottom of screen
   jr  nc,_levselect
+  ld  hl,availevels
+  inc (hl)
+  ld  a,b ;(ix+0)
+  ld  h,(ix+1)
+  ld  l,(ix+2) ;ahl=(ix)
+  cal _load_ram_ahl ;hl=ahl
+  cal _vputs
   jr  dispnextlevel
 
+loadlevel1:
+  psh ix
+  pop hl
+  ld  b,-2
+  jr  loadlevel
+
+readylevelfile: ;selected level at ahl
+; ld  hl,templevels-3/0
+  ld  a,b
+  add a,3 ;sellev+3 (#0==-3)
+  ld  e,a
+  add a,a ;*2
+  add a,e ;*3
+  ld  e,a
+  ld  d,0 ;de=sellev*3
+  add hl,de
+  ld  a,(hl) ;ade=(hl)
+  inc hl
+  ld  d,(hl)
+  inc hl
+  ld  e,(hl)
+  ex  de,hl ;ahl=ade
+skiptitle:
+  ld  d,a ;psh ahl
+  psh hl
+  cal _GETB_AHL ;ld a,(ahl)
+  pop hl
+  or  a
+  ld  a,d ;pop ahl
+  psh af
+  cal _INC_PTR_AHL ;ahl++
+  pop af ;cp 0
+  jr  nz,skiptitle ;goto #0-terminator
+  ret
+
 levup:
   cal menupos
   dec b ;up
   ld  a,b
   cp  -3
-  jr  nz,levselected
+  jr  nz,levselect
   inc b ;undo
-  jr  levselected
+  jr  levselect
 levdown:
   cal menupos
   inc b ;down
   ld  a,b
   cp  -2
 availevels =$-1
-  jr  nz,levselected
+  jr  nz,levselect
   dec b ;back up
-levselected:
-  jp  menupos
-
+levselect:
+  cal menupos
+  ld  hl,$3900
+  ld  (_penCol),hl
+  pop hl
+  psh hl
+  cal readylevelfile
+  cal _load_ram_ahl ;hl=ahl
+  ld  de,$FFA0 ;desc.text
+  xor a
+levdescclearloop:
+  ld (de),a ;empty
+  inc de
+  cp  d ;de>$FFFF (offscreen)
+  jr  nz,levdescclearloop
+  cal _vputs
+  jr  levselectmenu+1
 __levselect:
+#ifdef intlevels
   ld  ix,templevels-6 ;reset 2 1st page
+#else
+  ld  ix,templevels-3
+#endif
 _levselect:
   ld  b,-2 ;level selected
-  cal menupos
-  pop hl
-levselect:
+  jr  levselect
+levselectmenu:
   psh hl
   psh bc
   cal ubergetkey
   pop bc ;GET_KEY destr. b
-  cp  K_UP
-  cal z,levup
-  dec a ;K_DOWN
-  cal z,levdown
-  cp  K_SECOND-1
-  jr  z,loadlevel
+  dec a ;cp K_DOWN
+  jr  z,levdown
+  sub K_UP-1
+  jr  z,levup
   pop hl
-  cp  K_RIGHT-1
+  inc a ;cp K_RIGHT
   jp  z,levelselectmenu
-  cp  K_EXIT-1
+  cp  K_ENTER-K_RIGHT
+  jr  z,loadlevel
+  sub K_EXIT-K_RIGHT
   jp  z,ExitNoStats
-  cp  K_ENTER-1
-  jr  nz,levselect
+  inc a ;cp K_SECOND
+  jr  nz,levselectmenu
 ; jr  z,loadlevel
 
 loadlevel:
-; ld  hl,templevels-3
-  ld  a,b
-  add a,3 ;sellev+3 (#0==-3)
-  ld  e,a
-  add a,a ;*2
-  add a,e ;*3
-  ld  e,a
-  ld  d,0 ;de=sellev*3
-  add hl,de
-  ld  a,(hl) ;ade=(hl)
+  cal readylevelfile
   or  a ;levelfile on page 0 (=internal)
   jr  z,levelloaded
-
-  inc hl
-  ld  d,(hl)
-  inc hl
-  ld  e,(hl)
-  ex  de,hl ;ahl=ade
-skiptitle:
-  ld  b,a ;psh ahl
-  psh hl
-  cal _GETB_AHL ;ld a,(ahl)
-  pop hl
-  or  a
-  ld  a,b ;pop ahl
-  psh af
-  cal _INC_PTR_AHL ;ahl++
-  pop af ;cp 0
-  jr  nz,skiptitle ;goto #0-terminator
+  cal skiptitle ;skip description
 
   cal _Get_Word_ahl
   ld  (leveldataSize),de
 
-  ld  d,8 ;counter
+  ld  d,9 ;counter
   ld  bc,datalevels
   cal loadgametype
   ld  (hilvlposa),a ;singleplayer levels
   ld  (hilvlposhl),hl
-  ld  d,4
+  ld  d,5
   cal loadgametype
 
   cal _SET_ABS_SRC_ADDR ;levelsstart
@@ -427,15 +520,19 @@ setdeflevels:
   add hl,de ;next
   dnz setdeflevels
 
-;-----------------------------
-;----- build trig tables -----
-;-----------------------------
+;_______________________________________________________________________________
+; _______  _____   ______ _______       _____ __   _ _____ _______
+; |  |  | |     | |_____/ |______         |   | \  |   |      |
+; |  |  | |_____| |    \_ |______ _____ __|__ |  \_| __|__    |
+;_______________________________________________________________________________
 
-  ld  hl,TrigPrecalc
-  ld  de,SinCosTable
-  psh de
-  ld  bc,65
-  ldir
+;--- trig tables ---
+
+  ld  hl,TrigPrecalc ;I believe this
+  ld  de,SinCosTable  ;is one of the few
+  psh de              ;pieces of original
+  ld  bc,65            ;Peaworm code still
+  ldir                  ;left intact..
   dec hl
   ld  b,63
 MirrorSineWave:
@@ -454,11 +551,15 @@ NegativeSineWave:
   inc de
   dnz NegativeSineWave
 
-;-----------------------------
-;---------- menu -------------
-;-----------------------------
+;_______________________________________________________________________________
+; _______ _______ _____ __   _       _______ _______ __   _ _     _
+; |  |  | |_____|   |   | \  |       |  |  | |______ | \  | |     |
+; |  |  | |     | __|__ |  \_| _____ |  |  | |______ |  \_| |_____|
+;_______________________________________________________________________________
 
-DisplayMenu: ;---draw menu---
+;---draw menu---
+
+DisplayMenu:
   cal linkok
   cal _clrWindow
   ld  de,$FC40 ;(0,4)
@@ -472,11 +573,9 @@ DisplayMenu: ;---draw menu---
   ld  hl,$FC00+$3E0 ;(*,62)
   cal hr
 dispmainmenu:
-  ld  hl,Gametype
-  ld  a,(hl)
+  ld  a,(Gametype)
   dec a ;will be inced @changegame
-  and 7
-  ld  (hl),a
+  cal changedgame
   ld  hl,changegame ;dispmenusets ;mainMenu
   psh hl ;jump here after ret
   ld  hl,txtMenu
@@ -484,9 +583,10 @@ dispmainmenu:
 ;Mode|Level|Limit|Worms|worm #|controls
 ; jr  dispmenucommon ;cal
 
+dispmenucommon_:
+  ld  b,36*16/4
 dispmenucommon:
   ld  de,$FD80 ;begin pos
-  ld  b,36*16/3
   xor a
 clroldmenuloop:
   ld  (de),a
@@ -495,13 +595,15 @@ clroldmenuloop:
   inc de
   ld  (de),a
   inc de
+  ld  (de),a
+  inc de
   dnz clroldmenuloop
 
-  ld  b,6
+  ld  b,(ix)
 dispmenuloop:
-  ld  d,(ix)
+  ld  d,(ix+1)
   inc ix
-  ld  e,(ix)
+  ld  e,(ix+1)
   inc ix
   ld  (_penCol),de
   cal _vputs
@@ -511,24 +613,34 @@ dispmenuloop:
 
 hr: ;draw horizontal line at hl
   ld  b,16
-  jp  menuinvloop
+; jp  menuinvloop ;shorter but not good for pausescreen
+hrloop:
+  ld  (hl),-1
+  inc hl
+  dnz hrloop
+  ret
 
-;--- menu loop ---
+;--- options menu ---
 
 dispoptionmenu:
   ld  hl,txtoMenu
   ld  ix,posoMenu
 ;Back|Lives|Limit|Speed|Rotation|Growth
-  cal dispmenucommon
+  cal dispmenucommon_
 
 dispomenusets:
+#ifdef spprotect
+  ld  a,(gameCar)
+  and _datasingl
+  jp  nz,LetsGetThisPartyOn
+#endif
   cal clrold
 
   ld  hl,$1E3E
   ld  (_penCol),hl
   cal loadgamecar
   psh hl
-  cal cshowA0 ;lives
+  cal cshowA ;lives
 
   ld  hl,$2A3E
   ld  (_penCol),hl
@@ -542,7 +654,7 @@ dispomenusets:
   cal _vputs
   jr  dispspeeddone
 dispspeed:
-  cal cshowA0
+  cal cshowA
 dispspeeddone:
 
   ld  hl,$363E
@@ -552,7 +664,7 @@ dispspeeddone:
   psh hl
   ld  a,(hl) ;(growth)
   inc a ;-1=None; 0..98->1..99
-  cal cshowA0
+  cal cshowA
 
   ld  hl,$303E
   ld  (_penCol),hl
@@ -566,7 +678,7 @@ dispspeeddone:
   cal _vputs
   jr  dispturndone
 dispturn:
-  cal cshowA0 ;turn speed
+  cal cshowA ;turn speed
 dispturndone:
 
   ld  hl,$243E
@@ -576,7 +688,7 @@ dispturndone:
   ld  a,(hl) ;(scorelimit)
   or  a
   psh af
-  cal cshowA0 ;limit
+  cal cshowA ;limit
   pop af ;a==0?
   jr  z,optionMenu ;do not display 0 behind 'None'
   ld  a,'0'
@@ -728,7 +840,7 @@ bchangegrowth:
   jr  z,__optionMenu ;<-1
   jr  changedgrowth
 
-;---
+;--- main menu ---
 
 mainMenu:
   cal menupos
@@ -744,6 +856,17 @@ select:   ;2nd/enter
   dec a ;6th: controls
   jp  z,changekeys
   jp  LetsGetThisPartyOn ;otherwise
+
+changenrworms:
+  cal change4spOnly
+  ld  a,(hl) ;hl=nrworms
+  inc a
+  cp  5
+  jr  nc,mainMenu ;may not be >4
+changednrworms:
+  ld  (hl),a
+  jr  _dispmenusets ;mainMenu
+
 notselect
   cp  K_EXIT
   jp  z,ExitNoStats
@@ -799,15 +922,6 @@ bchangecurworm:
   jr  nz,changedcurworm ;save >0
   jr  mainMenu
 
-changenrworms:
-  cal change4spOnly
-  ld  a,(hl) ;hl=nrworms
-  inc a
-  cp  5
-  jr  nc,mainMenu ;may not be >4
-changednrworms:
-  ld  (hl),a
-  jr  _dispmenusets ;mainMenu
 changenrwormsInit:
   ld  a,(Gametype)
   cp  3
@@ -847,7 +961,12 @@ changegame:
   jr  z,changegame
   jr  dispmenusets ;mainMenu
 changedgame:
-  and 7 ;mod 8
+  cp  9
+  jr  c,changedgameok
+  sub 255-8 ;-1 -> 8
+  jr  nc,changedgameok
+  xor a ;9 -> 0
+changedgameok:
   ld  (Gametype),a
   cal getnrlevels
   xor a
@@ -910,7 +1029,7 @@ changekeys:
   cal getcustomkey ;right
 ; jr  dispmenusets
 
-;---display current settings---
+;--- display current settings ---
 
 dispmenusets:
   cal clrold
@@ -956,7 +1075,7 @@ displink:
   or  a ;gamesingle
   jr  nz,displvlname
   ld  a,(Level)
-  cal cshowA0
+  cal cshowA00
   jr  lvldisped
 displvlname:
   cal _vputs
@@ -1134,7 +1253,11 @@ nameentered:
   ld  (ix+8),0 ;end mark
   jp  DisplayMenu
 
-;--proc
+;_______________________________________________________________________________
+; _______ _____ _______ _______        _____   ______  _____  _______ _______
+; |  |  |   |   |______ |             |_____] |_____/ |     | |       |______
+; |  |  | __|__ ______| |_____  _____ |       |    \_ |_____| |_____  ______|
+;_______________________________________________________________________________
 
 ubergetkey:
   halt ;woo hoo
@@ -1165,7 +1288,7 @@ hlatlevel: ;d must be 0
   add a,a ;nc
   add a,e
   add a,a ;8->2bytes
-  add a,24+1 ;go to datalevels
+  add a,28+1 ;go to datalevels
   ld  e,a ;=de
   sbc hl,de ;datalevels
   cal _ldHLind ;ld hl,(hl)
@@ -1221,18 +1344,21 @@ skipballs:
 
 skipflags:
   ld  a,(Gametype)
+  cp  gamerace
+  jr  z,skiplaps
   cp  gamectf
   jr  nz,noflagstoskip
   inc hl
   inc hl
+skiplaps:
   inc hl
   inc hl
 noflagstoskip:
 
 skipobjects:
-  ld  a,(hl)
-  inc hl
-  or  a
+  xor a
+  or  (hl)
+  inc hl ;nf
   ret z ;0=end
   inc hl
   inc hl
@@ -1262,44 +1388,57 @@ loadgamecar:
   ld  (wormbeglives),a
   ret
 
-;-----------------------------
-;-------- start game ---------
-;-----------------------------
+;_______________________________________________________________________________
+; _______ _______ _______  ______ _______        ______ _______ _______ _______
+; |______    |    |_____| |_____/    |          |  ____ |_____| |  |  | |______
+; ______|    |    |     | |    \_    |    _____ |_____| |     | |  |  | |______
+;_______________________________________________________________________________
 
 LetsGetThisPartyOn:
   ld  a,$17 ;no exit
   ld  (CheckExit),a ;set exit state
 
-  cal loadgamecar
-  psh hl
-  ld  (wormbeglives),a
-  inc hl ;nrworms
-  ld  a,(hl)
-  ld  (nrworms),a
-  inc hl ;level
+setupworms:
+  ld  hl,worm1set
+  ld  de,worm1
+  ld  a,4 ;4x (all worms)
+createwormsloop:
+  ex  de,hl
+  ld  bc,died ;0008
+  add hl,bc
+; ld  b,0
+  ld  (hl),b ;died=0
   inc hl
-  ld  a,(hl)
-  ld  (customspeed),a
+  ld  (hl),b ;score=0
   inc hl
-  ld  a,(hl)
-  ld  (growspeed),a
+  ld  (hl),b ;score+1=0
   inc hl
-  ld  a,(hl)
-  ld  (turnleft),a
-  ld  (turnright),a ;more efficient
+  ld  (hl),2 ;delay=2
   inc hl
-  ld  l,(hl)
-  ld  h,0
-  cal _HLTIMES10       ;hl=10*(hl)
-  ld  (scorelimit),hl
+  ld  (hl),3 ;lives=x
+wormbeglives =$-1
+  inc hl
+  ex  de,hl  ;de=wormX+head
+  ld  bc,18
+  ldir       ;copy 18 bytes
+  dec a      ;loop
+  jr  nz,createwormsloop
 
-  pop hl ;loadgamecar
+OhMyGodItsALabel:      ;pj34r my coding skillz
+  cal loadgamecar
+  psh hl ;datatype
   cal hlatlevel
-
-  psh hl  ;1st level
-  ld  a,(gameCar)
+  ex  (sp),hl ;pop \ psh leveldata
+  psh hl ;psh loadgamecar
+  ld  a,0
+gameCar =$-1
   rra ;and _datalink
-  jr  nc,GameOver
+  jr  nc,_StartLevel
+
+;--- link ---
+
+wormVhost   = 095
+wormVclient = 195
 
 linkmatch:
   cal _clrWindow
@@ -1319,56 +1458,146 @@ linkmatch:
 host:
   ld  c,wormVclient
   cal Qsend
-  ld  a,$18
-  jr  multiplayer
+  jr  sethost
 client:
   ld  hl,txtReceive
   cal _puts
-  ld  a,$E6
-multiplayer:
-;  cal linkok
-;  ld a,D0LD1L
-;  out (7),a
-  ld  (SwapPos),a
-  ld  a,2
-  ld  (nrworms),a
-
-;-----------------------------
-;--------- game over ---------
-;-----------------------------
-
-GameOver:
-  ld  hl,worm1set
-  ld  de,worm1
-  ld  a,4 ;4x (all worms)
-createwormsloop:
+setclient:
+ ;name/keys: wormy#1 = link    = 0  + link (name1)
+ ;                 2 = worm #1 = #1 + local(name1)
+ ;                 3 = link    = 0  + link (name2)
+ ;                 4 = worm #2 = #2 + local(name2)
+  ld  de,worm2+left
+  ld  hl,worm1+left
+  ldi ;keys worm#2 = worm#1
+  ldi ;+right
+  ld  de,worm4+left
+  ldi ;keys worm#4 = worm#2
+  ldi
+  xor a
+  ld  (worm1+left),a ;worm 1...
+  ld  (worm3+left),a ;and worm 3 via link
+  ld  hl,worm1+name
+  ld  b,9
+  cal recvstuff
+  ld  hl,worm2+name
+  ld  b,9
+  cal sendstuff
+  pop hl ;loadgamecar
+  pop de ;leveldata
+  psh de
+  psh hl ;loadgamecar
+  ld  b,8
+  cal sendstuff
+  ld  hl,gameCar
+  ld  b,1
+  cal sendstuff
+  ld  hl,Gametype
+  ld  b,1
+  cal sendstuff
   ex  de,hl
-  ld  bc,died ;8
-  add hl,bc
-; ld  b,0
-  ld  (hl),b ;died=0
-  inc hl
-  ld  (hl),b ;score=0
-  inc hl
-  ld  (hl),b ;score+1=0
-  inc hl
-  ld  (hl),2 ;delay=2
+  ld  b,168
+  cal sendstuff
+_StartLevel:
+  jr  StartLevel
+sendstuff:
+  psh de
+sendstuffloop:
+  psh bc
+  ld  c,(hl)
   inc hl
-  ld  (hl),3 ;lives=x
-wormbeglives =$-1
+  cal Qsend
+  pop bc
+  dnz sendstuffloop
+  pop de
+  ret
+recvstuff:
+  psh de
+recvstuffloop:
+  psh bc
+  cal Qrecv
+  ld  (hl),c
   inc hl
-  ex  de,hl  ;de=wormX+head
-  ld  bc,18
-  ldir       ;copy 18 bytes
-  dec a      ;loop
-  jr  nz,createwormsloop
-
+  pop bc
+  dnz recvstuffloop
+  pop de
+  ret
+sethost:
+ ;name/keys: wormy#1 = worm #1 = ok
+ ;                 2 = link    = 0  + link (name1)
+ ;                 3 = worm #2 = #2 + local(name2)
+ ;                 4 = link    = 0  + link (name2)
+  ld  de,worm3+left
+  ld  hl,worm2+left
+  ldi ;keys worm#3 = worm#2
+  ldi ;+right
+  xor a
+  ld  (worm2+left),a ;worm 2+4..
+  ld  (worm4+left),a ;..over link
+  ld  hl,worm1+name
+  ld  b,9
+  cal sendstuff
+  ld  hl,worm2+name
+  ld  b,9
+  cal recvstuff
+  pop hl ;loadgamecar
+  pop de ;leveldata
+  ld  hl,templevels ;&&&&&&&
+  ld  de,datalink
+  psh de
+  psh hl ;loadgamecar
+  ld  b,8
+  cal recvstuff
+  ld  hl,gameCar
+  ld  b,1
+  cal recvstuff
+  ld  hl,Gametype
+  ld  b,1
+  cal recvstuff
+  ex  de,hl
+  ld  b,168
+  cal recvstuff
+
+;--- load level ---
+
 StartLevel:
+  pop hl ;loadgamecar
+  ld  a,(hl)
+  ld  (wormbeglives),a
+  inc hl ;nrworms
+  ld  a,(hl)
+  ld  (nrworms),a
+  inc hl ;level
+  inc hl
+  ld  a,(hl)
+  ld  (customspeed),a
+  inc hl
+  ld  a,(hl)
+  ld  (growspeed),a
+  inc hl
+  ld  a,(hl)
+  ld  (turnleft),a
+  ld  (turnright),a ;more efficient
+  inc hl
+  ld  l,(hl)
+  ld  h,0
+  ld  a,(gameCar)
+  rla
+  ld  a,h ;(Left)=256
+  jr  nc,setscorelimit
+  ld  a,l
+  ld  l,h
+setscorelimit:
+  ld  (customleft),a
+  cal _HLTIMES10 ;hl=10*(hl)
+  ld  (scorelimit),hl
+
+Nextlevel:
   cal _clrWindow
   pop hl ;begin of current level
   ld  a,(hl)
   inc a ;=255?
-  jp  nz,nextlevel
+  jp  nz,donextlevel
 
   psh hl
   ld  hl,Level
@@ -1380,13 +1609,30 @@ StartLevel:
   psh bc   ;where to go afterwards
   inc hl   ;location of ending-code
   jp  (hl) ;go there ("call")
-nextlevel:
+donextlevel:
   ld  a,(Gametype)
   or  a ;gamesingle
   psh af
   cal nz,skiplvltitle
-  ld  de,Left
-  ldi
+  ld  a,(hl)
+  and 127
+  jr  nz,setleft
+customleft =$+1
+  ld  a,0
+setleft:
+  ld  (Left),a
+  inc a
+  ld  (lefttotal),a
+  dec a
+  xor (hl) ;bit (hl),7
+  ld  bc,gameCar
+  ld  a,(bc)
+  jr  z,leftloaded
+  or  128+32
+leftloaded:
+  ld  (bc),a
+  inc de
+  inc hl
   ld  de,Speed
   ld  a,0
 customspeed =$-1
@@ -1465,8 +1711,11 @@ worminit:
   ld  (hl),a ;y2
   inc hl
   ld  (hl),a ;x2
+  inc hl
+  inc hl
+  ld  (hl),a ;growH=0
 
-  ld  bc,(worm2-worm1)-5
+  ld  bc,(worm2-worm1)-7
   add hl,bc
   pop bc ;<0
   dnz worminit
@@ -1476,8 +1725,9 @@ worminit:
   inc a ;ld a,2
   ld  (flashtime),a
 
-;-------- draw level ---------
+;--- draw level ---
 
+initlevel:
   ld  a,(de)
   inc de
   sub 128
@@ -1573,17 +1823,11 @@ levelhasbeensetup:
 
   cal drawstuff
 
-;-----------------------------
+;--- prepare ---
 
+leveldone:
   psh hl                        ; >> levelp new
   cal forceshowstats
-  ld  a,(gameCar)
-  and _datafood
-  jr  z,nofood
-  cal NewPea
-nofood:
-  ld  bc,(worm1+pos)
-  cal DisplayField
 
 #ifdef readymask
   ld  hl,$FC70
@@ -1603,7 +1847,6 @@ maskline:
   dec c
   jr  nz,maskloop
 #endif
-
 #ifdef readytext
   ld  hl,$FDE0
   ld  de,$FDE1
@@ -1618,56 +1861,14 @@ maskline:
   res 3,(iy+5)
 #endif
 
-  ld  a,0
-gameCar =$-1
-  rra ;and _datalink
-  jr  nc,initfinished ;no link
-  xor a
-SwapPos: ;$18 xx -> $E6 xx
-         ; jr xx -> and xx
-  jr  sethost
-setclient:
-  ld  (worm1+left),a ;worm 1...
-  ld  (worm3+left),a ;and worm 3 via link
-  ld  hl,worm1+name
-  ld  b,9
-  cal recvstuff
-  ld  hl,worm2+name
-  ld  b,9
-  cal sendstuff
-  jr  initfinished
-sendstuff:
-  psh bc
-  ld  c,(hl)
-  inc hl
-  psh hl
-  cal Qsend
-  pop hl
-  pop bc
-  dnz sendstuff
-  ret
-recvstuff:
-  psh bc
-  psh hl
-  cal Qrecv
-  pop hl
-  ld  (hl),c
-  inc hl
-  pop bc
-  dnz recvstuff
-  ret
-sethost:
- halt
- dnz sethost
-  ld  (worm2+left),a ;worm 2+4..
-  ld  (worm4+left),a ;..over link
-  ld  hl,worm1+name
-  ld  b,9
-  cal sendstuff
-  ld  hl,worm2+name
-  ld  b,9
-  cal recvstuff
-initfinished:
+  ld  a,(gameCar)
+  rla ;bit _bitmultpeas,a
+  cal c,multnewpea ;a=0
+  rla ;bit _bitfood,a
+  cal nc,NewPea
+nofood:
+  ld  bc,(worm1+pos)
+  cal DisplayField
 
   ld  b,startdelay
 ReadyDelay:
@@ -1675,9 +1876,11 @@ ReadyDelay:
   dnz ReadyDelay
   cal releasekeys
 
-;-----------------------------
-;----------- LOOP ------------
-;-----------------------------
+;_______________________________________________________________________________
+;  ______ _______ _______ _______               _____   _____   _____
+; |  ____ |_____| |  |  | |______       |      |     | |     | |_____]
+; |_____| |     | |  |  | |______ _____ |_____ |_____| |_____| |
+;_______________________________________________________________________________
 
 GameLoop:
   ld  bc,(worm1+pos) ;camera worm #1
@@ -1768,85 +1971,110 @@ handleworms:
   inc (hl) ;0..nrworms-1
   dnz handleworms
 
-;-----------------------------
-;---------- keys -------------
-;-----------------------------
+;_______________________________________________________________________________
+; _     _ _______ __   __ _______
+; |____/  |______   \_/   |______
+; |    \_ |______    |    ______|
+;_______________________________________________________________________________
 
 HandleKeys:
   ld  a,%10111111
   out (1),a
   in  a,(1)
   rla ;MORE?
-  jr  c,CheckExit
-  ld  bc,$0103
-  out (c),b
-  halt ;pause/off
-  ld  b,11
-  out (c),b
-
+  jr  nc,disppausemenu
 CheckExit:
   rla  ;=$17 (c=EXIT-key)
       ;or$A7 (c=0)
   jp  c,GameLoop
-  jr  Exit
+  jp  Exit ;jr?
 
-WormDead:
-#ifdef invincible
-  jp  stopworm
-#else
-  ld  a,2
-  ld  (flashtime),a
-  ld  (ix+delay),respawndelay
+;--- pause menu ---
 
-thislevel =$+1
-  ld  hl,0
-  ld  de,0
-handledworm =$-2
-  add hl,de
-  add hl,de
-  add hl,de
-  ld  a,(hl)
-  inc hl
-  ld  (ix+heading),a
-  ld  a,(hl)
-  ld  (ix+pos),a ;y
-  inc hl
-  ld  a,(hl)
-  ld  (ix+pos+1),a ;x
-  xor a
-  ld  (ix+pos2),a ;y2
-  ld  (ix+pos2+1),a ;x2
+disppausemenu:
+  ld  hl,txtpMenu
+  ld  ix,pospMenu
+;Resume|Turn Off|Contrast|Exit||
+  ld  b,25*16/4
+  cal dispmenucommon
+  ld  hl,$FC00+$170 ;(*,23)
+  cal hr
+  ld  hl,$FC00+$310 ;(*,49)
+  cal hr
+  cal menupos
 
-  inc (ix+died)
-  dec (ix+lives)
+pauseMenu:
+  psh bc
+pmenuwaitkey:
+  cal ubergetkey
+  or  a
+  jr  z,pmenuwaitkey
+  pop bc ;GET_KEY
+  ld  d,b ;c=new b
+  cp  K_UP
+  jr  nz,pmenunotup
+  dec d
+pmenunotup:
+  cp  K_DOWN
+  jr  nz,pmenunotdown
+  inc d
+pmenunotdown:
   psh af
-  ld  de,10
-  ld  a,(Gametype)
-  cp  gamectf ;ctf no death penalty
-  cal nz,DecScore
+  cal menupos
+  ld  a,d ;new pos
+  and 3 ;0-3
+  ld  b,a
+  cal menupos
   pop af
-  ret nz ;HandleWorm done
-  ld  a,(wormbeglives)
-  or  a ;0=no live limit
-  ret z ;don't end game
-  ld  a,(gameCar)
-  and _datatime
-  jr  z,anyworm ;quit at any worm's death
-  ld  a,(nrworms) ;timematch: all worms must've died
-  ld  b,a ;# of worms
-  ld  hl,worm1+lives-(worm2-worm1)
-  ld  de,worm2-worm1
-  xor a ;check for 0 lives
-checklives:
-  add hl,de ;next worm
-  cp  (hl) ;lives==0?
-  ret nz ;any >0: don't exit
-  dnz checklives
-anyworm:
-  ld  a,$A7 ;exit@end of turn
-  ld  (CheckExit),a ;set exit state
-  ret ;finish turn
-#endif
+  cp  K_ENTER
+  jr  z,pselect
+  cp  K_SECOND
+  jr  nz,notpselect
+pselect:
+  ld  a,b
+  or  a ;1: continue
+  jr  z,donepausing
+  dec a ;2: off
+  jr  z,turnoff
+  dec a ;3: contrast
+  jr  z,pauseMenu
+  jr  Exit ;4: exit
+notpselect
+  cp  K_EXIT
+  jr  z,donepausing
+  ld  hl,CONTRAST
+  sub K_LEFT
+  jr  z,contrastdown
+  dec a ;K_RIGHT
+  jr  nz,pauseMenu
+contrastup:
+  inc (hl)
+  jr  setcontrast
+contrastdown:
+  dec (hl)
+setcontrast:
+  ld  a,(hl)
+  out (2),a
+  jr  pauseMenu
+
+turnoff:
+  ld  bc,$0103
+  out (c),b
+  halt ;pause/off
+  ld  b,11
+  out (c),b
+  ld  b,1
+  jr  pauseMenu
+
+donepausing:
+  cal releasekeys
+  jp  GameLoop
+
+;_______________________________________________________________________________
+;  ______ _______ _______ _______        _____  _    _ _______  ______
+; |  ____ |_____| |  |  | |______       |     |  \  /  |______ |_____/
+; |_____| |     | |  |  | |______ _____ |_____|   \/   |______ |    \_
+;_______________________________________________________________________________
 
 Exit:
   cal releasekeys
@@ -1937,7 +2165,7 @@ nonewdmwinner:
   dnz finddmwinner
   ld  (dmwinner),a
 
-;---display worms---
+;--- display worms ---
 
   ld  a,(nrworms)
   ld  b,a
@@ -1994,7 +2222,8 @@ notwinner:
   psh hl
   inc hl ;worm+score
   cal _ldHLind
-  cal showHL
+  cal _D_HL_DECI
+  cal _puts ;showHL
   pop hl
 
   ld  bc,worm2-worm1
@@ -2065,7 +2294,8 @@ NotNewHigh:
   ld  hl,highsave+2
   cal _puts
   ld  hl,(highsave)
-  cal showHL
+  cal _D_HL_DECI
+  cal _puts ;showHL
 hiscorecheckdone:
 
 waitkey:
@@ -2143,16 +2373,18 @@ iswinner:
   ld  a,'*'
   jp  _putc ;..then put *
 
-;-----------------------------
-;----------- worm ------------
-;-----------------------------
+;_______________________________________________________________________________
+; _  _  _  _____   ______ _______       _______ _______ _     _ _______ _______
+; |  |  | |     | |_____/ |  |  |       |______    |    |     | |______ |______
+; |__|__| |_____| |    \_ |  |  | _____ ______|    |    |_____| |       |
+;_______________________________________________________________________________
 
 respawncheck:
   cp  respawndelay-1
   jr  nz,unnamedlabel
   cal saverespawncounter
 removeworm:
-  ld  h,(ix+tail+1)
+  ld  h,(ix+tail+1) ;&
   ld  l,(ix+tail)
   ld  d,(ix+head+1)
   ld  e,(ix+head)
@@ -2167,6 +2399,8 @@ removewormloop:
   cal res4pixels
   pop hl
   inc (ix+grow)
+  jr  nz,DoesWormTailEqualsWormHead
+  inc (ix+grow+1) ;+256
 DoesWormTailEqualsWormHead:
   cal _cphlde
   jr  nz,removewormloop
@@ -2177,6 +2411,7 @@ DoesWormTailEqualsWormHead:
   ld  a,0
 beginsize =$-1
   ld  (ix+grow),a
+  ld  (ix+grow+1),0 ;high
 safewormsizedone:
 
   ;de=ix+head
@@ -2255,7 +2490,11 @@ inlink:
   ld  l,c
   ret
 
-;------- handle worm ---------
+;_______________________________________________________________________________
+; _  _  _  _____   ______ _______       _______        _____ _    _ _______
+; |  |  | |     | |_____/ |  |  |       |_____| |        |    \  /  |______
+; |__|__| |_____| |    \_ |  |  | _____ |     | |_____ __|__   \/   |______
+;_______________________________________________________________________________
 
 HandleWorm:
   xor a
@@ -2294,6 +2533,7 @@ donediddelydone:
   ld  e,(ix+pos2)
   ld  d,(ix+pos2+1)
 
+#ifdef optdie
   psh hl
   ld  hl,previouspos
   ld  (hl),c
@@ -2305,8 +2545,9 @@ donediddelydone:
   inc hl
   ld  (hl),d
   pop hl
+#endif
 
-;-------- move worm ----------
+;--- move worm ---
 
 Wormmove:
   psh bc                        ; >> pos
@@ -2359,43 +2600,46 @@ GotFour:
   rl  d
   jp  nc,Drawworm
 
-;--------- worm hit ----------
+;--- worm hit ---
 
 Hitworm:
   ld  a,(gameCar)
-  ld  h,a
 #ifdef optdie
-  and _datadie
-  cal z,checkhitotherworm
+; if race - originally
+  ld  h,a
+  cal checkhitotherworm
   ld  a,h
 #endif
-  and _datamultpeas ;&&bit
+  bit _bitctfpeas,a
   jr  nz,chkctfpeas
-  ld  a,h
-  and _datafood
-  jp  z,WormDead ;no food
+  rla ;and _datamultpeas
+  jp  c,chkmultpeas
+  rla ;and _datafood
+  jp  c,WormDead ;no food
 
   ld  hl,0
 PeaY =$-2
 PeaX =$-1
   cal chkpeahit
   jp  nc,WormDead
+pickpea:
   cal DrawPea ;remove pea
+  cal NewPea
+pickpea_:
   ld  a,(ix+grow)
   add a,15
 peagrowth =$-1
   ld  (ix+grow),a
-  cal NewPea
+  jr  nc,wormset2grow
+  inc (ix+grow+1)
+wormset2grow:
   ld  hl,Left
-  dec (hl)
+  dec (hl) ;dec left before display
   psh af
   ld  e,10
   cal IncScore
   pop af
   jp  nz,Drawworm ;continue
-  ld  a,(gameCar)
-  and _datafoodl
-  jp  z,Drawworm
   ld  a,(Gametype)
   or  a
   jp  nz,Exit ;stack restored
@@ -2414,7 +2658,7 @@ peagrowth =$-1
   pop hl                         ; << call
   pop hl                         ; << call
   ld  (ix+delay),2
-  jp  StartLevel
+  jp  Nextlevel
 
 chkpeahit: ;bc=ownpos;hl=peapos (destr:ad)
   ld  a,(sprsize)
@@ -2452,14 +2696,12 @@ sel_otherpea:
   pop af ;which flag? (=and %1)
   ld  hl,drawctfpea1 ;restore #1
   jr  z,nottheotherflag
-  ld  hl,drawctfpea2 ;restore #2
+  inc hl ;ld hl,drawctfpea2
+  inc hl ;restore #2
 nottheotherflag:
   ld  (hl),3 ;draw delay 3 turns
   ret
 
-drawctfpea1: .db 0
-drawctfpea2: .db 0
-
 hitflag: ;correct pea hit
   ld  a,e ;pop a
   xor %11 ;invert flag taken + ownflag
@@ -2478,8 +2720,37 @@ hitflag: ;correct pea hit
   ld  (drawctfpea2),a ;..both flags
   jr  Drawworm
 
-;-----------------------------
+chkmultpeas:
+  ld  de,peaspos-1
+  ld  hl,lefttotalb
+lefttotal =$+1
+  ld  (hl),0
+chkmultpealoop:
+  ex  de,hl
+  inc hl
+  ld  e,(hl)
+  inc hl
+  ld  d,(hl)
+  ex  de,hl
+  psh de
+  cal chkpeahit
+  pop de
+  jr  c,pickmultpea
+  ld  hl,lefttotalb
+  dec (hl)
+  jr  nz,chkmultpealoop
+  jp  WormDead
+pickmultpea:
+  ld  a,247
+  ld  (de),a
+  dec de
+  ld  (de),a
+  cal DrawPea
+  jp  pickpea_
+
+;--- misc ---
 
+#ifdef optdie
 nextotherwormbit:
   ld  a,c
   sub (hl)
@@ -2499,6 +2770,7 @@ ChkWorm:
   cal _cphlde
   jr  nz,nextotherwormbit
   ret ;z
+#endif
 
 checkhitlapline:
   ld  a,(peaspos) ;63
@@ -2527,38 +2799,7 @@ checklap:
   pop bc
   ret
 
-#ifdef optdie
-checkhitotherworm:
- .db  $dd,$7d ;ld a,lx
-  cp  worm2&255
- psh ix
- jr nz,chkworm2 ; ret nz
-  ld  ix,worm1
- jr chkworm
-chkworm2:
- ld ix,worm2
-
-chkworm:
- ld h,(ix+tail+1)
- ld l,(ix+tail)
- ld e,(ix+head)
- ld d,(ix+head+1)
-  cal ChkWorm
- pop ix
-  ret z ;not hit
-  pop bc ;call
-#endif
-stopworm:
-  ld  bc,0
-previouspos =$-2
-  ld  de,0
-  ld  (ix+pos),c
-  ld  (ix+pos+1),b
-  ld  (ix+pos2),e
-  ld  (ix+pos2+1),d
-  ret
-
-;-------- draw worm ----------
+;--- draw worm ---
 
 Drawworm:
   ld  c,(ix+pos)
@@ -2584,10 +2825,14 @@ Drawworm:
   ld  (ix+head),l
   ld  (ix+head+1),h
 
-  ld  a,(ix+grow)
-  dec a
+  ld  l,(ix+grow)
+  ld  h,(ix+grow+1)
+  dec hl
+  ld  a,h
+  or  l
   jr  z,removetail
-  ld  (ix+grow),a
+  ld  (ix+grow),l
+  ld  (ix+grow+1),h
   ret
 
 removetail:
@@ -2600,6 +2845,13 @@ removetail:
   cal resbit
   ld  (ix+tail),l
   ld  (ix+tail+1),h
+  psh hl
+  cal res4pixels
+  pop hl
+  ld  c,(hl)
+  inc hl
+  ld  b,(hl)
+  jr  set4pixels
 
 res4pixels:
   cal ResPixel
@@ -2615,9 +2867,11 @@ ResPixel:
   ld  (hl),a
   ret
 
-;-----------------------------
-;---------- ball -------------
-;-----------------------------
+;_______________________________________________________________________________
+; ______   _____  _     _ __   _ _______ __   __ ______  _______
+; |_____] |     | |     | | \  | |         \_/   |_____] |_____| |      |
+; |_____] |_____| |_____| |  \_| |_____     |    |_____] |     | |_____ |_____
+;_______________________________________________________________________________
 
 handlethoseneatlittleballs:
   ld  hl,balls
@@ -2710,9 +2964,11 @@ checkballhit:
   pop de
   ret
 
-;-----------------------------
-;----------- procs -----------
-;-----------------------------
+;_______________________________________________________________________________
+;  _____   ______  _____  _______ _______
+; |_____] |_____/ |     | |       |______
+; |       |    \_ |_____| |_____  ______|
+;_______________________________________________________________________________
 
 releasekeys:
   halt
@@ -2743,6 +2999,92 @@ Seed =$+1
   add a,2
   ret
 
+WormDead:
+#ifdef invincible
+  jp  stopworm
+#endif
+  ld  a,2
+  ld  (flashtime),a
+  ld  (ix+delay),respawndelay
+
+thislevel =$+1
+  ld  hl,0
+  ld  de,0
+handledworm =$-2
+  add hl,de
+  add hl,de
+  add hl,de
+  ld  a,(hl)
+  inc hl
+  ld  (ix+heading),a
+  ld  a,(hl)
+  ld  (ix+pos),a ;y
+  inc hl
+  ld  a,(hl)
+  ld  (ix+pos+1),a ;x
+  xor a
+  ld  (ix+pos2),a ;y2
+  ld  (ix+pos2+1),a ;x2
+
+  inc (ix+died)
+  dec (ix+lives)
+  psh af
+  ld  a,(Gametype)
+  cp  gamectf ;ctf no death penalty
+  cal nz,DecScore10
+  pop af
+  ret nz ;HandleWorm done
+  ld  a,(wormbeglives)
+  or  a ;0=no live limit
+  ret z ;don't end game
+  ld  a,(gameCar)
+  and _datatime
+  jr  z,anyworm ;quit at any worm's death
+  ld  a,(nrworms) ;timematch: all worms must've died
+  ld  b,a ;# of worms
+  ld  hl,worm1+lives-(worm2-worm1)
+  ld  de,worm2-worm1
+  xor a ;check for 0 lives
+checklives:
+  add hl,de ;next worm
+  cp  (hl) ;lives==0?
+  ret nz ;any >0: don't exit
+  dnz checklives
+anyworm:
+  ld  a,$A7 ;exit@end of turn
+  ld  (CheckExit),a ;set exit state
+  ret ;finish turn
+
+#ifdef optdie
+checkhitotherworm:
+ .db  $dd,$7d ;ld a,lx
+  cp  worm2&255
+  psh ix
+  jr  nz,chkworm2 ;ret nz
+  ld  ix,worm1
+  jr  chkworm
+chkworm2:
+  ld  ix,worm2
+chkworm:
+  ld  h,(ix+tail+1)
+  ld  l,(ix+tail)
+  ld  e,(ix+head)
+  ld  d,(ix+head+1)
+  cal ChkWorm
+  pop ix
+  ret z ;not hit
+  pop bc ;call
+stopworm:
+  ld  bc,0
+previouspos =$-2
+  ld  de,0
+  ld  (ix+pos),c
+  ld  (ix+pos+1),b
+  ld  (ix+pos2),e
+  ld  (ix+pos2+1),d
+  ret
+#endif
+
 CheckPea: ;@hl; destr:abcde
   ld  c,l
   ld  a,(sprsize)
@@ -2779,6 +3121,23 @@ tryDrawPea: ;hl=peapos;bc=dopea
   jr  z,DrawPea
   ret
 
+multnewpea:
+  ld  de,peaspos
+  ld  a,(Left)
+  ld  b,a
+multpealoop:
+  psh bc
+  psh de
+  cal NewPea
+  pop de
+  ld  hl,PeaY
+  ldi
+  ldi
+  pop bc
+  dnz multpealoop
+  ld  a,-1
+  ret
+
 NewPea:
   ld  a,(FieldWidth)
   add a,127-4
@@ -2788,9 +3147,10 @@ NewPea:
   add a,56-4
   cal randompos
   ld  l,a
-  ld  (PeaY),hl
   cal CheckPea
   jr  nz,NewPea
+  cal sendnewpeaoverlink
+  ld  (PeaY),hl
 DrawPea: ;hl=(PeaY)
   ld  b,h
   ld  c,l
@@ -2836,11 +3196,18 @@ nextbitok:
   dnz sprloopy
   ret
 
-;----------- score -----------
+;_______________________________________________________________________________
+; _______ _______  _____   ______ _______
+; |______ |       |     | |_____/ |______
+; ______| |_____  |_____| |    \_ |______
+;_______________________________________________________________________________
 
-timematchscore: ;piece of crap checking whether you've already won in timematch
+timematchscore:
+  ;piece of crap checking whether you've already won in timematch
   ld  a,(nrworms)
   ld  b,a ;# of worms
+  dec a
+  ret z ;singleplayer
   ld  hl,worm1+lives
   ld  de,worm2-worm1
   ld  a,(handledworm)
@@ -2851,7 +3218,7 @@ chktimematchover:
   jr  z,nneexxtt ;yourself
   cp  (hl)
   ret nz ;someone else still alive
-  dec hl ;+delay
+  dec hl ;+del0ay
   dec hl ;+score+1
   ld  a,(hl)
   cp  (ix+score+1)
@@ -2891,19 +3258,17 @@ extralives:
   inc (ix+lives)
   jr  scorecommon
 
-DecScore: ;&&&
-  ld  h,(ix+score+1)
-  ld  l,(ix+score)
-  or  a
-  sbc hl,de
-  jr  c,showstats ;<0=0
+DecScore10:
+  cal ldscoreinhl
+  ld  de,-10
+  add hl,de
+  jr  nc,showstats ;<0=0
   jr  scorecommon
 
 IncScore: ;inc score by e
   ld  d,0
 _IncScore ; inc by de
-  ld  h,(ix+score+1)
-  ld  l,(ix+score)
+  cal ldscoreinhl
   ld  a,(Gametype)
   or  a ;if singleplayer...
   jr  z,extralives
@@ -2926,7 +3291,8 @@ showstats:
   ld  hl,$FC00 ;&&&only necessary in deathmatch
   ld  b,6*16-1 
 clearstats:
-  ld  (hl),a ;=0
+ ; ld  (hl),a ;=0
+ ;&&&&&&
   inc hl
   dnz clearstats
 forceshowstats:
@@ -2979,15 +3345,19 @@ showstat: ;(multiplayer)
   cp  gamedeathm
   jr  z,showlives
 showscore:
-  ld  h,(ix+score+1)
-  ld  l,(ix+score)
+  cal ldscoreinhl
   cal _D_HL_DECI
   jr  __vputs
 showlives:
   ld  a,(ix+lives)
-showA: ;small w/o leading 0 (&&&combine w/ cshowA0??)
-  cp  10
-  jr  c,showleastsign
+  jr  cshowA00
+
+cshowA: ;small w/ leading 0
+  or  a
+  jr  nz,cshowA00
+  ld  hl,txtNone
+  jp  _vputs
+cshowA00:
   ld  l,a
   ld  h,0
   cal _divHLby10
@@ -2996,7 +3366,7 @@ showA: ;small w/o leading 0 (&&&combine w/ cshowA0??)
   add a,'0'
   cal __vputmap
   pop af
-showleastsign:
+cshowA0:
   add a,'0'
 __vputmap:
   psh ix
@@ -3009,24 +3379,29 @@ showstatsS: ;(singleplayer)
   cal __vputs
   ld  a,0
 Level =$-1
-  cal showA
+  cp  10
+  psh af
+  cal c,cshowA0
+  pop af
+  cal nc,cshowA00
 
-  ld  a,98
+  ld  a,97
   ld  (_penCol),a
   cal showscore
   ld  a,123
   ld  (_penCol),a
-  cal showlives
+  ld  a,(ix+lives)
+  cal cshowA0 ;showlives
   ld  a,(gameCar)
   and _datafoodl
   pop ix
   ret z
 showleft:
-  ld  a,31
+  ld  a,33
   ld  (_penCol),a
   ld  a,0
 Left =$-1
-  cal cshowA0
+  cal cshowA
 
   ld  hl,txtLeft
 __vputs:
@@ -3051,27 +3426,6 @@ showA0: ;big w/ leading 0
   add a,'0'
   jp  _putc
 
-cshowA0: ;small w/ leading 0
-  or  a
-  jr  nz,cshowavalue
-  ld  hl,txtNone
-  jp  _vputs
-cshowavalue:
-  ld  l,a
-  ld  h,0
-  cal _divHLby10
-  psh af
-  ld  a,l
-  add a,'0'
-  cal __vputmap
-  pop af
-  add a,'0'
-  jp  __vputmap
-
-showHL:
-  cal _D_HL_DECI
-  jp  _puts
-
 _D_HL_DECI:
   ld  de,savestr+4
   ld  b,5
@@ -3086,11 +3440,20 @@ ldhld:
 savestr:
   .db "00000",0
 
-;-----------------------------
+ldscoreinhl:
+  ld  h,(ix+score+1)
+  ld  l,(ix+score)
+  ret
 
-DisplayField:
-  ld  a,c
-  sub 29
+;_______________________________________________________________________________
+; ______  _____ _______  _____         _______ __   __
+; |     \   |   |______ |_____] |      |_____|   \_/
+; |_____/ __|__ ______| |       |_____ |     |    |
+;_______________________________________________________________________________
+
+DisplayField:  ;all done by Jonah Cohen iirc. (-scrolling is scary-)
+  ld  a,c      ;not the fastest routine outthere, but it does get the job done.
+  sub 29       ;if something doesn't work, I probably _did_ alter it ;)
   jr  nc,NotMinYScroll
   xor a
 NotMinYScroll:
@@ -3310,7 +3673,11 @@ ChunkRow:
   ld  c,$37 ;scf
   ret
 
-;----------- draw ------------
+;_______________________________________________________________________________
+; ______   ______ _______ _  _  _
+; |     \ |_____/ |_____| |  |  |
+; |_____/ |    \_ |     | |__|__|
+;_______________________________________________________________________________
 
 ;--- pixel ---
 
@@ -3359,7 +3726,7 @@ FindPixel: ;(b,c) to hl:a
 FP_Bit =$+1
   set 0,a
 
-  psh de ;&&&
+  psh de ;&&& C`MON! THIS IS UGLY! *hit myself*
   ld  de,ScrBuffer
   add hl,de
   pop de
@@ -3397,6 +3764,14 @@ drawsmtn:
 ; dec a ;4 = circle
 ; jp  z,drawcircle
 
+;--- circle ---
+
+;IMHO, one weery nice routine. Oh lemme be proud just *once*.
+;Not perfect since it uses a screwy lineroutine to draw stuff
+;but, just _look_ at it. Ain't it grand? Z80 at its best.
+;Nice comments, also. So here we go: Shiar's Circle Routine:
+;(using the Bresenham method)
+
 drawcircle: ;(d,e),h   ;de=x,y; h=z
   ld  c,h              ;c=yy=z
   ld  a,h
@@ -3460,7 +3835,9 @@ circledraw:               ;destr:de
   ld  b,c
   ld  c,a              ;ex b,c
   pop hl
-  ret
+  ret                  ;thats it
+
+;--- box ---
 
 drawbox: ;(d,e)-(h,l)
   ld  b,l ;Delta-y
@@ -3472,6 +3849,8 @@ boxloop:
   dnz boxloop
   ret
 
+;--- fatline ---
+
 drawfatline:
   cal drawline
   inc d
@@ -3484,159 +3863,145 @@ drawfatline:
   dec h
 ; jp  drawline
 
-;LINE (d,e)-(h,l)
-;destroyes a
+;--- line ---
+
+;A lot like Scabby's line routine in Wonderworm (dunno whether he wrote it
+;himself tho.) I did make a few optimizations and commented the thing.
+;A nice routine (also Bresenham), but for Wormy not perfect since really
+;large lines (>128 pixels in length or something) won't be flawless.
 
 drawline: ;(d,e)-(h,l)
   psh bc  ;destr: a
   psh hl
   psh de
-  ld  a,d
-  cp  h
-  jr  c,lineOrdered
-  ex  de,hl
-lineOrdered:
+  ld  a,d              ;a=d=x
+  cp  h                        ;h=xx
+  jr  c,lineXincs      ;if x>xx
+  ex  de,hl            ;make x<xx
+lineXincs:             ;deltax=|deltax|
   ld  b,d
-  ld  c,e
-  psh hl
-  psh bc
-  cal FindPixel
-  pop bc
-  pop de
-connectedLine:
-  psh hl
-  ld  h,c
-  ld  c,a
-  ld  a,d
-  sub b
-  ld  b,a
-  ld  a,e
-  jr  nz,LineNotPoint
-  cp  h
-  jr  nz,LineNotPoint
-  pop hl
-  jr  DoneLine
-LineNotPoint:
-  sub h
-  ld  de,32
-  jr  nc,LinePositiveY
-  neg
-  ld  de,-32
-LinePositiveY:
-  cp  b
-  jr  nc,SteepLine
-  add a,a
-  ld  (line2sm+1),a
+  ld  c,e              ;bc=x,y
+  psh hl               ;hl=xx,yy
+  cal FindPixel                ;  (ahl=x+y)
+  ex  (sp),hl          ;       :hl>
+  ld  c,a              ;c=mask (always)
+  ld  a,h              ;a=xx
+  sub b                        ;xx-x
+  ld  b,a              ;b=deltax (always>0)
+  ld  a,l              ;a=yy
+  jr  nz,lineexists    ;deltax!=0
+  cp  e                        ;yy==y
+  jr  nz,lineexists    ;deltay!=0
+  pop hl               ;       :<hl
+  jr  DoneLine         ;line is 1 pixel
+lineexists:
+  sub e                        ;a=deltay
+  ld  de,32            ;go down (def)
+  jr  nc,lineYincs     ;cf = deltay<0
+  neg                  ;a=|deltay|
+  ld  de,-32           ;go up
+lineYincs:             ;lets assume y incs as well
+  cp  b                
+  jr  nc,SteepLine     ;deltay<deltax?
+  ld  (line2sm+1),a    ;a=deltay
   ld  h,a
   xor a
   sub b
-  add a,a
-  ld  (line1sm+1),a
-  ld  a,h
-  sub b
-  pop hl
-LineLoopGentle:
-  psh af
+  ld  (line1sm+1),a    ;a=-deltax
+  rra
+  add a,h              ;a=deltay-deltax/2
+  pop hl               ;       :<hl
+
+linedrawxloop:
+  psh af               ;init z=dy-dx
   ld  a,(hl)
-  or  c
+  or  c                        ;|c:hl
   ld  (hl),a
   rrc c
   jr  nc,$+3
-  inc hl
+  inc hl               ;(x++,y)
   pop af
-  jp  m,line2sm
+  jp  m,line2sm                ;z<0
 line1sm:
-  add a,0
-  add hl,de
+  add a,0              ;z-=deltax
+  add hl,de            ;(x,y++)
 line2sm:
-  add a,0
-  dnz LineLoopGentle
+  add a,0              ;z+=deltay
+  dnz linedrawxloop    ;|deltax| times
+
 DoneLine:
   ld  a,(hl)
-  or  c
-  ld  (hl),a
+  or  c                        ;|c:hl
+  ld  (hl),a           ;draw end
   pop de
   pop hl
   pop bc
   ret
+
 SteepLine:
-  ld  h,a
-  neg
-  add a,a
+  ld  h,a              ;h=deltay
+  neg                  ;a=-deltay
   ld  (line3sm+1),a
   ld  a,b
-  add a,a
-  ld  (line4sm+1),a
-  sub h
-  ld  b,h
-  pop hl
-LineLoopSteep:
-  psh af
+  ld  (line4sm+1),a    ;a=deltax
+  sub h                        ;a=deltax-deltay
+  ld  b,h              ;b=deltay
+  pop hl               ;       :<hl
+linedrawyloop:
+  psh af               ;init z=2(dx-dy)
   ld  a,(hl)
-  or  c
+  or  c                        ;|c:hl
   ld  (hl),a
-  add hl,de
+  add hl,de            ;(x,y++)
   pop af
-  jp  m,line4sm
+  jp  m,line4sm                ;z<0
 line3sm:
-  add a,0
+  add a,0              ;z-=deltay
   rrc c
   jr  nc,$+3
-  inc hl
+  inc hl               ;(x++,y)
 line4sm:
-  add a,0
-  dnz LineLoopSteep
+  add a,0              ;z+=deltax
+  dnz linedrawyloop    ;|deltay| times
   jr  DoneLine
 
-;-----------------------------
-;----------- link ------------
-;-----------------------------
 
-linkok:
-  ld  a,D0HD1H
-  out (7),a    ;raise both wires = link ok
+;--- foo ---
+;wild insert
+receivenewpeaoverlink:
+  cal Qrecv ;Crecv
+  ld  l,c
+  cal Qrecv ;Crecv
+  ld  h,c
   ret
+sendnewpeaoverlink:
+  ld  a,(gameCar)
+  rra ;and _datalink
+  ret nc ;no link
+  ld  a,(ix+left)
+  or  a
+  jr  z,receivenewpeaoverlink
+  ld  c,l ;send pea's pos
+  cal Qsend
+  ld  c,h
+; jr  Qsend ;that's why it's here
+
+;..and last but not least..:
+;_______________________________________________________________________________
+;        _____ __   _ _     _   /
+; |        |   | \  | |____/   /
+; |_____ __|__ |  \_| |    \_ .
+;_______________________________________________________________________________
 
 timeout = $8000
 lossout = 20
 
-checklink:             ;load wires in A and check for timeout
-  dec de               ;decrease timer
-  ld  a,d
-  or  e
-  jr  nz,linkfine      ;ok if de>0
-;de==0 = timeout
-  cal linkok
-  ld a,0               ;losses so far
-linklosses =$-1
-  inc a                ;and this is another one
-  ld (linklosses),a
-  pop de               ;return from link
-  cp  lossout          ;quit if too many errors
-  ret c                ;otherwise just continue
-linkerr:
-  jp  Exit
-linkfine:
-  in  a,(7)
-  and %11
-  ret
-
-;---- SEND ----
+;--- send ---
 
-Csend:
-  ld  b,32
-csendwait:
-  nop
-  dnz csendwait
-  cal Qsend
-  jr  c,Csend
-  ret
-
-Qsend:         ;try to send 8 bits in C; CF=error --- destr:abcde
+Qsend:         ;--- try to send 8 bits in C; CF=error --- destr:abcde ---
   nop \ nop
-  in  a,(7)
-  and %11              ;both wires low = exit signal
-  jr  z,linkerr        ;error otherwise
+  cal linkfine
+  jr  z,linkerr                ;both wires low = exit signal
   ld  b,8              ;bits to send
 sendloop:
   ld  de,timeout
@@ -3649,8 +4014,7 @@ sendbit:
 sendwaitack:
   cal checklink                ;other calc must lower other wire
   jr  nz,sendwaitack
-  ld  a,D0HD1H         ;raise one, ok to raise other
-  out (7),a
+  cal linkok           ;raise one, ok to raise other (out (7),D0HD1H)
 sendfinish:
   cal checklink
   cp  %11              ;both raised (by other calc)
@@ -3661,24 +4025,24 @@ sendfinish:
   ld (linklosses),a    ;reset number of losses
   ret                  ;=ok
 
-;---- RECV ----
+;--- recv ---
 
-Crecv:         ;--- receive 8 bits into A/C --- destr:abcdehl ---
+Crecv:         ;--- receive 8 bits into A/C --- destr:abCde ---
   cal Qrecv
   ret nc               ;return if all went ok
   jr  Crecv            ;and try again
 
-Qrecv:         ;--- receive 8 bits into A/C; CF=error --- destr:abcde ---
-  in  a,(7)
-  and %11
+Qrecv:         ;--- receive 8 bits into A/C; CF=error --- destr:abCde ---
+  cal linkfine
   jr  z,linkerr        ;both low = error, quit
   ld  b,8              ;bits to receive
 recvloop:
-  ld  de,timeout
+;  ld  de,timeout
 recvwait:
-  cal checklink
+  cal linktest ;checklink
   cp  %11
   jr  z,recvwait       ;both high = nothing sent (yet)
+  ld  de,timeout
   rra                  ;received bit in cf
   ld  a,D0LD1H
   jr  c,received       ;lower white wire as well
@@ -3689,18 +4053,53 @@ received:
 recvwaitack:
   cal checklink
   jr  z,recvwaitack    ;same wire will be raised again by other calc
-  ld  a,D0HD1H
-  out (7),a            ;raise both
+  cal linkok           ;raise both
 recvfinish:
   dnz recvloop         ;repeat for all bits
   xor a                        ;nc=no error
   ld (linklosses),a    ;reset number of losses
   ret
 
+;--- common ---
 
-;-----------------------------
-;---------- levels -----------
-;-----------------------------
+linkok:
+  ld  a,D0HD1H
+  out (7),a            ;raise both wires = link ok
+  ret
+
+checklink:             ;load wires in A and check for timeout
+  dec de               ;decrease timer
+  ld  a,d
+  or  e
+  jr  z,linktimeout    ;timeout if de==0
+linktest:
+  ld  a,$BF
+  out (1),a            ;select keys
+  in  a,(1)            ;input
+  bit 6,a              ;exit key
+  jp  z,Exit           ;break!
+linkfine:
+  in  a,(7)
+  and %11
+  ret
+linktimeout: ;de==0
+  cal linkok
+  ld a,0               ;losses so far
+linklosses =$-1
+  inc a                ;and this is another one
+  ld (linklosses),a
+  pop de               ;return from link
+  cp  lossout          ;quit if too many errors
+  ret c                ;otherwise just continue
+linkerr:
+  jp  Exit
+
+#ifdef intlevels
+;_______________________________________________________________________________
+; _____ __   _ _______              _______ _    _ _______        _______
+;   |   | \  |    |          |      |______  \  /  |______ |      |______
+; __|__ |  \_|    |    _____ |_____ |______   \/   |______ |_____ ______|
+;_______________________________________________________________________________
 
 LevelDef:
   .db 5,4,15,15,0,0    ;peas,speed,growth,begin_size,sprite,balls
@@ -3713,13 +4112,13 @@ LevelDef:
 
 LevelDefM:
   .db "Empty Arena",0
-  .db 8,4,15,15,0,0
+  .db 5,4,15,15,0,0
   .db $40,30,2,$C0,30,125, $00,2,64,$80,54,64
   .db 128,57
   .db 0
 
   .db "Fast Fun :))",0
-  .db 8,0,18,12,5
+  .db 8+128,0,18,12,5
   .db %1110000,%10001000,%10001000,%10001000,%1110000,0
   .db $40,30,2,$C0,30,125, $00,2,64,$80,54,64
   .db 128,57
@@ -3758,11 +4157,15 @@ LevelDefC: ;ctf
   .db 27,8,27,116
   .db 0
 
-;-----------------------------
-;---------- data -------------
-;-----------------------------
+#endif
+
+;_______________________________________________________________________________
+; ______  _______ _______ _______
+; |     \ |_____|    |    |_____|
+; |_____/ |     |    |    |     |
+;_______________________________________________________________________________
 
-wtPicture:
+wtPicture: ;title
 .db %00011110,%00000000,%00000000,%00000000,%00000000,%00000000,%00000000,%00000001,%11000000,%00000000,%00000000,%00000001,%10000000,%00000000,%00001111,%10000000
 .db %00111110,%00000000,%00000000,%00000000,%00000000,%00000000,%00000000,%00000001,%11100000,%00000000,%00000011,%11000010,%01000011,%10011100,%00110000,%01100000
 .db %01110000,%00000000,%00000000,%00000000,%00000000,%00000000,%00000000,%00000000,%01110000,%00111000,%00001100,%00110001,%00110100,%01100011,%01000000,%00010000
@@ -3780,7 +4183,7 @@ wtPicture:
 .db %00011111,%11101111,%11110000,%00111111,%10000111,%00000001,%10000000,%00000011,%11111111,%11110000,%00111110,%00011110,%00010111,%01011101,%10010101,%01100101
 .db %00001111,%10000011,%11100000,%00011111,%00000011,%00000001,%10000000,%00000000,%11111111,%11100000,%00000000,%00000000,%11100101,%01010101,%01000110,%01010011
 
-wtWormy:
+wtWormy: ;g/o
 .db %00000110,%00111000,%00000000,%00111100,%00001111,%10001110,%00111100,%01111000,%00001100,%11011000,%11100011,%10000000,%00000000,%11111100,%00111000,%00000000
 .db %00011001,%01000110,%11100000,%11000010,%00011100,%11011111,%01111110,%11111000,%00011110,%11011101,%11110111,%11000000,%00111111,%00000011,%01000110,%11100000
 .db %00100000,%10000011,%00010001,%00110010,%00011000,%00010011,%01101010,%11000000,%00111011,%01001101,%10000110,%01000000,%11000000,%00000000,%10000001,%00010000
@@ -3796,8 +4199,8 @@ txtMenu:  .db "Mode",0  ;1st menu item
          .db "Link",0  ;...
          .db "Worms",0
          .db "worm #",0
-         .db 0
-posMenu:  .dw $2418,$231E,$2824,$1F2A,$1730,$1936
+posMenu:  .db 5
+         .dw $2418,$231E,$2824,$1F2A,$1730
 txtMenuR: .db "controls",0
 txtoMenu: .db "Back",0  ;1st menu item
          .db "Lives",0 ;2nd
@@ -3805,15 +4208,23 @@ txtoMenu: .db "Back",0  ;1st menu item
          .db "Speed",0
          .db "Rotation",0
          .db "Growth",0
-posoMenu: .dw $2618,$251E,$2524,$222A,$1A30,$1C36
+posoMenu: .db 6
+         .dw $2618,$251E,$2524,$222A,$1A30,$1C36
+txtpMenu: .db "Resume",0  ;1st menu item
+         .db "Turn Off",0 ;2nd
+         .db "Contrast",$CF,5,0  ;...
+         .db "Exit",0
+pospMenu: .db 4
+         .dw $3418,$331E,$2F24,$3A2A
 txtGame:  .db "Singleplayer",0
 txtGame1: .db "Peaworm",0
 txtGame2: .db "Tron",0
 txtGame3: .db "Deathmatch",0
 txtGame4: .db "Foodmatch",0
-txtGame5: .db "Timematch",0
-txtGame6: .db "Race",0
-txtGame7: .db "CTF" ;,0
+txtGame5: .db "Multifoodmatch",0
+txtGame6: .db "Timematch",0
+txtGame7: .db "Race",0
+txtGame8: .db "CTF" ;,0
 
 posGame:  .db 0
          .db txtGame1-txtGame
@@ -3823,6 +4234,7 @@ posGame:  .db 0
          .db txtGame5-txtGame
          .db txtGame6-txtGame
          .db txtGame7-txtGame
+         .db txtGame8-txtGame
 
 txtNone:   .db "None",0
 txtDef:    .db "Default",0
@@ -3862,45 +4274,55 @@ txtposReady = 7
 txtGO:     .db 5,5,5,5,5," GAME OVER ",$CF,$CF,$CF,$CF,$CF,0
 #endif
 
+#ifdef intlevels
 datalevels: .dw LevelDef, LevelDefM
             .dw LevelDefT,LevelDefM
-            .dw LevelDefM,LevelDefM
+            .dw LevelDefM,LevelDefM,LevelDefM
             .dw LevelDefR,LevelDefC
-nrlevels:   .db 1,3,1,3,3,3,3,1     ;=defaults
-
-_datalink      = %0000001 ;linkplay
-_datafoodl     = %0000010 ;left=0 limit
-_datasingl     = %0000100 ;singleplayer=1 (=hiscore+keep_length+nextlevel)
-_datamultpeas  = %0001000 ;multiple peas=1
-_datatime      = %0010000 ;time incs score (timematch)
-_datafood      = %0100000 ;food present
-_datadie       = %1000000 ;worm dies on impact
-;reserved       %10000000 ;for future use
-
-savestart:        ;     �game info (see above)
-                  ;     | �lives (0=unlimited)
-                  ;     | | �nrworms (can be altered by user)
-                  ;     | | | �level (reset to 1 at startup)
-                  ;     | | | |  �speed/delay (-1=def)
-                  ;     | | | |  |  �grow speed (-1=none; 0=continuous)
-                  ;     | | | |  |  |  �turn speed (8=def)
-                  ;     | | | |  |  |  |  �score limit (0=none)
-gamesingle   =  0 ;used | | | |  |  |  |  |
-datasingle: .db  %1100110,3,1,1,-1,-1, 8, 0 ;Singleplayer (SP; 3 lives)
+nrlevels:   .db 2,3,1,3,3,3,3,3,1,0     ;=defaults
+#else
+datalevels: .dw 0,0,0,0,0,0,0,0,0
+nrlevels:   .dw 0,0,0,0,0
+#endif
+
+_datalink      = %00000001 ;linkplay
+_datafoodl     = %00000010 ;left=0 limit
+_datasingl     = %00000100 ;singleplayer=1 (=hiscore+keep_length+nextlevel)
+_datactfpeas   = %00001000 ;ctf peas=1
+_bitctfpeas    = 3
+_datatime      = %00010000 ;time incs score (timematch)
+_datafood      = %01000000 ;food present=0
+_bitfood       = 6
+_datamultpeas  = %10000000 ;multiple peas=1 (no _datafood)
+_bitmultpeas   = 7
+;reserved        %00100000 ;for future use
+
+savestart:        ;      �game info (see above)
+                  ;      | �lives (0=unlimited)
+                  ;      | | �nrworms (can be altered by user)
+                  ;      | | | �level (reset to 1 at startup)
+                  ;      | | | |  �speed/delay (-1=def)
+                  ;      | | | |  |  �grow speed (-1=none; 0=continuous)
+                  ;      | | | |  |  |  �turn speed (8=def)
+                  ;      | | | |  |  |  |  �score limit (0=none)
+gamesingle   =  0 ;      | | | |  |  |  |  |
+datasingle: .db  %00000110,3,1,1,-1,-1, 8, 0 ;Singleplayer (SP; 3 lives)
 gamepeas     =  1
-datapeas:   .db  %1100100,1,1,1,-1,-1, 8, 0 ;Peaworm (SP; 1 live)
+datapeas:   .db  %00000100,1,1,1,-1,-1, 8, 0 ;Peaworm (SP; 1 live)
 gametron     =  2
-datatron:   .db  %1010100,1,1,1,-1, 0, 8, 0 ;Tron (SP; 1 live, cont.growth)
-gamedeathm   =  3 ;used
-datadeathm: .db  %1000000,3,2,1,-1,49, 8, 0 ;Deathmatch (3 lives, 50 growth)
+datatron:   .db  %01010100,1,1,1,-1, 0, 8, 0 ;Tron (SP; 1 live, cont.growth)
+gamedeathm   =  3
+datadeathm: .db  %01000000,3,2,1,-1,49, 8, 0 ;Deathmatch (3 lives, 50 growth)
 gamefoodm    =  4
-datafoodm:  .db  %1100000,0,2,1,-1,-1, 8,10 ;Foodmatch (100 score)
-gametimem    =  5
-datatimem:  .db  %1010000,1,2,1,-1, 0, 8, 0 ;Timematch (tron, 100 score)
-gamerace     =  6 ;used
-datarace:   .db  %0000000,0,2,1,-1,-1, 8,10 ;Race (100 score, no collision)
-gamectf      =  7 ;used
-datactf:    .db  %1001000,0,4,1,-1,-1, 8,10 ;CTF (100 score)
+datafoodm:  .db  %00000000,0,2,1,-1,-1, 8,10 ;Foodmatch (100 score)
+gamemfoodm   =  5
+datamfoodm: .db  %11000010,0,2,1,-1,-1, 8,10 ;MuchFoodmatch (10 peas))
+gametimem    =  6
+datatimem:  .db  %01010000,1,2,1,-1, 0, 8, 0 ;Timematch (tron, 100 score)
+gamerace     =  7
+datarace:   .db  %01000000,0,2,1,-1,-1, 8,10 ;Race (100 score)
+gamectf      =  8
+datactf:    .db  %01001000,0,4,1,-1,-1, 8,10 ;CTF (100 score)
 
 worm1set:  .dw worm1p,worm1p
            .db %11110111,1,-01,K_RIGHT,K_LEFT
@@ -3916,11 +4338,13 @@ worm4set:  .dw worm4p,worm4p
 worm4name: .db "JIM     ",0
 
 defhiscrpos:
+#ifdef intlevels
   .db 0,0,"SHI"
   .db 0,0,"SHI"
   .db 0,0,"SHI"
   .db 0,0,"SHI"
   .db 0,0,"SHI"
+#endif
 
 Gametype:  .db 0 ;last gamemode
 
@@ -3940,7 +4364,7 @@ head     = 13  ;4B (head=tail)
 tail     = 15  ;also@next level
 storepos = 17
 reserv   = 18  ;loop (race:lap|ctf:pea)
-input    = 19
+input    = 19 ;---currently unused afaik---
 left     = 20
 right    = 21
 name     = 22
@@ -3950,52 +4374,26 @@ startdelay    = 30
 respawndelay  = 30
 maxnamelength = 8+1
 
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "WWW.SHIAR.ORG  WWW.SHIAR.ORG  "
-.db "   wormy@shiar.org",0
-
 defspritesz = 4
 defspriteimg:  .db %01100000
                .db %11110000
                .db %11110000
                .db %01100000
 
+#ifdef intlevels
 deflevels:
-  .db "Internal Levels" ;,0
+  .db "Internal Levels",0
+  .db "by Shiar" ;,0
 
   .db 0,deflevels/256,deflevels&255
+#endif
 templevels:
 
-;-----------------------------
-;----------- end -------------
-;-----------------------------
+;_______________________________________________________________________________
+; _______ _     _ _______ _______ . _______       _______
+;    |    |_____| |_____|    |    ' |______       |_____| |      |
+;    |    |     | |     |    |      ______| _____ |     | |_____ |_____
+;_______________________________________________________________________________
 
   .end