]>
git.cameronkatri.com Git - bsdgames-darwin.git/blob - gomoku/pickmove.c
1 /* $NetBSD: pickmove.c,v 1.9 1999/09/18 19:38:51 jsm Exp $ */
5 * The Regents of the University of California. All rights reserved.
7 * This code is derived from software contributed to Berkeley by
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 * 3. All advertising materials mentioning features or use of this software
19 * must display the following acknowledgement:
20 * This product includes software developed by the University of
21 * California, Berkeley and its contributors.
22 * 4. Neither the name of the University nor the names of its contributors
23 * may be used to endorse or promote products derived from this software
24 * without specific prior written permission.
26 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
27 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
28 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
29 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
30 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
31 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
32 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
33 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
34 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
35 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
39 #include <sys/cdefs.h>
42 static char sccsid
[] = "@(#)pickmove.c 8.2 (Berkeley) 5/3/95";
44 __RCSID("$NetBSD: pickmove.c,v 1.9 1999/09/18 19:38:51 jsm Exp $");
51 #include <machine/limits.h>
55 #define BITS_PER_INT (sizeof(int) * CHAR_BIT)
56 #define MAPSZ (BAREA / BITS_PER_INT)
58 #define BIT_SET(a, b) ((a)[(b)/BITS_PER_INT] |= (1 << ((b) % BITS_PER_INT)))
59 #define BIT_CLR(a, b) ((a)[(b)/BITS_PER_INT] &= ~(1 << ((b) % BITS_PER_INT)))
60 #define BIT_TEST(a, b) ((a)[(b)/BITS_PER_INT] & (1 << ((b) % BITS_PER_INT)))
62 struct combostr
*hashcombos
[FAREA
]; /* hash list for finding duplicates */
63 struct combostr
*sortcombos
; /* combos at higher levels */
64 int combolen
; /* number of combos in sortcombos */
65 int nextcolor
; /* color of next move */
66 int elistcnt
; /* count of struct elist allocated */
67 int combocnt
; /* count of struct combostr allocated */
68 int forcemap
[MAPSZ
]; /* map for blocking <1,x> combos */
69 int tmpmap
[MAPSZ
]; /* map for blocking <1,x> combos */
70 int nforce
; /* count of opponent <1,x> combos */
76 struct spotstr
*sp
, *sp1
, *sp2
;
77 union comboval
*Ocp
, *Tcp
;
80 /* first move is easy */
84 /* initialize all the board values */
85 for (sp
= &board
[PT(T
,20)]; --sp
>= &board
[PT(A
,1)]; ) {
86 sp
->s_combo
[BLACK
].s
= MAXCOMBO
+ 1;
87 sp
->s_combo
[WHITE
].s
= MAXCOMBO
+ 1;
88 sp
->s_level
[BLACK
] = 255;
89 sp
->s_level
[WHITE
] = 255;
90 sp
->s_nforce
[BLACK
] = 0;
91 sp
->s_nforce
[WHITE
] = 0;
92 sp
->s_flg
&= ~(FFLAGALL
| MFLAGALL
);
95 memset(forcemap
, 0, sizeof(forcemap
));
97 /* compute new values */
102 /* find the spot with the highest value */
103 for (sp
= sp1
= sp2
= &board
[PT(T
,19)]; --sp
>= &board
[PT(A
,1)]; ) {
104 if (sp
->s_occ
!= EMPTY
)
106 if (debug
&& (sp
->s_combo
[BLACK
].c
.a
== 1 ||
107 sp
->s_combo
[WHITE
].c
.a
== 1)) {
108 sprintf(fmtbuf
, "- %s %x/%d %d %x/%d %d %d", stoc(sp
- board
),
109 sp
->s_combo
[BLACK
].s
, sp
->s_level
[BLACK
],
111 sp
->s_combo
[WHITE
].s
, sp
->s_level
[WHITE
],
116 /* pick the best black move */
117 if (better(sp
, sp1
, BLACK
))
119 /* pick the best white move */
120 if (better(sp
, sp2
, WHITE
))
125 sprintf(fmtbuf
, "B %s %x/%d %d %x/%d %d %d",
127 sp1
->s_combo
[BLACK
].s
, sp1
->s_level
[BLACK
],
128 sp1
->s_nforce
[BLACK
],
129 sp1
->s_combo
[WHITE
].s
, sp1
->s_level
[WHITE
],
130 sp1
->s_nforce
[WHITE
], sp1
->s_wval
);
132 sprintf(fmtbuf
, "W %s %x/%d %d %x/%d %d %d",
134 sp2
->s_combo
[WHITE
].s
, sp2
->s_level
[WHITE
],
135 sp2
->s_nforce
[WHITE
],
136 sp2
->s_combo
[BLACK
].s
, sp2
->s_level
[BLACK
],
137 sp2
->s_nforce
[BLACK
], sp2
->s_wval
);
140 * Check for more than one force that can't
141 * all be blocked with one move.
143 sp
= (us
== BLACK
) ? sp2
: sp1
;
145 if (sp
->s_combo
[!us
].c
.a
== 1 && !BIT_TEST(forcemap
, m
))
146 dlog("*** Can't be blocked");
149 Ocp
= &sp1
->s_combo
[BLACK
];
150 Tcp
= &sp2
->s_combo
[WHITE
];
152 Tcp
= &sp1
->s_combo
[BLACK
];
153 Ocp
= &sp2
->s_combo
[WHITE
];
159 * Block their combo only if we have to (i.e., if they are one move
160 * away from completing a force and we don't have a force that
161 * we can complete which takes fewer moves to win).
163 if (Tcp
->c
.a
<= 1 && (Ocp
->c
.a
> 1 ||
164 Tcp
->c
.a
+ Tcp
->c
.b
< Ocp
->c
.a
+ Ocp
->c
.b
))
165 return (sp2
- board
);
166 return (sp1
- board
);
170 * Return true if spot 'sp' is better than spot 'sp1' for color 'us'.
174 const struct spotstr
*sp
;
175 const struct spotstr
*sp1
;
180 if (sp
->s_combo
[us
].s
< sp1
->s_combo
[us
].s
)
182 if (sp
->s_combo
[us
].s
!= sp1
->s_combo
[us
].s
)
184 if (sp
->s_level
[us
] < sp1
->s_level
[us
])
186 if (sp
->s_level
[us
] != sp1
->s_level
[us
])
188 if (sp
->s_nforce
[us
] > sp1
->s_nforce
[us
])
190 if (sp
->s_nforce
[us
] != sp1
->s_nforce
[us
])
196 if (BIT_TEST(forcemap
, s
) && !BIT_TEST(forcemap
, s1
))
198 if (!BIT_TEST(forcemap
, s
) && BIT_TEST(forcemap
, s1
))
200 if (sp
->s_combo
[them
].s
< sp1
->s_combo
[them
].s
)
202 if (sp
->s_combo
[them
].s
!= sp1
->s_combo
[them
].s
)
204 if (sp
->s_level
[them
] < sp1
->s_level
[them
])
206 if (sp
->s_level
[them
] != sp1
->s_level
[them
])
208 if (sp
->s_nforce
[them
] > sp1
->s_nforce
[them
])
210 if (sp
->s_nforce
[them
] != sp1
->s_nforce
[them
])
213 if (sp
->s_wval
> sp1
->s_wval
)
215 if (sp
->s_wval
!= sp1
->s_wval
)
221 return (random() & 1);
225 int curcolor
; /* implicit parameter to makecombo() */
226 int curlevel
; /* implicit parameter to makecombo() */
229 * Scan the sorted list of non-empty frames and
230 * update the minimum combo values for each empty spot.
231 * Also, try to combine frames to find more complex (chained) moves.
237 struct combostr
*cbp
, *ecbp
;
240 struct elist
*ep
, *nep
;
246 /* check for empty list of frames */
247 cbp
= sortframes
[color
];
248 if (cbp
== (struct combostr
*)0)
251 /* quick check for four in a row */
252 sp
= &board
[cbp
->c_vertex
];
253 cb
.s
= sp
->s_fval
[color
][d
= cbp
->c_dir
].s
;
256 for (i
= 5 + cb
.c
.b
; --i
>= 0; sp
+= d
) {
257 if (sp
->s_occ
!= EMPTY
)
259 sp
->s_combo
[color
].s
= cb
.s
;
260 sp
->s_level
[color
] = 1;
266 * Update the minimum combo value for each spot in the frame
267 * and try making all combinations of two frames intersecting at
273 sp
= &board
[cbp
->c_vertex
];
274 cp
= &sp
->s_fval
[color
][r
= cbp
->c_dir
];
278 * Since this is the first spot of an open ended
279 * frame, we treat it as a closed frame.
281 cb
.c
.a
= cp
->c
.a
+ 1;
283 if (cb
.s
< sp
->s_combo
[color
].s
) {
284 sp
->s_combo
[color
].s
= cb
.s
;
285 sp
->s_level
[color
] = 1;
288 * Try combining other frames that intersect
291 makecombo2(cbp
, sp
, 0, cb
.s
);
294 else if (color
!= nextcolor
)
295 memset(tmpmap
, 0, sizeof(tmpmap
));
302 for (; i
< 5; i
++, sp
+= d
) { /* for each spot */
303 if (sp
->s_occ
!= EMPTY
)
305 if (cp
->s
< sp
->s_combo
[color
].s
) {
306 sp
->s_combo
[color
].s
= cp
->s
;
307 sp
->s_level
[color
] = 1;
309 if (cp
->s
== 0x101) {
310 sp
->s_nforce
[color
]++;
311 if (color
!= nextcolor
) {
317 * Try combining other frames that intersect
320 makecombo2(cbp
, sp
, i
, cb
.s
);
322 if (cp
->s
== 0x101 && color
!= nextcolor
) {
324 memcpy(forcemap
, tmpmap
, sizeof(tmpmap
));
326 for (i
= 0; (unsigned int)i
< MAPSZ
; i
++)
327 forcemap
[i
] &= tmpmap
[i
];
330 /* mark frame as having been processed */
331 board
[cbp
->c_vertex
].s_flg
|= MFLAG
<< r
;
332 } while ((cbp
= cbp
->c_next
) != ecbp
);
335 * Try to make new 3rd level combos, 4th level, etc.
336 * Limit the search depth early in the game.
339 while (d
<= ((movenum
+ 1) >> 1) && combolen
> n
) {
341 sprintf(fmtbuf
, "%cL%d %d %d %d", "BW"[color
],
342 d
, combolen
- n
, combocnt
, elistcnt
);
351 /* scan for combos at empty spots */
352 for (sp
= &board
[PT(T
,20)]; --sp
>= &board
[PT(A
,1)]; ) {
353 for (ep
= sp
->s_empty
; ep
; ep
= nep
) {
355 if (cbp
->c_combo
.s
<= sp
->s_combo
[color
].s
) {
356 if (cbp
->c_combo
.s
!= sp
->s_combo
[color
].s
) {
357 sp
->s_combo
[color
].s
= cbp
->c_combo
.s
;
358 sp
->s_level
[color
] = cbp
->c_nframes
;
359 } else if (cbp
->c_nframes
< sp
->s_level
[color
])
360 sp
->s_level
[color
] = cbp
->c_nframes
;
366 sp
->s_empty
= (struct elist
*)0;
367 for (ep
= sp
->s_nempty
; ep
; ep
= nep
) {
369 if (cbp
->c_combo
.s
<= sp
->s_combo
[color
].s
) {
370 if (cbp
->c_combo
.s
!= sp
->s_combo
[color
].s
) {
371 sp
->s_combo
[color
].s
= cbp
->c_combo
.s
;
372 sp
->s_level
[color
] = cbp
->c_nframes
;
373 } else if (cbp
->c_nframes
< sp
->s_level
[color
])
374 sp
->s_level
[color
] = cbp
->c_nframes
;
380 sp
->s_nempty
= (struct elist
*)0;
383 /* remove old combos */
384 if ((cbp
= sortcombos
) != (struct combostr
*)0) {
385 struct combostr
*ncbp
;
393 } while ((cbp
= ncbp
) != ecbp
);
394 sortcombos
= (struct combostr
*)0;
400 sprintf(fmtbuf
, "scanframes: %c combocnt %d", "BW"[color
],
406 sprintf(fmtbuf
, "scanframes: %c elistcnt %d", "BW"[color
],
415 * Compute all level 2 combos of frames intersecting spot 'osp'
416 * within the frame 'ocbp' and combo value 's'.
419 makecombo2(ocbp
, osp
, off
, s
)
420 struct combostr
*ocbp
;
426 struct combostr
*ncbp
;
428 int baseB
, fcnt
, emask
, bmask
, n
;
429 union comboval ocb
, fcb
;
430 struct combostr
**scbpp
, *fcbp
;
432 /* try to combine a new frame with those found so far */
434 baseB
= ocb
.c
.a
+ ocb
.c
.b
- 1;
436 emask
= fcnt
? ((ocb
.c
.b
? 0x1E : 0x1F) & ~(1 << off
)) : 0;
437 for (r
= 4; --r
>= 0; ) { /* for each direction */
438 /* don't include frames that overlap in the same direction */
439 if (r
== ocbp
->c_dir
)
443 * Frame A combined with B is the same value as B combined with A
444 * so skip frames that have already been processed (MFLAG).
445 * Also skip blocked frames (BFLAG) and frames that are <1,x>
446 * since combining another frame with it isn't valid.
448 bmask
= (BFLAG
| FFLAG
| MFLAG
) << r
;
450 for (f
= 0; f
< 5; f
++, fsp
-= d
) { /* for each frame */
451 if (fsp
->s_occ
== BORDER
)
453 if (fsp
->s_flg
& bmask
)
456 /* don't include frames of the wrong color */
457 fcb
.s
= fsp
->s_fval
[curcolor
][r
].s
;
462 * Get the combo value for this frame.
463 * If this is the end point of the frame,
464 * use the closed ended value for the frame.
466 if ((f
== 0 && fcb
.c
.b
) || fcb
.s
== 0x101) {
471 /* compute combo value */
472 c
= fcb
.c
.a
+ ocb
.c
.a
- 3;
475 n
= fcb
.c
.a
+ fcb
.c
.b
- 1;
479 /* make a new combo! */
480 ncbp
= (struct combostr
*)malloc(sizeof(struct combostr
) +
481 2 * sizeof(struct combostr
*));
483 panic("Out of memory!");
484 scbpp
= (struct combostr
**)(ncbp
+ 1);
485 fcbp
= fsp
->s_frame
[r
];
493 ncbp
->c_combo
.c
.a
= c
;
494 ncbp
->c_combo
.c
.b
= n
;
495 ncbp
->c_link
[0] = ocbp
;
496 ncbp
->c_link
[1] = fcbp
;
497 ncbp
->c_linkv
[0].s
= ocb
.s
;
498 ncbp
->c_linkv
[1].s
= fcb
.s
;
499 ncbp
->c_voff
[0] = off
;
501 ncbp
->c_vertex
= osp
- board
;
504 ncbp
->c_frameindex
= 0;
505 ncbp
->c_flg
= (ocb
.c
.b
) ? C_OPEN_0
: 0;
507 ncbp
->c_flg
|= C_OPEN_1
;
508 ncbp
->c_framecnt
[0] = fcnt
;
509 ncbp
->c_emask
[0] = emask
;
510 ncbp
->c_framecnt
[1] = fcb
.c
.a
- 2;
511 ncbp
->c_emask
[1] = ncbp
->c_framecnt
[1] ?
512 ((fcb
.c
.b
? 0x1E : 0x1F) & ~(1 << f
)) : 0;
515 if ((c
== 1 && debug
> 1) || debug
> 3) {
516 sprintf(fmtbuf
, "%c c %d %d m %x %x o %d %d",
518 ncbp
->c_framecnt
[0], ncbp
->c_framecnt
[1],
519 ncbp
->c_emask
[0], ncbp
->c_emask
[1],
520 ncbp
->c_voff
[0], ncbp
->c_voff
[1]);
522 printcombo(ncbp
, fmtbuf
);
526 /* record the empty spots that will complete this combo */
529 /* add the new combo to the end of the list */
530 appendcombo(ncbp
, curcolor
);
532 updatecombo(ncbp
, curcolor
);
537 if (c
== 1 && debug
> 1 || debug
> 5) {
549 * Scan the sorted list of frames and try to add a frame to
550 * combinations of 'level' number of frames.
556 struct combostr
*cbp
, *ecbp
;
557 struct spotstr
*sp
, *fsp
;
558 struct elist
*ep
, *nep
;
560 struct combostr
**cbpp
, *pcbp
;
561 union comboval fcb
, cb
;
565 /* scan for combos at empty spots */
567 for (sp
= &board
[PT(T
,20)]; --sp
>= &board
[PT(A
,1)]; ) {
568 for (ep
= sp
->s_empty
; ep
; ep
= nep
) {
570 if (cbp
->c_combo
.s
<= sp
->s_combo
[i
].s
) {
571 if (cbp
->c_combo
.s
!= sp
->s_combo
[i
].s
) {
572 sp
->s_combo
[i
].s
= cbp
->c_combo
.s
;
573 sp
->s_level
[i
] = cbp
->c_nframes
;
574 } else if (cbp
->c_nframes
< sp
->s_level
[i
])
575 sp
->s_level
[i
] = cbp
->c_nframes
;
581 sp
->s_empty
= sp
->s_nempty
;
582 sp
->s_nempty
= (struct elist
*)0;
585 /* try to add frames to the uncompleted combos at level curlevel */
586 cbp
= ecbp
= sortframes
[curcolor
];
588 fsp
= &board
[cbp
->c_vertex
];
590 /* skip frames that are part of a <1,x> combo */
591 if (fsp
->s_flg
& (FFLAG
<< r
))
595 * Don't include <1,x> combo frames,
596 * treat it as a closed three in a row instead.
598 fcb
.s
= fsp
->s_fval
[curcolor
][r
].s
;
603 * If this is an open ended frame, use
604 * the combo value with the end closed.
606 if (fsp
->s_occ
== EMPTY
) {
608 cb
.c
.a
= fcb
.c
.a
+ 1;
612 makecombo(cbp
, fsp
, 0, cb
.s
);
616 * The next four spots are handled the same for both
617 * open and closed ended frames.
621 for (i
= 1; i
< 5; i
++, sp
+= d
) {
622 if (sp
->s_occ
!= EMPTY
)
624 makecombo(cbp
, sp
, i
, fcb
.s
);
626 } while ((cbp
= cbp
->c_next
) != ecbp
);
628 /* put all the combos in the hash list on the sorted list */
629 cbpp
= &hashcombos
[FAREA
];
632 if (cbp
== (struct combostr
*)0)
634 *cbpp
= (struct combostr
*)0;
636 if (ecbp
== (struct combostr
*)0)
639 /* append to sort list */
642 ecbp
->c_prev
= cbp
->c_prev
;
643 cbp
->c_prev
->c_next
= ecbp
;
646 } while (cbpp
!= hashcombos
);
650 * Compute all level N combos of frames intersecting spot 'osp'
651 * within the frame 'ocbp' and combo value 's'.
654 makecombo(ocbp
, osp
, off
, s
)
655 struct combostr
*ocbp
;
660 struct combostr
*cbp
, *ncbp
;
665 struct combostr
**scbpp
;
666 int baseB
, fcnt
, emask
, verts
;
668 struct ovlp_info vertices
[1];
671 baseB
= ocb
.c
.a
+ ocb
.c
.b
- 1;
673 emask
= fcnt
? ((ocb
.c
.b
? 0x1E : 0x1F) & ~(1 << off
)) : 0;
674 for (ep
= osp
->s_empty
; ep
; ep
= ep
->e_next
) {
675 /* check for various kinds of overlap */
677 verts
= checkframes(cbp
, ocbp
, osp
, s
, vertices
);
681 /* check to see if this frame forms a valid loop */
683 sp
= &board
[vertices
[0].o_intersect
];
685 if (sp
->s_occ
!= EMPTY
) {
686 sprintf(fmtbuf
, "loop: %c %s", "BW"[curcolor
],
693 * It is a valid loop if the intersection spot
694 * of the frame we are trying to attach is one
695 * of the completion spots of the combostr
696 * we are trying to attach the frame to.
698 for (nep
= sp
->s_empty
; nep
; nep
= nep
->e_next
) {
699 if (nep
->e_combo
== cbp
)
701 if (nep
->e_combo
->c_nframes
< cbp
->c_nframes
)
704 /* frame overlaps but not at a valid spot */
710 /* compute the first half of the combo value */
711 c
= cbp
->c_combo
.c
.a
+ ocb
.c
.a
- verts
- 3;
715 /* compute the second half of the combo value */
716 n
= ep
->e_fval
.c
.a
+ ep
->e_fval
.c
.b
- 1;
720 /* make a new combo! */
721 ncbp
= (struct combostr
*)malloc(sizeof(struct combostr
) +
722 (cbp
->c_nframes
+ 1) * sizeof(struct combostr
*));
724 panic("Out of memory!");
725 scbpp
= (struct combostr
**)(ncbp
+ 1);
726 if (sortcombo(scbpp
, (struct combostr
**)(cbp
+ 1), ocbp
)) {
732 ncbp
->c_combo
.c
.a
= c
;
733 ncbp
->c_combo
.c
.b
= n
;
734 ncbp
->c_link
[0] = cbp
;
735 ncbp
->c_link
[1] = ocbp
;
736 ncbp
->c_linkv
[1].s
= ocb
.s
;
737 ncbp
->c_voff
[1] = off
;
738 ncbp
->c_vertex
= osp
- board
;
739 ncbp
->c_nframes
= cbp
->c_nframes
+ 1;
740 ncbp
->c_flg
= ocb
.c
.b
? C_OPEN_1
: 0;
741 ncbp
->c_frameindex
= ep
->e_frameindex
;
743 * Update the completion spot mask of the frame we
744 * are attaching 'ocbp' to so the intersection isn't
747 ncbp
->c_framecnt
[0] = ep
->e_framecnt
;
748 ncbp
->c_emask
[0] = ep
->e_emask
;
750 ncbp
->c_flg
|= C_LOOP
;
751 ncbp
->c_dir
= vertices
[0].o_frameindex
;
752 ncbp
->c_framecnt
[1] = fcnt
- 1;
753 if (ncbp
->c_framecnt
[1]) {
754 n
= (vertices
[0].o_intersect
- ocbp
->c_vertex
) /
756 ncbp
->c_emask
[1] = emask
& ~(1 << n
);
758 ncbp
->c_emask
[1] = 0;
759 ncbp
->c_voff
[0] = vertices
[0].o_off
;
762 ncbp
->c_framecnt
[1] = fcnt
;
763 ncbp
->c_emask
[1] = emask
;
764 ncbp
->c_voff
[0] = ep
->e_off
;
767 if ((c
== 1 && debug
> 1) || debug
> 3) {
768 sprintf(fmtbuf
, "%c v%d i%d d%d c %d %d m %x %x o %d %d",
769 "bw"[curcolor
], verts
, ncbp
->c_frameindex
, ncbp
->c_dir
,
770 ncbp
->c_framecnt
[0], ncbp
->c_framecnt
[1],
771 ncbp
->c_emask
[0], ncbp
->c_emask
[1],
772 ncbp
->c_voff
[0], ncbp
->c_voff
[1]);
774 printcombo(ncbp
, fmtbuf
);
778 /* record the empty spots that will complete this combo */
782 /* update board values */
783 updatecombo(ncbp
, curcolor
);
786 if (c
== 1 && debug
> 1 || debug
> 4) {
797 struct elist einfo
[MAXDEPTH
];
798 struct combostr
*ecombo
[MAXDEPTH
]; /* separate from elist to save space */
801 * Add the combostr 'ocbp' to the empty spots list for each empty spot
802 * in 'ocbp' that will complete the combo.
806 struct combostr
*ocbp
;
808 struct combostr
*cbp
, *tcbp
, **cbpp
;
809 struct elist
*ep
, *nep
;
811 int s
, d
, m
, emask
, i
;
815 sprintf(fmtbuf
, "E%c ", "bw"[curcolor
]);
816 printcombo(ocbp
, fmtbuf
+ 3);
820 /* should never happen but check anyway */
821 if ((nframes
= ocbp
->c_nframes
) >= MAXDEPTH
)
825 * The lower level combo can be pointed to by more than one
826 * higher level 'struct combostr' so we can't modify the
827 * lower level. Therefore, higher level combos store the
828 * real mask of the lower level frame in c_emask[0] and the
829 * frame number in c_frameindex.
831 * First we traverse the tree from top to bottom and save the
832 * connection info. Then we traverse the tree from bottom to
833 * top overwriting lower levels with the newer emask information.
835 ep
= &einfo
[nframes
];
836 cbpp
= &ecombo
[nframes
];
837 for (cbp
= ocbp
; (tcbp
= cbp
->c_link
[1]) != NULL
;
838 cbp
= cbp
->c_link
[0]) {
841 *--cbpp
= cbp
->c_link
[1];
842 ep
->e_off
= cbp
->c_voff
[1];
843 ep
->e_frameindex
= cbp
->c_frameindex
;
844 ep
->e_fval
.s
= cbp
->c_linkv
[1].s
;
845 ep
->e_framecnt
= cbp
->c_framecnt
[1];
846 ep
->e_emask
= cbp
->c_emask
[1];
851 *--cbpp
= cbp
->c_link
[0];
852 ep
->e_off
= cbp
->c_voff
[0];
853 ep
->e_frameindex
= 0;
854 ep
->e_fval
.s
= cbp
->c_linkv
[0].s
;
855 ep
->e_framecnt
= cbp
->c_framecnt
[0];
856 ep
->e_emask
= cbp
->c_emask
[0];
858 /* now update the emask info */
860 for (i
= 2, ep
+= 2; i
< nframes
; i
++, ep
++) {
862 nep
= &einfo
[ep
->e_frameindex
];
863 nep
->e_framecnt
= cbp
->c_framecnt
[0];
864 nep
->e_emask
= cbp
->c_emask
[0];
866 if (cbp
->c_flg
& C_LOOP
) {
869 * Account for the fact that this frame connects
870 * to a previous one (thus forming a loop).
872 nep
= &einfo
[cbp
->c_dir
];
873 if (--nep
->e_framecnt
)
874 nep
->e_emask
&= ~(1 << cbp
->c_voff
[0]);
881 * We only need to update the emask values of "complete" loops
882 * to include the intersection spots.
884 if (s
&& ocbp
->c_combo
.c
.a
== 2) {
885 /* process loops from the top down */
886 ep
= &einfo
[nframes
];
890 if (!(cbp
->c_flg
& C_LOOP
))
894 * Update the emask values to include the
895 * intersection spots.
897 nep
= &einfo
[cbp
->c_dir
];
899 nep
->e_emask
= 1 << cbp
->c_voff
[0];
901 ep
->e_emask
= 1 << ep
->e_off
;
902 ep
= &einfo
[ep
->e_frameindex
];
905 ep
->e_emask
= 1 << ep
->e_off
;
906 ep
= &einfo
[ep
->e_frameindex
];
908 } while (ep
!= einfo
);
911 /* check all the frames for completion spots */
912 for (i
= 0, ep
= einfo
, cbpp
= ecombo
; i
< nframes
; i
++, ep
++, cbpp
++) {
913 /* skip this frame if there are no incomplete spots in it */
914 if ((emask
= ep
->e_emask
) == 0)
917 sp
= &board
[cbp
->c_vertex
];
919 for (s
= 0, m
= 1; s
< 5; s
++, sp
+= d
, m
<<= 1) {
920 if (sp
->s_occ
!= EMPTY
|| !(emask
& m
))
923 /* add the combo to the list of empty spots */
924 nep
= (struct elist
*)malloc(sizeof(struct elist
));
926 panic("Out of memory!");
929 nep
->e_frameindex
= i
;
930 if (ep
->e_framecnt
> 1) {
931 nep
->e_framecnt
= ep
->e_framecnt
- 1;
932 nep
->e_emask
= emask
& ~m
;
937 nep
->e_fval
.s
= ep
->e_fval
.s
;
939 sprintf(fmtbuf
, "e %s o%d i%d c%d m%x %x",
949 /* sort by the number of frames in the combo */
950 nep
->e_next
= sp
->s_nempty
;
958 * Update the board value based on the combostr.
959 * This is called only if 'cbp' is a <1,x> combo.
960 * We handle things differently depending on whether the next move
961 * would be trying to "complete" the combo or trying to block it.
964 updatecombo(cbp
, color
)
965 struct combostr
*cbp
;
969 struct combostr
*tcbp
;
975 /* save the top level value for the whole combo */
976 cb
.c
.a
= cbp
->c_combo
.c
.a
;
977 nframes
= cbp
->c_nframes
;
979 if (color
!= nextcolor
)
980 memset(tmpmap
, 0, sizeof(tmpmap
));
982 for (; (tcbp
= cbp
->c_link
[1]) != NULL
; cbp
= cbp
->c_link
[0]) {
984 cb
.c
.b
= cbp
->c_combo
.c
.b
;
985 if (color
== nextcolor
) {
986 /* update the board value for the vertex */
987 sp
= &board
[cbp
->c_vertex
];
988 sp
->s_nforce
[color
]++;
989 if (cb
.s
<= sp
->s_combo
[color
].s
) {
990 if (cb
.s
!= sp
->s_combo
[color
].s
) {
991 sp
->s_combo
[color
].s
= cb
.s
;
992 sp
->s_level
[color
] = nframes
;
993 } else if (nframes
< sp
->s_level
[color
])
994 sp
->s_level
[color
] = nframes
;
997 /* update the board values for each spot in frame */
998 sp
= &board
[s
= tcbp
->c_vertex
];
1000 i
= (flg
& C_OPEN_1
) ? 6 : 5;
1001 for (; --i
>= 0; sp
+= d
, s
+= d
) {
1002 if (sp
->s_occ
!= EMPTY
)
1004 sp
->s_nforce
[color
]++;
1005 if (cb
.s
<= sp
->s_combo
[color
].s
) {
1006 if (cb
.s
!= sp
->s_combo
[color
].s
) {
1007 sp
->s_combo
[color
].s
= cb
.s
;
1008 sp
->s_level
[color
] = nframes
;
1009 } else if (nframes
< sp
->s_level
[color
])
1010 sp
->s_level
[color
] = nframes
;
1016 /* mark the frame as being part of a <1,x> combo */
1017 board
[tcbp
->c_vertex
].s_flg
|= FFLAG
<< tcbp
->c_dir
;
1020 if (color
!= nextcolor
) {
1021 /* update the board values for each spot in frame */
1022 sp
= &board
[s
= cbp
->c_vertex
];
1024 i
= (flg
& C_OPEN_0
) ? 6 : 5;
1025 for (; --i
>= 0; sp
+= d
, s
+= d
) {
1026 if (sp
->s_occ
!= EMPTY
)
1028 sp
->s_nforce
[color
]++;
1029 if (cb
.s
<= sp
->s_combo
[color
].s
) {
1030 if (cb
.s
!= sp
->s_combo
[color
].s
) {
1031 sp
->s_combo
[color
].s
= cb
.s
;
1032 sp
->s_level
[color
] = nframes
;
1033 } else if (nframes
< sp
->s_level
[color
])
1034 sp
->s_level
[color
] = nframes
;
1039 memcpy(forcemap
, tmpmap
, sizeof(tmpmap
));
1041 for (i
= 0; (unsigned int)i
< MAPSZ
; i
++)
1042 forcemap
[i
] &= tmpmap
[i
];
1047 /* mark the frame as being part of a <1,x> combo */
1048 board
[cbp
->c_vertex
].s_flg
|= FFLAG
<< cbp
->c_dir
;
1052 * Add combo to the end of the list.
1055 appendcombo(cbp
, color
)
1056 struct combostr
*cbp
;
1057 int color
__attribute__((__unused__
));
1059 struct combostr
*pcbp
, *ncbp
;
1063 if (ncbp
== (struct combostr
*)0) {
1069 pcbp
= ncbp
->c_prev
;
1077 * Return zero if it is valid to combine frame 'fcbp' with the frames
1078 * in 'cbp' and forms a linked chain of frames (i.e., a tree; no loops).
1079 * Return positive if combining frame 'fcbp' to the frames in 'cbp'
1080 * would form some kind of valid loop. Also return the intersection spots
1081 * in 'vertices[]' beside the known intersection at spot 'osp'.
1082 * Return -1 if 'fcbp' should not be combined with 'cbp'.
1083 * 's' is the combo value for frame 'fcpb'.
1086 checkframes(cbp
, fcbp
, osp
, s
, vertices
)
1087 struct combostr
*cbp
;
1088 struct combostr
*fcbp
;
1089 struct spotstr
*osp
;
1091 struct ovlp_info
*vertices
;
1093 struct combostr
*tcbp
, *lcbp
;
1094 int i
, n
, mask
, flg
, verts
, loop
, index
, fcnt
;
1106 index
= cbp
->c_nframes
;
1107 n
= (fcbp
- frames
) * FAREA
;
1111 * i == which overlap bit to test based on whether 'fcbp' is
1112 * an open or closed frame.
1115 for (; (tcbp
= cbp
->c_link
[1]) != NULL
;
1116 lcbp
= cbp
, cbp
= cbp
->c_link
[0]) {
1118 return (-1); /* fcbp is already included */
1120 /* check for intersection of 'tcbp' with 'fcbp' */
1122 mask
= str
[tcbp
- frames
];
1124 n
= i
+ ((flg
& C_OPEN_1
) != 0);
1125 if (mask
& (1 << n
)) {
1127 * The two frames are not independent if they
1128 * both lie in the same line and intersect at
1129 * more than one point.
1131 if (tcbp
->c_dir
== fcbp
->c_dir
&& (mask
& (0x10 << n
)))
1134 * If this is not the spot we are attaching
1135 * 'fcbp' to and it is a reasonable intersection
1136 * spot, then there might be a loop.
1138 n
= ip
[tcbp
- frames
];
1139 if (osp
!= &board
[n
]) {
1140 /* check to see if this is a valid loop */
1143 if (fcnt
== 0 || cbp
->c_framecnt
[1] == 0)
1146 * Check to be sure the intersection is not
1147 * one of the end points if it is an open
1150 if ((flg
& C_OPEN_1
) &&
1151 (n
== tcbp
->c_vertex
||
1152 n
== tcbp
->c_vertex
+ 5 * dd
[tcbp
->c_dir
]))
1153 return (-1); /* invalid overlap */
1155 (n
== fcbp
->c_vertex
||
1156 n
== fcbp
->c_vertex
+ 5 * dd
[fcbp
->c_dir
]))
1157 return (-1); /* invalid overlap */
1159 vertices
->o_intersect
= n
;
1160 vertices
->o_fcombo
= cbp
;
1161 vertices
->o_link
= 1;
1162 vertices
->o_off
= (n
- tcbp
->c_vertex
) /
1164 vertices
->o_frameindex
= index
;
1168 n
= i
+ ((flg
& C_OPEN_0
) != 0);
1171 return (-1); /* fcbp is already included */
1173 /* check for intersection of 'cbp' with 'fcbp' */
1174 mask
= str
[cbp
- frames
];
1175 if (mask
& (1 << n
)) {
1177 * The two frames are not independent if they
1178 * both lie in the same line and intersect at
1179 * more than one point.
1181 if (cbp
->c_dir
== fcbp
->c_dir
&& (mask
& (0x10 << n
)))
1184 * If this is not the spot we are attaching
1185 * 'fcbp' to and it is a reasonable intersection
1186 * spot, then there might be a loop.
1188 n
= ip
[cbp
- frames
];
1189 if (osp
!= &board
[n
]) {
1190 /* check to see if this is a valid loop */
1193 if (fcnt
== 0 || lcbp
->c_framecnt
[0] == 0)
1196 * Check to be sure the intersection is not
1197 * one of the end points if it is an open
1200 if ((flg
& C_OPEN_0
) &&
1201 (n
== cbp
->c_vertex
||
1202 n
== cbp
->c_vertex
+ 5 * dd
[cbp
->c_dir
]))
1203 return (-1); /* invalid overlap */
1205 (n
== fcbp
->c_vertex
||
1206 n
== fcbp
->c_vertex
+ 5 * dd
[fcbp
->c_dir
]))
1207 return (-1); /* invalid overlap */
1209 vertices
->o_intersect
= n
;
1210 vertices
->o_fcombo
= lcbp
;
1211 vertices
->o_link
= 0;
1212 vertices
->o_off
= (n
- cbp
->c_vertex
) /
1214 vertices
->o_frameindex
= 0;
1222 * Merge sort the frame 'fcbp' and the sorted list of frames 'cbpp' and
1223 * store the result in 'scbpp'. 'curlevel' is the size of the 'cbpp' array.
1224 * Return true if this list of frames is already in the hash list.
1225 * Otherwise, add the new combo to the hash list.
1228 sortcombo(scbpp
, cbpp
, fcbp
)
1229 struct combostr
**scbpp
;
1230 struct combostr
**cbpp
;
1231 struct combostr
*fcbp
;
1233 struct combostr
**spp
, **cpp
;
1234 struct combostr
*cbp
, *ecbp
;
1241 sprintf(fmtbuf
, "sortc: %s%c l%d", stoc(fcbp
->c_vertex
),
1242 pdir
[fcbp
->c_dir
], curlevel
);
1245 for (cpp
= cbpp
; cpp
< cbpp
+ curlevel
; cpp
++) {
1246 sprintf(str
, " %s%c", stoc((*cpp
)->c_vertex
),
1247 pdir
[(*cpp
)->c_dir
]);
1254 /* first build the new sorted list */
1257 cpp
= cbpp
+ curlevel
;
1264 while (cpp
-- != cbpp
);
1268 } while (cpp
!= cbpp
);
1272 /* now check to see if this list of frames has already been seen */
1273 cbp
= hashcombos
[inx
= *scbpp
- frames
];
1274 if (cbp
== (struct combostr
*)0) {
1276 * Easy case, this list hasn't been seen.
1277 * Add it to the hash list.
1279 fcbp
= (struct combostr
*)
1280 ((char *)scbpp
- sizeof(struct combostr
));
1281 hashcombos
[inx
] = fcbp
;
1282 fcbp
->c_next
= fcbp
->c_prev
= fcbp
;
1287 cbpp
= (struct combostr
**)(cbp
+ 1);
1290 cbpp
++; /* first frame is always the same */
1292 if (*--spp
!= *--cpp
)
1294 } while (cpp
!= cbpp
);
1295 /* we found a match */
1300 sprintf(fmtbuf
, "sort1: n%d", n
);
1303 for (cpp
= scbpp
; cpp
< scbpp
+ n
; cpp
++) {
1304 sprintf(str
, " %s%c", stoc((*cpp
)->c_vertex
),
1305 pdir
[(*cpp
)->c_dir
]);
1309 printcombo(cbp
, fmtbuf
);
1313 for (cpp
= cbpp
; cpp
< cbpp
+ n
; cpp
++) {
1314 sprintf(str
, " %s%c", stoc((*cpp
)->c_vertex
),
1315 pdir
[(*cpp
)->c_dir
]);
1324 } while ((cbp
= cbp
->c_next
) != ecbp
);
1326 * This list of frames hasn't been seen.
1327 * Add it to the hash list.
1330 fcbp
= (struct combostr
*)((char *)scbpp
- sizeof(struct combostr
));
1332 fcbp
->c_prev
= ecbp
;
1334 ecbp
->c_next
= fcbp
;
1339 * Print the combo into string 'str'.
1342 printcombo(cbp
, str
)
1343 struct combostr
*cbp
;
1346 struct combostr
*tcbp
;
1348 sprintf(str
, "%x/%d", cbp
->c_combo
.s
, cbp
->c_nframes
);
1350 for (; (tcbp
= cbp
->c_link
[1]) != NULL
; cbp
= cbp
->c_link
[0]) {
1351 sprintf(str
, " %s%c%x", stoc(tcbp
->c_vertex
), pdir
[tcbp
->c_dir
],
1355 sprintf(str
, " %s%c", stoc(cbp
->c_vertex
), pdir
[cbp
->c_dir
]);
1361 struct combostr
*ocbp
;
1363 struct combostr
*cbp
, *tcbp
, **cbpp
;
1364 struct elist
*ep
, *nep
, **epp
;
1368 int r
, n
, flg
, cmask
, omask
;
1370 /* should never happen but check anyway */
1371 if ((nframes
= ocbp
->c_nframes
) >= MAXDEPTH
)
1375 * The lower level combo can be pointed to by more than one
1376 * higher level 'struct combostr' so we can't modify the
1377 * lower level. Therefore, higher level combos store the
1378 * real mask of the lower level frame in c_emask[0] and the
1379 * frame number in c_frameindex.
1381 * First we traverse the tree from top to bottom and save the
1382 * connection info. Then we traverse the tree from bottom to
1383 * top overwriting lower levels with the newer emask information.
1385 ep
= &einfo
[nframes
];
1386 cbpp
= &ecombo
[nframes
];
1387 for (cbp
= ocbp
; tcbp
= cbp
->c_link
[1]; cbp
= cbp
->c_link
[0]) {
1390 *--cbpp
= cbp
->c_link
[1];
1391 ep
->e_off
= cbp
->c_voff
[1];
1392 ep
->e_frameindex
= cbp
->c_frameindex
;
1393 ep
->e_fval
.s
= cbp
->c_linkv
[1].s
;
1394 ep
->e_framecnt
= cbp
->c_framecnt
[1];
1395 ep
->e_emask
= cbp
->c_emask
[1];
1400 *--cbpp
= cbp
->c_link
[0];
1401 ep
->e_off
= cbp
->c_voff
[0];
1402 ep
->e_frameindex
= 0;
1403 ep
->e_fval
.s
= cbp
->c_linkv
[0].s
;
1404 ep
->e_framecnt
= cbp
->c_framecnt
[0];
1405 ep
->e_emask
= cbp
->c_emask
[0];
1407 /* now update the emask info */
1409 for (i
= 2, ep
+= 2; i
< nframes
; i
++, ep
++) {
1411 nep
= &einfo
[ep
->e_frameindex
];
1412 nep
->e_framecnt
= cbp
->c_framecnt
[0];
1413 nep
->e_emask
= cbp
->c_emask
[0];
1415 if (cbp
->c_flg
& C_LOOP
) {
1418 * Account for the fact that this frame connects
1419 * to a previous one (thus forming a loop).
1421 nep
= &einfo
[cbp
->c_dir
];
1422 if (--nep
->e_framecnt
)
1423 nep
->e_emask
&= ~(1 << cbp
->c_voff
[0]);
1430 * We only need to update the emask values of "complete" loops
1431 * to include the intersection spots.
1433 if (s
&& ocbp
->c_combo
.c
.a
== 2) {
1434 /* process loops from the top down */
1435 ep
= &einfo
[nframes
];
1439 if (!(cbp
->c_flg
& C_LOOP
))
1443 * Update the emask values to include the
1444 * intersection spots.
1446 nep
= &einfo
[cbp
->c_dir
];
1447 nep
->e_framecnt
= 1;
1448 nep
->e_emask
= 1 << cbp
->c_voff
[0];
1450 ep
->e_emask
= 1 << ep
->e_off
;
1451 ep
= &einfo
[ep
->e_frameindex
];
1454 ep
->e_emask
= 1 << ep
->e_off
;
1455 ep
= &einfo
[ep
->e_frameindex
];
1457 } while (ep
!= einfo
);
1460 /* mark all the frames with the completion spots */
1461 for (i
= 0, ep
= einfo
, cbpp
= ecombo
; i
< nframes
; i
++, ep
++, cbpp
++) {
1464 sp
= &board
[cbp
->c_vertex
];
1465 d
= dd
[s
= cbp
->c_dir
];
1467 omask
= (IFLAG
| CFLAG
) << s
;
1468 s
= ep
->e_fval
.c
.b
? 6 : 5;
1469 for (; --s
>= 0; sp
+= d
, m
>>= 1)
1470 sp
->s_flg
|= (m
& 1) ? omask
: cmask
;
1475 clearcombo(cbp
, open
)
1476 struct combostr
*cbp
;
1480 struct combostr
*tcbp
;
1483 for (; tcbp
= cbp
->c_link
[1]; cbp
= cbp
->c_link
[0]) {
1484 clearcombo(tcbp
, cbp
->c_flg
& C_OPEN_1
);
1485 open
= cbp
->c_flg
& C_OPEN_0
;
1487 sp
= &board
[cbp
->c_vertex
];
1488 d
= dd
[n
= cbp
->c_dir
];
1489 mask
= ~((IFLAG
| CFLAG
) << n
);
1491 for (; --n
>= 0; sp
+= d
)
1496 list_eq(scbpp
, cbpp
, n
)
1497 struct combostr
**scbpp
;
1498 struct combostr
**cbpp
;
1501 struct combostr
**spp
, **cpp
;
1506 if (*--spp
!= *--cpp
)
1508 } while (cpp
!= cbpp
);
1509 /* we found a match */