Mostlikely the tabling solution is in essence Dijkstras algorithm.
Here is a tabling solution, only “i”, “d” and “s”, no “o” yet.
The following fails:
% edge(+Atom, +Integer, -Integer)
edge(s, N, M) :- (N = 0 -> M = 16; H is sqrt(N),
M is truncate(H), H is float(M)).
edge(i, N, M) :- (N = 0 -> M = 255; M is N-1).
edge(d, N, M) :- M is N+1.
% path(+Integer, -Integer, -List)
:- table(path(_, _, lattice(shortest/3))).
path(N, N, [o]).
path(N, M, [X|L]) :- path(N, H, L), edge(X, H, M).
It gives this answer:
?- path(90, 0, L), write(L), nl.
ERROR: '$tbl_wkl_add_answer'/4: Not enough resources: private_table_space
Now if I bound the length of L:
path(N, N, [o]).
path(N, M, [X|L]) :- path(N, H, L), length(L, K), K =< 21, edge(X, H, M).
It works, and it is quite speedy:
?- time(path(90, 0, L)), write(L), nl.
% 2,979 inferences, 0.000 CPU in 0.001 seconds (0% CPU, Infinite Lips)
[i,i,i,s,s,i,i,i,i,i,i,i,i,i,o]
Edit 07.10.2023
All together, the whole dead fish challenge with SWI-Prolog lattice tabling:
?- time(total(S,M,C)).
% 835,418 inferences, 0.188 CPU in 0.204 seconds (92% CPU, 4455563 Lips)
S = 3215,
M = 22,
C = 255.
Little bit slower than my hand rolled memoization,
since the bound K =< 21
leaves a lot of search space room,
making it slower than iterative deepending with tabling.
0.2 seconds is ca 9-times faster than the 1.8 seconds reported
by @peter.ludemann . But need to check what it does for
“o”. With “o” we can use a smaller bound, maybe its faster.