Fix jumper PST
[hachu.git] / hachu.c
diff --git a/hachu.c b/hachu.c
index def5a0d..50a34f0 100644 (file)
--- a/hachu.c
+++ b/hachu.c
@@ -10,7 +10,7 @@
 // promotions by pieces with Lion power stepping in & out the zone in same turn\r
 // promotion on capture\r
 \r
-#define VERSION "0.18"\r
+#define VERSION "0.19"\r
 \r
 //define PATH level==0 || path[0] == 0x1103a &&  (level==1 || path[1] == 0x6f0f6 && (level == 2 /*|| path[2] == 0x8710f && (level == 3 /*|| path[3] == 0x3e865 && (level == 4 || path[4] == 0x4b865 && (level == 5)))*/))\r
 #define PATH 0\r
 #define CHECKEXT\r
 #define LMR 4\r
 #define LIONTRAP\r
-#define WINGS\r
+#define XWINGS\r
 #define KINGSAFETY\r
 #define KSHIELD\r
 #define XFORTRESS\r
 #define PAWNBLOCK\r
+#define TANDEM 100 /* bonus for pairs of attacking light steppers */\r
 #define KYLIN 100 /* extra end-game value of Kylin for promotability */\r
 #define PROMO 0 /* extra bonus for 'vertical' piece when it actually promotes (diagonal pieces get half) */\r
 \r
 #define PST_BPPROM  (3*BW*BH+BH)\r
 #define PST_BJUMPER (4*BW*BH)\r
 #define PST_ZONDIST (4*BW*BH+BH)\r
+#define PST_ADVANCE (5*BW*BH)\r
+#define PST_RETRACT (5*BW*BH+BH)\r
+#define PST_WFLYER  (6*BW*BH)\r
+#define PST_BFLYER  (6*BW*BH+BH)\r
+#define PST_END     (7*BW*BH)\r
 \r
 typedef unsigned int Move;\r
 \r
@@ -614,7 +620,7 @@ int attackMaps[200*BSIZE], *attacks = attackMaps;
 char distance[2*BSIZE]; // distance table\r
 char promoBoard[BSIZE]; // flags to indicate promotion zones\r
 char rawFire[BSIZE+2*BWMAX]; // flags to indicate squares controlled by Fire Demons\r
-signed char PST[5*BSIZE];\r
+signed char PST[7*BSIZE];\r
 \r
 #define board     (rawBoard + 6*BHMAX + 3)\r
 #define fireBoard (rawFire + BWMAX + 1)\r
@@ -730,7 +736,7 @@ Range (signed char *r)
   int i, m=0;\r
   for(i=0; i<8; i++) {\r
     int d = r[i];\r
-    if(r[i] < 0) d == r[i] >= L ? 2 : 36;\r
+    if(r[i] < 0) d = r[i] >= L ? 2 : 36;\r
     if(d > m) m = d;\r
   }\r
   return m;\r
