Skip to content

Latest commit

 

History

History
73 lines (58 loc) · 2.24 KB

0103-trivial-tco.org

File metadata and controls

73 lines (58 loc) · 2.24 KB

trivial-tco

This library could be considered as a portability layer for tail call optimization.

When I first found it, I decided it implements a TCO for implementations which do not support it by doing a trampolining trick like this. But I was wrong.

It does ensure the proper declaration is used on implementations which support a proper TCO and signals warning or error on others.

Here is an example on SBCL, which supports TCO only if speed declared to be greater or equal to debug:

POFTHEDAY> (declaim (optimize (debug 3) (speed 1))

POFTHEDAY> (labels ((sum-aux (acc x)
                        (if (zerop x)
                            acc
                            (sum-aux (+ acc x) (- x 1))))
                      (sum (n)
                        (sum-aux 0 n)))
               (sum 1000000))
Control stack guard page temporarily disabled: proceed with caution
; Debugger entered on #<SB-KERNEL::CONTROL-STACK-EXHAUSTED {1004F12E73}>
[1] POFTHEDAY> 
; Evaluation aborted on #<SB-KERNEL::CONTROL-STACK-EXHAUSTED {1004F12E73}>

POFTHEDAY> (tco:with-tail-call-optimization ()
             (labels ((sum-aux (acc x)
                        (if (zerop x)
                            acc
                            (sum-aux (+ acc x) (- x 1))))
                      (sum (n)
                        (sum-aux 0 n)))
               (sum 1000000)))
500000500000 (39 bits, #x746A5A2920)

This macro gets expanded into:

(let ()
  (declare (optimize (speed 3)))
  (labels ((sum-aux (acc x)
             (if (zerop x)
                 acc
                 (sum-aux (+ acc x) (- x 1))))
           (sum (n)
             (sum-aux 0 n)))
    (sum 1000000)))

That is it. Use this library, if you want to employ a tail call and want to ensure they a properly optimized by your Lisp implementation.

Maybe this article by Marc Simpson will be interesting for you. It investigates which Common Lisp implementations have a proper TCO implementation.