-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdominator-sniper-dec.sml
executable file
·227 lines (206 loc) · 9.29 KB
/
dominator-sniper-dec.sml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
(* This is the legacy version of sniper
that uses repeated decrement, which is pretty dumb. *)
structure Sniper :> DOMINATOR =
struct
structure GS = GameState
datatype src = datatype Kompiler.src
datatype dosturn = datatype DOS.dosturn
structure EP = EmitProgram
infix 9 --
val op -- = Apply
val $ = Var
fun \ x exp = Lambda (x, exp)
infixr 1 `
fun a ` b = a b
datatype mode =
(* First need to create the gun program, which never changes. *)
CreateGun
(* Once the program is in place, find a target and put it in target_slot. *)
| BuildingGun of { status : EP.status ref, gun_slot : int, target_slot : int }
(* We loop between the following two for the rest of time. We could consider
making a new gun, but the current strategy is to hope for the medic to
save us. *)
| FindTarget of { gun_slot : int, target_slot : int }
(* Now keep attacking until it's dead, or something happens to our
equipment. *)
| Attacking of { status : EP.status ref,
shots : int ref,
(* Slot containing the gun. *)
gun_slot : int,
(* Slot containing the index of
the target, which is referred
to by gun_slot. *)
target_slot : int,
target : int }
val compare_scores = ListUtil.bysecond Real.compare
val lastmsg = ref ""
val eprint =
fn s => if s = !lastmsg
then ()
else (eprint ("[SNIPER] " ^ s ^ "\n"); lastmsg := s)
fun create () =
let
(* Maybe should have a lower bound on what it will
consider valuable, and reduce priority if there
are no current high-value targets. *)
(* Makes a program that attacks the target slot index,
and then returns itself (so it sticks around). *)
fun attackprogram target_slot prog_slot =
let
fun repeat 1 e = e
| repeat n e = repeat (n - 1) e -- e
val dec =
(\"target" `
(* empirical. Attack is MUCH more efficient.
Should use attack. *)
repeat 100 (Card LTG.Dec -- $"target")) --
(Card LTG.Get -- Int target_slot)
val prog = Kompiler.run_and_return_self dec
in
eprint (Kompiler.src2str prog);
Kompiler.compile prog prog_slot
end handle (e as Kompiler.Kompiler s) =>
let in
eprint ("Kompilation failed: " ^ s ^ "\n");
raise e
end
fun preview dos = ()
(* This is just for diagnostics. *)
val was_stuck = ref false
val mode = ref CreateGun
fun taketurn dos =
let val gs = DOS.gamestate dos
in
case !mode of
CreateGun =>
(case (DOS.reserve_slot dos, DOS.reserve_addressable_slot dos) of
(NONE, NONE) => DOS.Can'tRun
| (NONE, SOME s) => (DOS.release_slot dos s; DOS.Can'tRun)
| (SOME s, NONE) => (DOS.release_slot dos s; DOS.Can'tRun)
| (SOME gun_slot, SOME target_slot) =>
let
val prog = attackprogram target_slot gun_slot
val (stat, child_pid) = EP.emitspawn dos prog
in
eprint ("Assembling gun in: " ^ Int.toString gun_slot ^
" reading from: " ^ Int.toString target_slot ^
" Program length: " ^ Int.toString (length prog));
mode := BuildingGun { status = stat, gun_slot = gun_slot,
target_slot = target_slot };
was_stuck := false;
taketurn dos
end)
| BuildingGun { status = ref (EP.Progress _), ... } => DOS.Can'tRun
(* Hope that medic helps us. *)
| BuildingGun { status = ref (EP.Paused _), ... } => DOS.Can'tRun
| BuildingGun { status = ref EP.Done, gun_slot, target_slot } =>
let in
eprint ("Gun is assembled :D");
mode := FindTarget { gun_slot = gun_slot, target_slot = target_slot };
was_stuck := false;
taketurn dos
end
| FindTarget { gun_slot, target_slot } =>
(* XXX check if gun is dead? *)
if LTG.slotisdead (GS.myside gs) gun_slot orelse
LTG.slotisdead (GS.myside gs) target_slot
then
let in
if !was_stuck
then eprint ("Gun/target slots are dead!")
else ();
was_stuck := true;
Can'tRun
end
else
let
val slots = List.tabulate (256, fn i =>
(i, GS.scoreopponentslot gs i))
(* Maybe should have a lower bound on what it will
consider valuable, and just heal/revive if there
are no current high-value targets. *)
val (best, _) = ListUtil.max compare_scores slots
(* Put the number in our targeting slot. *)
val prog = Kompiler.compile (Int (255 - best)) target_slot
(* Ignore child pid since we never kill it. *)
val (stat, child_pid) = EP.emitspawn dos prog
in
mode := Attacking { shots = ref 0,
gun_slot = gun_slot,
target_slot = target_slot,
target = best,
status = stat };
was_stuck := false;
Can'tRun
end
| Attacking { status = ref (EP.Progress _), ... } =>
let in
if !was_stuck
then eprint ("Unstuck! Thanks!")
else ();
was_stuck := false;
Can'tRun
end
| Attacking { status = ref EP.Done, shots, gun_slot, target, target_slot } =>
if LTG.slotisdead (GS.myside gs) gun_slot orelse
LTG.slotisdead (GS.myside gs) target_slot
then
let in
if !was_stuck
then eprint ("Sniper's gun/target slot " ^
Int.toString gun_slot ^ "/" ^
Int.toString target_slot ^
" was killed! Hoping for medic.\n")
else ();
was_stuck := true;
Can'tRun
end
else
let val theirside = GS.theirside gs
val health = Array.sub (#2 theirside, target)
val num = Array.sub (#1 (GS.myside gs), target_slot)
(*
val () = eprint ("target_slot contains: " ^
LTG.valtos num ^ " and target health is " ^
Int.toString health)
val () = Array.appi (fn (i, n) =>
if n < 10000
then eprint ("But " ^ Int.toString i ^ " has vitality " ^
Int.toString n)
else ()) (#2 theirside)
*)
in
was_stuck := false;
if health <= 0
then
let in
eprint ("Success! Killed slot " ^
Int.toString target ^
" in " ^ Int.toString (!shots) ^ " shots.");
mode := FindTarget { gun_slot = gun_slot,
target_slot = target_slot };
taketurn dos
end
else
let in
(* Otherwise keep attacking. *)
(* eprint ("Attack!"); *)
shots := !shots + 1;
DOS.Turn (LTG.RightApply (gun_slot, LTG.I))
end
end
(* Optimistically hope that medic will heal it? *)
| Attacking { status = ref (EP.Paused i), ... } =>
let in
if !was_stuck
then ()
else eprint ("Stuck because slot " ^ Int.toString i ^ " is dead.");
was_stuck := true;
Can'tRun
end
end
in
{ preview = preview,
taketurn = taketurn }
end
end