@@ -866,11 +872,14 @@ SetUp(char *array, int var)
   for(i=0; i<BH; i++) for(j=0; j<BH; j++) board[BW*i+j] = EMPTY;\r
   for(i=WHITE+2; i<=last[WHITE]; i+=2) if(p[i].pos != ABSENT) {\r
     int g = p[i].promoGain;\r
+    if(i == kylin[WHITE]) p[i].promoGain = 1.25*KYLIN, p[i].value += KYLIN;\r
+//    if(j > 0 && p[i].pst == PST_STEPPER) p[i].pst = PST_WPPROM; // use white pre-prom bonus\r
+    if(j > 0 && p[i].pst == PST_STEPPER && p[i].value >= 100)\r
+       p[i].pst = p[i].value <= 150 ? PST_ADVANCE : PST_NEUTRAL; // light steppers advance\r
+    if(j > 0 && p[i].bulk == 6) p[i].pst = PST_WFLYER, p[i].mobWeight = 4; // SM defends zone\r
     if((j = p[i].promo) > 0 && g)\r
-      p[i].promoGain = (p[j].value - p[i].value - g)*1.25, p[i].value = p[j].value - g;\r
+      p[i].promoGain = (p[j].value - p[i].value - g)*0.9, p[i].value = p[j].value - g;\r
     else p[i].promoGain = 0;\r
-    if(i == kylin[WHITE]) p[i].promoGain = 1.25*KYLIN, p[i].value += KYLIN;\r
-    if(j > 0 && p[i].pst == PST_STEPPER) p[i].pst = PST_WPPROM; // use white pre-prom bonus\r
     board[p[i].pos] = i;\r
     rootEval += p[i].value + PST[p[i].pst + p[i].pos];\r
     promoDelta += p[i].promoGain;\r
@@ -878,12 +887,15 @@ SetUp(char *array, int var)
   } else p[i].promoGain = 0;\r
   for(i=BLACK+2; i<=last[BLACK]; i+=2) if(p[i].pos != ABSENT) {\r
     int g = p[i].promoGain;\r
+//    if(j > 0 && p[i].pst == PST_STEPPER) p[i].pst = PST_BPPROM; // use black pre-prom bonus\r
+    if(j > 0 && p[i].pst == PST_STEPPER && p[i].value >= 100)\r
+       p[i].pst = p[i].value <= 150 ? PST_RETRACT : PST_NEUTRAL; // light steppers advance\r
+    if(j > 0 && p[i].pst == PST_WJUMPER) p[i].pst = PST_BJUMPER;  // use black pre-prom bonus\r
+    if(j > 0 && p[i].bulk == 6) p[i].pst = PST_BFLYER, p[i].mobWeight = 4; // SM defends zone\r
     if((j = p[i].promo) > 0 && g)\r
-      p[i].promoGain = (p[j].value - p[i].value - g)*1.25, p[i].value = p[j].value - g;\r
+      p[i].promoGain = (p[j].value - p[i].value - g)*0.9, p[i].value = p[j].value - g;\r
     else p[i].promoGain = 0;\r
     if(i == kylin[BLACK]) p[i].promoGain = 1.25*KYLIN, p[i].value += KYLIN;\r
-    if(j > 0 && p[i].pst == PST_STEPPER) p[i].pst = PST_BPPROM;  // use black pre-prom bonus\r
-    if(j > 0 && p[i].pst == PST_WJUMPER) p[i].pst = PST_BJUMPER; // use black pre-prom bonus\r
     board[p[i].pos] = i;\r
     rootEval -= p[i].value + PST[p[i].pst + p[i].pos];\r
     promoDelta -= p[i].promoGain;\r
@@ -978,12 +990,16 @@ Init (int var)
     PST[PST_STEPPER+s] = d/4 - (i < 2 || i > BH-3 ? 3 : 0) - (j == 0 || j == BH-1 ? 5 : 0)\r
                     + 3*(i==zone || i==BH-zone-1);          // stepper centralization\r
     PST[PST_WJUMPER+s] = d/6;                               // double-stepper centralization\r
-    PST[PST_SLIDER +s] = d/12 - 5*(i==BH/2 || i==(BH-1)/2); // slider centralization\r
+    PST[PST_SLIDER +s] = d/12 - 15*(i==BH/2 || i==(BH-1)/2);// slider centralization\r
     PST[PST_TRAP  +s] = j < 3 || j > BH-4 ? (i < 3 ? 7 : i == 3 ? 4 : i == 4 ? 2 : 0) : 0;\r
     PST[PST_CENTER+s] = ((BH-1)*(BH-1) - (2*i - BH + 1)*(2*i - BH + 1) - (2*j - BH + 1)*(2*j - BH + 1))/6;\r
     PST[PST_WPPROM+s] = PST[PST_BPPROM+s] = PST[PST_STEPPER+s]; // as stepper, but with pre-promotion bonus W/B\r
     PST[PST_BJUMPER+s] = PST[PST_WJUMPER+s];                // as jumper, but with pre-promotion bonus B\r
     PST[PST_ZONDIST+s] = BW*(zone - 1 - i);                 // board step to enter promo zone black\r
+    PST[PST_ADVANCE+s] = PST[PST_WFLYER-s-1] = 2*(5*i+i*i) - (i >= zone)*6*(i-zone+1)*(i-zone+1)\r
+       - (2*j - BH + 1)*(2*j - BH + 1)/BH + BH/2\r
+       - 50 - 35*(j==0 || j == BH-1) - 15*(j == 1 || BH-2); // advance-encouraging table\r
+    PST[PST_WFLYER +s] = PST[PST_END-s-1] = (i == zone-1)*40 + (i == zone-2)*20 - 20;\r
    }\r
    if(zone > 0) PST[PST_WPPROM+BW*(BH-1-zone) + j] += 10, PST[PST_BPPROM + BW*zone + j] += 10;\r
 #if KYLIN\r
