-rw-r--r-- | noncore/games/go/goplayutils.c | 2 | ||||
-rw-r--r-- | noncore/games/go/killable.c | 4 |
2 files changed, 3 insertions, 3 deletions
diff --git a/noncore/games/go/goplayutils.c b/noncore/games/go/goplayutils.c index 9e2ce4c..91c6db2 100644 --- a/noncore/games/go/goplayutils.c +++ b/noncore/games/go/goplayutils.c @@ -663,513 +663,513 @@ short x, y, z; playStack[playMark].kind = add; playStack[playMark].uval.add.who = me; playStack[playMark].uval.add.xl = x; playStack[playMark].uval.add.yl = y; playStack[playMark].gID = myGID; playStack[playMark].uval.add.sNumber = 0; if (isNew) playStack[playMark].uval.add.nextGID = newGID - 1; else playStack[playMark].uval.add.nextGID = newGID; if (showTrees) sstone(me, x, y, 0); /* merge adjacent groups */ if ((x > 0) && (bord[x - 1][y] == me) && (gMap[groupIDs[x - 1][y]] != myGID)) mergeGroup(gMap[groupIDs[x - 1][y]], myGID); if ((x < maxPoint) && (bord[x + 1][y] == me) && (gMap[groupIDs[x + 1][y]] != myGID)) mergeGroup(gMap[groupIDs[x + 1][y]], myGID); if ((y > 0) && (bord[x][y - 1] == me) && (gMap[groupIDs[x][y - 1]] != myGID)) mergeGroup(gMap[groupIDs[x][y - 1]], myGID); if ((y < maxPoint) && (bord[x][y + 1] == me) && (gMap[groupIDs[x][y + 1]] != myGID)) mergeGroup(gMap[groupIDs[x][y + 1]], myGID); /* kill opposing groups, listing affected groups */ nlcGroup.indx = 1; nlcGroup.v[1] = myGID; /* init list to include me */ if ((x > 0) && (bord[x - 1][y] == him) && (gList[gMap[groupIDs[x - 1][y]]].libC == 1)) { killFlag = TRUE; killGroup(x - 1, y, me, him); } if ((x < maxPoint) && (bord[x + 1][y] == him) && (gList[gMap[groupIDs[x + 1][y]]].libC == 1)) { killFlag = TRUE; killGroup(x + 1, y, me, him); } if ((y > 0) && (bord[x][y - 1] == him) && (gList[gMap[groupIDs[x][y - 1]]].libC == 1)) { killFlag = TRUE; killGroup(x, y - 1, me, him); } if ((y < maxPoint) && (bord[x][y + 1] == him) && (gList[gMap[groupIDs[x][y + 1]]].libC == 1)) { killFlag = TRUE; killGroup(x, y + 1, me, him); } /* list groups adjacent to me */ if ((x > 0) && (bord[x - 1][y] == him)) { nlcGroup.indx = nlcGroup.indx + 1; nlcGroup.v[nlcGroup.indx] = gMap[groupIDs[x - 1][y]]; } if ((x < maxPoint) && (bord[x + 1][y] == him)) { nlcGroup.indx = nlcGroup.indx + 1; nlcGroup.v[nlcGroup.indx] = gMap[groupIDs[x + 1][y]]; } if ((y > 0) && (bord[x][y - 1] == him)) { nlcGroup.indx = nlcGroup.indx + 1; nlcGroup.v[nlcGroup.indx] = gMap[groupIDs[x][y - 1]]; } if ((y < maxPoint) && (bord[x][y + 1] == him)) { nlcGroup.indx = nlcGroup.indx + 1; nlcGroup.v[nlcGroup.indx] = gMap[groupIDs[x][y + 1]]; } /* fix liberty count for affected groups */ grpMark = grpMark + 1; for (i = 1; i <= nlcGroup.indx; i++) if (gList[nlcGroup.v[i]].groupMark != grpMark) { if (gList[nlcGroup.v[i]].atLevel != tryLevel) { playMark = playMark + 1; playStack[playMark].kind = chLib; playStack[playMark].gID = nlcGroup.v[i]; playStack[playMark].uval.chLib.oldLevel = gList[nlcGroup.v[i]].atLevel; playStack[playMark].uval.chLib.oldLC = gList[nlcGroup.v[i]].libC; } gList[nlcGroup.v[i]].groupMark = grpMark; gList[nlcGroup.v[i]].atLevel = tryLevel; spanGroup(gList[nlcGroup.v[i]].lx, gList[nlcGroup.v[i]].ly, &pPlist); gList[nlcGroup.v[i]].libC = pPlist.indx; } } /* plei */ saveState() { /* saveState */ playMark = 0; tryLevel = 0; newGID = maxGroupID; } /* saveState */ /* undoes a move sequence back to uMark */ undoTo(uMark) short uMark; { /* undoTo */ short i, xl, yl; #ifdef DEBUG printf( "undoTo\n" ); #endif for (i = playMark; i >= uMark + 1; i--) if (playStack[i].kind == rem) { xl = playStack[i].uval.rem.xl; yl = playStack[i].uval.rem.yl; bord[xl][yl] = playStack[i].uval.rem.who; groupIDs[xl][yl] = playStack[i].gID; if (showTrees) sstone(playStack[i].uval.rem.who, xl, yl, playStack[i].uval.rem.sNumber); } else if (playStack[i].kind == add) { xl = playStack[i].uval.add.xl; yl = playStack[i].uval.add.yl; bord[xl][yl] = 0; groupIDs[xl][yl] = 0; tryLevel = tryLevel - 1; newGID = playStack[i].uval.add.nextGID; if (showTrees) rstone(xl, yl); } else if (playStack[i].kind == reMap) gMap[playStack[i].gID] = playStack[i].uval.reMap.oldGID; else /* change libs of group - gID is pre-mapped */ { gList[playStack[i].gID].libC = playStack[i].uval.chLib.oldLC; gList[playStack[i].gID].atLevel = playStack[i].uval.chLib.oldLevel; } playMark = uMark; } /* undoTo */ /* restores the state of the world after trying a move sequence */ restoreState() { /* restoreState */ #ifdef DEBUG printf( "restoreState\n" ); #endif if (playMark > 0) { undoTo(0); playMark = 0; tryLevel = 0; } } /* restoreState */ /* exception bpt; */ /* returns true if (the group (at gx, gy) is saveable. if so, returns the point to play at in savex, savey */ short saveable(gx, gy, savex, savey) short gx, gy, *savex, *savey; { /* saveable */ short me, him, gx1, gx2, i, j, smark, mark2, tl, result; char sChar; sPointList dList; point tp; short libList[maxSPoint+1]; #ifdef DEBUG printf( "saveable\n" ); #endif dbStop = TRUE; me = bord[gx][gy]; him = -me; if (me == 1) sChar = '|'; else sChar = '>'; /* write(sChar); */ spanGroup(gx, gy, &plist3); /* find my liberties */ if (adjInAtari) /* one of my options is to kill */ { listAdjacents(gx, gy, &aList); for (i = 1; i <= aList.indx; i++) if (gList[aList.v[i]].libC == 1) { spanGroup(gList[aList.v[i]].lx, gList[aList.v[i]].ly, &pList1); /* find it's liberty */ plist3.indx = plist3.indx + 1; plist3.p[plist3.indx].px = pList1.p[1].px; plist3.p[plist3.indx].py = pList1.p[1].py; } } for (i = 1; i <= maxSPoint; i++) libList[i] = -1; if ((utilPlayLevel > 4) && (gList[gMap[groupIDs[gx][gy]]].libC > 1)) /* account for diags */ { listDiags(gx, gy, &dList); j = 0; i = plist3.indx; while ((j < dList.indx) && (i < maxSPoint)) { j = j + 1; i = i + 1; libList[i] = 100; plist3.p[i].px = dList.p[j].px; plist3.p[i].py = dList.p[j].py; } plist3.indx = i; } if (plist3.indx > 1) /* sort by decreasing lib count */ { for (i = 1; i <= plist3.indx; i++) if (libList[i] != 100) { mark2 = playMark; tryPlay(plist3.p[i].px, plist3.p[i].py, me); libList[i] = gList[gMap[groupIDs[gx][gy]]].libC; if (libList[i] > treeLibLim) /* i'm safe */ { *savex = plist3.p[i].px; *savey = plist3.p[i].py; result = TRUE; goto one; } undoTo(mark2); } for (i = 1; i <= plist3.indx - 1; i++) for (j = i + 1; j <= plist3.indx; j++) if (libList[i] < libList[j]) { tl = libList[i]; libList[i] = libList[j]; libList[j] = tl; tp = plist3.p[i]; plist3.p[i] = plist3.p[j]; plist3.p[j] = tp; } } for (i = 1; i <= plist3.indx; i++) { *savex = plist3.p[i].px; *savey = plist3.p[i].py; if (legal[*savex][*savey]) { smark = playMark; tryPlay(*savex, *savey, me); - pause(); +//pause(); if (gList[gMap[groupIDs[*savex][*savey]]].libC > 1) if (gList[gMap[groupIDs[gx][gy]]].libC > treeLibLim) { restoreState(); /* sClearChar(sChar, rXor); */ return TRUE; } else if (gList[gMap[groupIDs[gx][gy]]].libC > 1) if (! killable(gx, gy, &gx1, &gx2)) { restoreState(); /* sClearChar(sChar, rXor); */ return TRUE; } undoTo(smark); } } result = FALSE; one: restoreState(); /* sClearChar(sChar, rXor); */ return result; } /* saveable */ /* marks unsavable groups as dead */ markDead() { /* markDead */ short i, j, gx, gy, result; #ifdef DEBUG printf( "markDead\n" ); #endif for (i = 1; i <= maxGroupID; i++) if (killable(gList[i].lx, gList[i].ly, &gx, &gy)) result = ! saveable(gList[i].lx, gList[i].ly, &gx, &gy); else result = FALSE; for (i = 0; i <= maxPoint; i++) for (j = 0; j <= maxPoint; j++) if (bord[i][j] == 0) ndbord[i][j] = 0; else if (gList[groupIDs[i][j]].isDead) ndbord[i][j] = 0; else ndbord[i][j] = bord[i][j]; } /* markDead */ /* marks groups with two eyes as live */ MLspan(x, y, saw1, sawm1, size, sMark) short x, y, *saw1, *sawm1, *size, sMark; { /* span */ if (ndbord[x][y] == 1) *saw1 = TRUE; else if (ndbord[x][y] == -1) *sawm1 = TRUE; else if (sGroups[x][y] == 0) { sGroups[x][y] = sMark; *size = *size + 1; if (x > 0) MLspan(x - 1, y, saw1, sawm1, size, sMark); if (x < maxPoint) MLspan(x + 1, y, saw1, sawm1, size, sMark); if (y > 0) MLspan(x, y - 1, saw1, sawm1, size, sMark); if (y < maxPoint) MLspan(x, y + 1, saw1, sawm1, size, sMark); } } /* span */ short CLspan(x, y, numEyes, who) short x, y, *numEyes, who; { /* span */ markBoard[x][y] = marker; if (ndbord[x][y] == 0) { if ((sList[sGroups[x][y]].sm != marker) && (sList[sGroups[x][y]].w == who)) { sList[sGroups[x][y]].sm = marker; if (sList[sGroups[x][y]].s > 6) return TRUE; *numEyes = *numEyes + 1; if (*numEyes > 1) return TRUE; } } else if (bord[x][y] == who) { if ((x > 0) && (markBoard[x - 1][y] != marker)) if (CLspan(x - 1, y, numEyes, who)) return TRUE; if ((x < maxPoint) && (markBoard[x + 1][y] != marker)) if (CLspan(x + 1, y, numEyes, who)) return TRUE; if ((y > 0) && (markBoard[x][y - 1] != marker)) if (CLspan(x, y - 1, numEyes, who)) return TRUE; if ((y < maxPoint) && (markBoard[x][y + 1] != marker)) if (CLspan(x, y + 1, numEyes, who)) return TRUE; } return FALSE; } /* span */ short checkLive(x, y) short x, y; { /* checkLive */ short numEyes, who; #ifdef DEBUG printf( "checkLive\n" ); #endif numEyes = 0; who = bord[x][y]; marker = marker + 1; return CLspan(x, y, &numEyes, who); } /* checkLive */ markLive() { /* markLive */ short i, j, size, sMark = 0; short saw1, sawm1; #ifdef DEBUG printf( "markLive\n" ); #endif initArray(sGroups); for (i = 0; i <= maxPoint; i++) for (j = 0; j <= maxPoint; j++) if ((sGroups[i][j] == 0) && (ndbord[i][j] == 0)) { size = 0; sMark = sMark + 1; sawm1 = FALSE; saw1 = FALSE; MLspan(i, j, &saw1, &sawm1, &size, sMark); sList[sMark].s = size; sList[sMark].sm = 0; if (sawm1) if (saw1) sList[sMark].w = 0; else sList[sMark].w = -1; else if (saw1) sList[sMark].w = 1; else sList[sMark].w = 0; } for (i = 1; i <= maxGroupID; i++) if (! gList[i].isDead) gList[i].isLive = checkLive(gList[i].lx, gList[i].ly); } /* markLive */ /* generates the connection map and the protected point map. */ genConnects() { /* genConnects */ short x, y, numStones; #ifdef DEBUG printf( "genConnects\n" ); #endif for (x = 0; x <= maxPoint; x++) for (y = 0; y <= maxPoint; y++) { connectMap[x][y] = 0; protPoints[x][y] = 0; } for (x = 0; x <= maxPoint; x++) for (y = 0; y <= maxPoint; y++) if (bord[x][y] == 1) /* map connections to this stone */ { if (x > 0) /* direct connection */ connectMap[x - 1][y] += 1; if (x < maxPoint) connectMap[x + 1][y] += 1; if (y > 0) connectMap[x][y - 1] += 1; if (y < maxPoint) connectMap[x][y + 1] += 1; if ((x > 0) && (y > 0) && /* diagonal connection */ (bord[x - 1][y] == 0) && (bord[x][y - 1] == 0)) connectMap[x - 1][y - 1] += 1; if ((x < maxPoint) && (y > 0) && (bord[x + 1][y] == 0) && (bord[x][y - 1] == 0)) connectMap[x + 1][y - 1] += 1; if ((x < maxPoint) && (y < maxPoint) && (bord[x + 1][y] == 0) && (bord[x][y + 1] == 0)) connectMap[x + 1][y + 1] += 1; if ((x > 0) && (y < maxPoint) && (bord[x - 1][y] == 0) && (bord[x][y + 1] == 0)) connectMap[x - 1][y + 1] += 1; if ((x > 1) && (claim[x - 1][y] > 3)) /* one point jump */ connectMap[x - 2][y] += 1; if ((x < (maxPoint - 1)) && (claim[x + 1][y] > 3)) connectMap[x + 2][y] += 1; if ((y > 1) && (claim[x][y - 1] > 3)) connectMap[x][y - 2] += 1; if ((y < (maxPoint - 1)) && (claim[x][y + 1] > 3)) connectMap[x][y + 2] += 1; if ((x > 1) && (y > 0) && /* knight's move */ (claim[x - 1][y] > 3) && (claim[x - 1][y - 1] > 3)) connectMap[x - 2][y - 1] += 1; if ((x > 0) && (y > 1) && (claim[x][y - 1] > 3) && (claim[x - 1][y - 1] > 3)) connectMap[x - 1][y - 2] += 1; if ((x < (maxPoint - 1)) && (y > 0) && (claim[x + 1][y] > 3) && (claim[x + 1][y - 1] > 3)) connectMap[x + 2][y - 1] += 1; if ((x < maxPoint) && (y > 1) && (claim[x][y - 1] > 3) && (claim[x + 1][y - 1] > 3)) connectMap[x + 1][y - 2] += 1; if ((x > 1) && (y < maxPoint) && (claim[x - 1][y] > 3) && (claim[x - 1][y + 1] > 3)) connectMap[x - 2][y + 1] += 1; if ((x > 0) && (y < (maxPoint - 1)) && (claim[x][y + 1] > 3) && (claim[x - 1][y + 1] > 3)) connectMap[x - 1][y + 2] += 1; if ((x < (maxPoint - 1)) && (y < maxPoint) && (claim[x + 1][y] > 3) && (claim[x + 1][y + 1] > 3)) connectMap[x + 2][y + 1] += 1; if ((x < maxPoint) && (y < (maxPoint - 1)) && (claim[x][y + 1] > 3) && (claim[x + 1][y + 1] > 3)) connectMap[x + 1][y + 2] += 1; } else if (bord[x][y] == 0) /* see if protected point */ { numStones = 0; if (x == 0) numStones = numStones + 1; if (y == 0) numStones = numStones + 1; if (x == maxPoint) numStones = numStones + 1; if (y == maxPoint) numStones = numStones + 1; if ((x > 0) && (bord[x - 1][y] == 1)) numStones = numStones + 1; if ((y > 0) && (bord[x][y - 1] == 1)) numStones = numStones + 1; if ((x < maxPoint) && (bord[x + 1][y] == 1)) numStones = numStones + 1; if ((y < maxPoint) && (bord[x][y + 1] == 1)) numStones = numStones + 1; if (numStones == 4) protPoints[x][y] = 1; else if (numStones == 3) { if ((x > 0) && ((bord[x - 1][y] == 0) || ((bord[x - 1][y] == -1) && (gList[groupIDs[x - 1][y]].libC == 1)))) protPoints[x][y] = 1; diff --git a/noncore/games/go/killable.c b/noncore/games/go/killable.c index 3ed2d2e..23a133d 100644 --- a/noncore/games/go/killable.c +++ b/noncore/games/go/killable.c @@ -1,373 +1,373 @@ /* By Stoney Ballard */ /* Ported from Pascal to C by Todd R. Johnson */ #include "go.h" #include "goplayutils.h" #include "amigo.h" extern intBoard bord, groupIDs; extern boolBoard legal; extern groupRec gList[maxGroup]; extern short gMap[maxGroup], adjInAtari, adj2Libs, playMark, treeLibLim, utilPlayLevel, killFlag, depthLimit, dbStop, showTrees; extern pointList plist2; /* returns true if the group (at x, y) is killable. if so, returns the point to play at in killx, killy. */ short me, him, depth, i, j, tryCount, tl, topMark, tkMark, mark2; char sChar; sPointList lList, dList; point tp; short libList[maxSPoint+1]; short esc; short mtNbrs(x, y) short x, y; { /* mtNbrs */ short n = 0; if ((x > 0) && (bord[x - 1][y] == 0)) n = n + 1; if ((x < maxPoint) && (bord[x + 1][y] == 0)) n = n + 1; if ((y > 0) && (bord[x][y - 1] == 0)) n = n + 1; if ((y < maxPoint) && (bord[x][y + 1] == 0)) n = n + 1; return n; } /* mtNbrs */ short killTree(tx, ty, gx, gy, escape, tkMark) short tx, ty, gx, gy, *escape, tkMark; { /* killTree */ short curMark, mark2, mark3, i, j, k, tl, dStart, result; sPointList lList1, lList2; short libList[maxSPoint+1]; point tp; short esc = FALSE; tryCount = tryCount + 1; if (tryCount > tryLimit) { undoTo(tkMark); /* for (i = 1; i <= depth - 1; i++) { sClearChar(sChar, rXor); } */ depth = 1; return FALSE; } /* write(sChar); */ depth = depth + 1; curMark = playMark; tryPlay(tx, ty, me); /* try my move */ - pause(); +// pause(); if (gList[gMap[groupIDs[tx][ty]]].libC == 0) /* I'm dead */ { result = FALSE; goto one; } else if (killFlag) /* I killed something of his */ { result = TRUE; goto one; } else if (gList[gMap[groupIDs[gx][gy]]].libC > treeLibLim) /* safe */ { result = FALSE; goto one; } else { sSpanGroup(gx, gy, &lList1); /* find his liberties */ if (gList[gMap[groupIDs[tx][ty]]].libC == 1) /* he can kill me */ { if (lList1.indx < maxSPoint) /* add that option to his list */ { lList1.indx = lList1.indx + 1; spanGroup(tx, ty, &plist2); /* find my liberty */ lList1.p[lList1.indx].px = plist2.p[1].px; lList1.p[lList1.indx].py = plist2.p[1].py; } else { result = FALSE; goto one; } } for (i = 1; i <= maxSPoint; i++) /* init liblist so diags can be marked */ libList[i] = -1; if ((utilPlayLevel > 4) && (lList1.indx > 1) && (gList[gMap[groupIDs[gx][gy]]].libC > 1)) /* try diags */ { listDiags(gx, gy, &dList); j = 0; i = lList1.indx; while ((j < dList.indx) && (i < maxSPoint)) { j = j + 1; i = i + 1; libList[i] = 0; /* mark this as a diag */ lList1.p[i].px = dList.p[j].px; lList1.p[i].py = dList.p[j].py; } lList1.indx = i; } if (lList1.indx > 1) /* sort by decreasing lib count */ { for (i = 1; i <= lList1.indx; i++) if (libList[i] != 0) /* diags are tried last */ { mark2 = playMark; tryPlay(lList1.p[i].px, lList1.p[i].py, him); libList[i] = gList[gMap[groupIDs[gx][gy]]].libC; if ((libList[i] > treeLibLim) || ((libList[i] > (depthLimit - depth)) && (libList[i] > 2))) { *escape = TRUE; result = FALSE; goto one; } undoTo(mark2); } for (i = 1; i <= lList1.indx - 1; i++) for (j = i + 1; j <= lList1.indx; j++) if (libList[i] < libList[j]) { tl = libList[i]; libList[i] = libList[j]; libList[j] = tl; tp = lList1.p[i]; lList1.p[i] = lList1.p[j]; lList1.p[j] = tp; } } for (i = 1; i <= lList1.indx + 1; i++) /* try his responses */ { mark2 = playMark; if (i <= lList1.indx) /* try his move */ { tryPlay(lList1.p[i].px, lList1.p[i].py, him); /* play his response */ - pause(); +// pause(); if (gList[gMap[groupIDs[lList1.p[i].px] [lList1.p[i].py]]].libC < 2) goto two; /* a bogus move */ } else if (gList[gMap[groupIDs[gx][gy]]].libC <= 1) { result = TRUE; goto one; } if (gList[gMap[groupIDs[gx][gy]]].libC > treeLibLim) { *escape = TRUE; result = FALSE; goto one; } if (gList[gMap[groupIDs[gx][gy]]].libC > 1) { /* look at my responses */ sSpanGroup(gx, gy, &lList2); /* list his liberties */ dStart = lList2.indx + 1; if (adjInAtari) /* he wins */ { result = FALSE; goto one; } if ((lList2.indx > 2) && adj2Libs) /* he wins */ { result = FALSE; goto one; } for (k = 1; k <= maxSPoint; k++) libList[k] = -1; if (utilPlayLevel > 4) /* account for diagonal moves */ { listDiags(gx, gy, &dList); j = 0; k = lList2.indx; while ((j < dList.indx) && (k < maxSPoint)) { j = j + 1; k = k + 1; libList[k] = 100; lList2.p[k].px = dList.p[j].px; lList2.p[k].py = dList.p[j].py; } lList2.indx = k; } if (lList2.indx > 1) /* sort by increasing lib count */ { for (k = 1; k <= lList2.indx; k++) if (libList[k] != 100) /* diags go last */ { mark3 = playMark; tryPlay(lList2.p[k].px, lList2.p[k].py, me); libList[k] = gList[gMap[groupIDs[gx][gy]]].libC; undoTo(mark3); } for (k = 1; k <= lList2.indx - 1; k++) for (j = k + 1; j <= lList2.indx; j++) if (libList[k] > libList[j]) { tl = libList[k]; libList[k] = libList[j]; libList[j] = tl; tp = lList2.p[k]; lList2.p[k] = lList2.p[j]; lList2.p[j] = tp; } else if ((libList[k] == libList[j]) && (libList[k] == 1)) if (mtNbrs(lList2.p[k].px, lList2.p[k].py) < mtNbrs(lList2.p[j].px, lList2.p[j].py)) { tl = libList[k]; libList[k] = libList[j]; libList[j] = tl; tp = lList2.p[k]; lList2.p[k] = lList2.p[j]; lList2.p[j] = tp; } } for (j = 1; j <= lList2.indx; j++) { if (killTree(lList2.p[j].px, lList2.p[j].py, gx, gy, &esc, tkMark)) goto two; /* this kills him */ if (esc && (j >= dStart)) { result = FALSE; goto one; /* don't bother with more diags if escapes */ } } result = FALSE; /* none of my responses kills him */ goto one; } two: undoTo(mark2); } result = TRUE; /* none of his responses saves him */ } one: undoTo(curMark); /* sClearChar(sChar, rXor); */ depth = depth - 1; return result; } /* killTree */ short tKillTree(tx, ty, gx, gy) short tx, ty, gx, gy; { /* tKillTree */ short tkMark, escape; tryCount = 0; tkMark = playMark; return killTree(tx, ty, gx, gy, &escape, tkMark); } /* tKillTree */ short killable(gx, gy, killx, killy) short gx, gy, *killx, *killy; { /* killable */ #ifdef DEBUG printf( "killable\n" ); showTrees = TRUE; #endif dbStop = TRUE; him = bord[gx][gy]; /* find out who I am */ me = -him; /* if (me == 1) sChar = '>'; else sChar = '|'; */ /* write(sChar); */ depth = 1; topMark = playMark; sSpanGroup(gx, gy, &lList); /* find his liberties */ if (lList.indx == 1) { *killx = lList.p[1].px; *killy = lList.p[1].py; return TRUE; } else if (lList.indx > treeLibLim) return FALSE; else if (adjInAtari) return FALSE; else if ((lList.indx > 2) && adj2Libs) return FALSE; else { for (i = 1; i <= maxSPoint; i++) libList[i] = -1; if (utilPlayLevel > 4) /* account for diagonal moves */ { listDiags(gx, gy, &dList); j = 0; i = lList.indx; while ((j < dList.indx) && (i < maxSPoint)) { j = j + 1; i = i + 1; libList[i] = 100; lList.p[i].px = dList.p[j].px; lList.p[i].py = dList.p[j].py; } lList.indx = i; } if (lList.indx > 1) /* sort by increasing lib count */ { for (i = 1; i <= lList.indx; i++) if (libList[i] != 100) /* diags go last */ { mark2 = playMark; tryPlay(lList.p[i].px, lList.p[i].py, me); libList[i] = gList[gMap[groupIDs[gx][gy]]].libC; undoTo(mark2); } for (i = 1; i <= lList.indx - 1; i++) for (j = i + 1; j <= lList.indx; j++) if (libList[i] > libList[j]) { tl = libList[i]; libList[i] = libList[j]; libList[j] = tl; tp = lList.p[i]; lList.p[i] = lList.p[j]; lList.p[j] = tp; } else if ((libList[i] == libList[j]) && (libList[i] == 1)) if (mtNbrs(lList.p[i].px, lList.p[i].py) < mtNbrs(lList.p[j].px, lList.p[j].py)) { tl = libList[i]; libList[i] = libList[j]; libList[j] = tl; tp = lList.p[i]; lList.p[i] = lList.p[j]; lList.p[j] = tp; } } for (i = 1; i <= lList.indx; i++) { if (legal[lList.p[i].px][lList.p[i].py]) { *killx = lList.p[i].px; *killy = lList.p[i].py; if (tKillTree(*killx, *killy, gx, gy)) { /* sClearChar(sChar, rXor); */ return TRUE; } } } return FALSE; } /* sClearChar(sChar, rXor); */ } /* killable */ |