@@ -1674,7 +1690,7 @@ Evaluate (int difEval)
 #endif\r
 \r
 #ifdef PAWNBLOCK\r
-  // penalty for blocking own P or GB: 20 by slider, 10 by other, but 50 if only retreat mode is straight back\r
+  // penalty for blocking own P or GB: 20 by slider, 10 by other, but 50 if only RETRACT mode is straight back\r
   for(i=last[WHITE]; i > 1 && p[i].value<=50; i-=2) {\r
     if((f = p[i].pos) != ABSENT) { // P present,\r
       if((j = board[f + BW])&1) // square before it white (odd) piece\r
@@ -1693,7 +1709,26 @@ Evaluate (int difEval)
   }\r
 #endif\r
 \r
-  return difEval - (filling*filling*promoDelta >> 16) + (stm ? score : -score);\r
+#ifdef TANDEM\r
+    if(zone > 0) {\r
+      int rw = BW*(BH-1-zone), rb = BW*zone, h=0;\r
+      for(f=0; f<BH; f++) {\r
+       if(p[board[rw+f]].pst == PST_ADVANCE) {\r
+         h += (p[board[rw+f-BW]].pst == PST_ADVANCE);\r
+         if(f > 0)    h += (p[board[rw+f-BW-1]].pst == PST_ADVANCE);\r
+         if(f+1 < BH) h += (p[board[rw+f-BW+1]].pst == PST_ADVANCE);\r
+       }\r
+       if(p[board[rb+f]].pst == PST_ADVANCE) {\r
+         h -= (p[board[rb+f+BW]].pst == PST_RETRACT);\r
+         if(f > 0)    h -= (p[board[rb+f+BW-1]].pst == PST_RETRACT);\r
+         if(f+1 < BH) h -= (p[board[rb+f+BW+1]].pst == PST_RETRACT);\r
+       }\r
+      }\r
+      score += h*TANDEM;\r
+    }\r
+#endif\r
+\r
+  return difEval - (filling*promoDelta >> 8) + (stm ? score : -score);\r
 }\r
 \r
 inline void\r
@@ -1991,6 +2026,7 @@ MapFromScratch(attacks); // for as long as incremental update does not work.
       score = -Search(-beta, -iterAlpha, -difEval - tb.booty, iterDep-1+ext,\r
                        curMove >= late && iterDep > QSdepth + LMR,\r
                                                       promoSuppress & ~PROMOTE, defer, depth ? INF : tb.gain);\r
+\r
 #else\r
       score = 0;\r
 #endif\r
@@ -2093,7 +2129,7 @@ pplist()
 {\r
   int i, j;\r
   for(i=0; i<182; i++) {\r
-       printf("%3d. %3d %3d %4d   %02x %d %d %x %3d ", i, p[i].value, p[i].promo, p[i].pos, p[i].promoFlag&255, p[i].mobWeight, p[i].qval, p[i].bulk, p[i].promoGain);\r
+       printf("%3d. %3d %3d %4d   %02x %d %d %x %3d %4d ", i, p[i].value, p[i].promo, p[i].pos, p[i].promoFlag&255, p[i].mobWeight, p[i].qval, p[i].bulk, p[i].promoGain, p[i].pst);\r
        for(j=0; j<8; j++) printf("  %2d", p[i].range[j]);\r
        if(i<2 || i>11) printf("\n"); else printf("  %02x\n", fireFlags[i-2]&255);\r
   }\r