image.lisp - clic - Clic is an command line interactive client for gopher written in Common LISP (HTM) git clone git://bitreich.org/clic/ git://enlrupgkhuxnvlhsf6lc3fziv5h2hhfrinws65d7roiv6bfj7d652fid.onion/clic/ (DIR) Log (DIR) Files (DIR) Refs (DIR) Tags (DIR) README (DIR) LICENSE --- image.lisp (22839B) --- 1 ;;;; ------------------------------------------------------------------------- 2 ;;;; Starting, Stopping, Dumping a Lisp image 3 4 (uiop/package:define-package :uiop/image 5 (:use :uiop/common-lisp :uiop/package :uiop/utility :uiop/pathname :uiop/stream :uiop/os) 6 (:export 7 #:*image-dumped-p* #:raw-command-line-arguments #:*command-line-arguments* 8 #:command-line-arguments #:raw-command-line-arguments #:setup-command-line-arguments #:argv0 9 #:*lisp-interaction* 10 #:fatal-condition #:fatal-condition-p 11 #:handle-fatal-condition 12 #:call-with-fatal-condition-handler #:with-fatal-condition-handler 13 #:*image-restore-hook* #:*image-prelude* #:*image-entry-point* 14 #:*image-postlude* #:*image-dump-hook* 15 #:quit #:die #:raw-print-backtrace #:print-backtrace #:print-condition-backtrace 16 #:shell-boolean-exit 17 #:register-image-restore-hook #:register-image-dump-hook 18 #:call-image-restore-hook #:call-image-dump-hook 19 #:restore-image #:dump-image #:create-image 20 )) 21 (in-package :uiop/image) 22 23 (with-upgradability () 24 (defvar *lisp-interaction* t 25 "Is this an interactive Lisp environment, or is it batch processing?") 26 27 (defvar *command-line-arguments* nil 28 "Command-line arguments") 29 30 (defvar *image-dumped-p* nil ; may matter as to how to get to command-line-arguments 31 "Is this a dumped image? As a standalone executable?") 32 33 (defvar *image-restore-hook* nil 34 "Functions to call (in reverse order) when the image is restored") 35 36 (defvar *image-restored-p* nil 37 "Has the image been restored? A boolean, or :in-progress while restoring, :in-regress while dumping") 38 39 (defvar *image-prelude* nil 40 "a form to evaluate, or string containing forms to read and evaluate 41 when the image is restarted, but before the entry point is called.") 42 43 (defvar *image-entry-point* nil 44 "a function with which to restart the dumped image when execution is restored from it.") 45 46 (defvar *image-postlude* nil 47 "a form to evaluate, or string containing forms to read and evaluate 48 before the image dump hooks are called and before the image is dumped.") 49 50 (defvar *image-dump-hook* nil 51 "Functions to call (in order) when before an image is dumped")) 52 53 (eval-when (#-lispworks :compile-toplevel :load-toplevel :execute) 54 (deftype fatal-condition () 55 `(and serious-condition #+clozure (not ccl:process-reset)))) 56 57 ;;; Exiting properly or im- 58 (with-upgradability () 59 (defun quit (&optional (code 0) (finish-output t)) 60 "Quits from the Lisp world, with the given exit status if provided. 61 This is designed to abstract away the implementation specific quit forms." 62 (when finish-output ;; essential, for ClozureCL, and for standard compliance. 63 (finish-outputs)) 64 #+(or abcl xcl) (ext:quit :status code) 65 #+allegro (excl:exit code :quiet t) 66 #+(or clasp ecl) (si:quit code) 67 #+clisp (ext:quit code) 68 #+clozure (ccl:quit code) 69 #+cormanlisp (win32:exitprocess code) 70 #+(or cmucl scl) (unix:unix-exit code) 71 #+gcl (system:quit code) 72 #+genera (error "~S: You probably don't want to Halt Genera. (code: ~S)" 'quit code) 73 #+lispworks (lispworks:quit :status code :confirm nil :return nil :ignore-errors-p t) 74 #+mcl (progn code (ccl:quit)) ;; or should we use FFI to call libc's exit(3) ? 75 #+mkcl (mk-ext:quit :exit-code code) 76 #+sbcl #.(let ((exit (find-symbol* :exit :sb-ext nil)) 77 (quit (find-symbol* :quit :sb-ext nil))) 78 (cond 79 (exit `(,exit :code code :abort (not finish-output))) 80 (quit `(,quit :unix-status code :recklessly-p (not finish-output))))) 81 #-(or abcl allegro clasp clisp clozure cmucl ecl gcl genera lispworks mcl mkcl sbcl scl xcl) 82 (not-implemented-error 'quit "(called with exit code ~S)" code)) 83 84 (defun die (code format &rest arguments) 85 "Die in error with some error message" 86 (with-safe-io-syntax () 87 (ignore-errors 88 (format! *stderr* "~&~?~&" format arguments))) 89 (quit code)) 90 91 (defun raw-print-backtrace (&key (stream *debug-io*) count condition) 92 "Print a backtrace, directly accessing the implementation" 93 (declare (ignorable stream count condition)) 94 #+abcl 95 (loop :for i :from 0 96 :for frame :in (sys:backtrace (or count most-positive-fixnum)) :do 97 (safe-format! stream "~&~D: ~A~%" i frame)) 98 #+allegro 99 (let ((*terminal-io* stream) 100 (*standard-output* stream) 101 (tpl:*zoom-print-circle* *print-circle*) 102 (tpl:*zoom-print-level* *print-level*) 103 (tpl:*zoom-print-length* *print-length*)) 104 (tpl:do-command "zoom" 105 :from-read-eval-print-loop nil 106 :count (or count t) 107 :all t)) 108 #+(or clasp ecl mkcl) 109 (let* ((top (si:ihs-top)) 110 (repeats (if count (min top count) top)) 111 (backtrace (loop :for ihs :from 0 :below top 112 :collect (list (si::ihs-fun ihs) 113 (si::ihs-env ihs))))) 114 (loop :for i :from 0 :below repeats 115 :for frame :in (nreverse backtrace) :do 116 (safe-format! stream "~&~D: ~S~%" i frame))) 117 #+clisp 118 (system::print-backtrace :out stream :limit count) 119 #+(or clozure mcl) 120 (let ((*debug-io* stream)) 121 #+clozure (ccl:print-call-history :count count :start-frame-number 1) 122 #+mcl (ccl:print-call-history :detailed-p nil) 123 (finish-output stream)) 124 #+(or cmucl scl) 125 (let ((debug:*debug-print-level* *print-level*) 126 (debug:*debug-print-length* *print-length*)) 127 (debug:backtrace (or count most-positive-fixnum) stream)) 128 #+gcl 129 (let ((*debug-io* stream)) 130 (ignore-errors 131 (with-safe-io-syntax () 132 (if condition 133 (conditions::condition-backtrace condition) 134 (system::simple-backtrace))))) 135 #+lispworks 136 (let ((dbg::*debugger-stack* 137 (dbg::grab-stack nil :how-many (or count most-positive-fixnum))) 138 (*debug-io* stream) 139 (dbg:*debug-print-level* *print-level*) 140 (dbg:*debug-print-length* *print-length*)) 141 (dbg:bug-backtrace nil)) 142 #+mezzano 143 (let ((*standard-output* stream)) 144 (sys.int::backtrace count)) 145 #+sbcl 146 (sb-debug:print-backtrace :stream stream :count (or count most-positive-fixnum)) 147 #+xcl 148 (loop :for i :from 0 :below (or count most-positive-fixnum) 149 :for frame :in (extensions:backtrace-as-list) :do 150 (safe-format! stream "~&~D: ~S~%" i frame))) 151 152 (defun print-backtrace (&rest keys &key stream count condition) 153 "Print a backtrace" 154 (declare (ignore stream count condition)) 155 (with-safe-io-syntax (:package :cl) 156 (let ((*print-readably* nil) 157 (*print-circle* t) 158 (*print-miser-width* 75) 159 (*print-length* nil) 160 (*print-level* nil) 161 (*print-pretty* t)) 162 (ignore-errors (apply 'raw-print-backtrace keys))))) 163 164 (defun print-condition-backtrace (condition &key (stream *stderr*) count) 165 "Print a condition after a backtrace triggered by that condition" 166 ;; We print the condition *after* the backtrace, 167 ;; for the sake of who sees the backtrace at a terminal. 168 ;; It is up to the caller to print the condition *before*, with some context. 169 (print-backtrace :stream stream :count count :condition condition) 170 (when condition 171 (safe-format! stream "~&Above backtrace due to this condition:~%~A~&" 172 condition))) 173 174 (defun fatal-condition-p (condition) 175 "Is the CONDITION fatal?" 176 (typep condition 'fatal-condition)) 177 178 (defun handle-fatal-condition (condition) 179 "Handle a fatal CONDITION: 180 depending on whether *LISP-INTERACTION* is set, enter debugger or die" 181 (cond 182 (*lisp-interaction* 183 (invoke-debugger condition)) 184 (t 185 (safe-format! *stderr* "~&Fatal condition:~%~A~%" condition) 186 (print-condition-backtrace condition :stream *stderr*) 187 (die 99 "~A" condition)))) 188 189 (defun call-with-fatal-condition-handler (thunk) 190 "Call THUNK in a context where fatal conditions are appropriately handled" 191 (handler-bind ((fatal-condition #'handle-fatal-condition)) 192 (funcall thunk))) 193 194 (defmacro with-fatal-condition-handler ((&optional) &body body) 195 "Execute BODY in a context where fatal conditions are appropriately handled" 196 `(call-with-fatal-condition-handler #'(lambda () ,@body))) 197 198 (defun shell-boolean-exit (x) 199 "Quit with a return code that is 0 iff argument X is true" 200 (quit (if x 0 1)))) 201 202 203 ;;; Using image hooks 204 (with-upgradability () 205 (defun register-image-restore-hook (hook &optional (call-now-p t)) 206 "Regiter a hook function to be run when restoring a dumped image" 207 (register-hook-function '*image-restore-hook* hook call-now-p)) 208 209 (defun register-image-dump-hook (hook &optional (call-now-p nil)) 210 "Register a the hook function to be run before to dump an image" 211 (register-hook-function '*image-dump-hook* hook call-now-p)) 212 213 (defun call-image-restore-hook () 214 "Call the hook functions registered to be run when restoring a dumped image" 215 (call-functions (reverse *image-restore-hook*))) 216 217 (defun call-image-dump-hook () 218 "Call the hook functions registered to be run before to dump an image" 219 (call-functions *image-dump-hook*))) 220 221 222 ;;; Proper command-line arguments 223 (with-upgradability () 224 (defun raw-command-line-arguments () 225 "Find what the actual command line for this process was." 226 #+abcl ext:*command-line-argument-list* ; Use 1.0.0 or later! 227 #+allegro (sys:command-line-arguments) ; default: :application t 228 #+(or clasp ecl) (loop :for i :from 0 :below (si:argc) :collect (si:argv i)) 229 #+clisp (coerce (ext:argv) 'list) 230 #+clozure ccl:*command-line-argument-list* 231 #+(or cmucl scl) extensions:*command-line-strings* 232 #+gcl si:*command-args* 233 #+(or genera mcl mezzano) nil 234 #+lispworks sys:*line-arguments-list* 235 #+mkcl (loop :for i :from 0 :below (mkcl:argc) :collect (mkcl:argv i)) 236 #+sbcl sb-ext:*posix-argv* 237 #+xcl system:*argv* 238 #-(or abcl allegro clasp clisp clozure cmucl ecl gcl genera lispworks mcl mezzano mkcl sbcl scl xcl) 239 (not-implemented-error 'raw-command-line-arguments)) 240 241 (defun command-line-arguments (&optional (arguments (raw-command-line-arguments))) 242 "Extract user arguments from command-line invocation of current process. 243 Assume the calling conventions of a generated script that uses -- 244 if we are not called from a directly executable image." 245 (block nil 246 #+abcl (return arguments) 247 ;; SBCL and Allegro already separate user arguments from implementation arguments. 248 #-(or sbcl allegro) 249 (unless (eq *image-dumped-p* :executable) 250 ;; LispWorks command-line processing isn't transparent to the user 251 ;; unless you create a standalone executable; in that case, 252 ;; we rely on cl-launch or some other script to set the arguments for us. 253 #+lispworks (return *command-line-arguments*) 254 ;; On other implementations, on non-standalone executables, 255 ;; we trust cl-launch or whichever script starts the program 256 ;; to use -- as a delimiter between implementation arguments and user arguments. 257 #-lispworks (setf arguments (member "--" arguments :test 'string-equal))) 258 (rest arguments))) 259 260 (defun argv0 () 261 "On supported implementations (most that matter), or when invoked by a proper wrapper script, 262 return a string that for the name with which the program was invoked, i.e. argv[0] in C. 263 Otherwise, return NIL." 264 (cond 265 ((eq *image-dumped-p* :executable) ; yes, this ARGV0 is our argv0 ! 266 ;; NB: not currently available on ABCL, Corman, Genera, MCL 267 (or #+(or allegro clisp clozure cmucl gcl lispworks sbcl scl xcl) 268 (first (raw-command-line-arguments)) 269 #+(or clasp ecl) (si:argv 0) #+mkcl (mkcl:argv 0))) 270 (t ;; argv[0] is the name of the interpreter. 271 ;; The wrapper script can export __CL_ARGV0. cl-launch does as of 4.0.1.8. 272 (getenvp "__CL_ARGV0")))) 273 274 (defun setup-command-line-arguments () 275 (setf *command-line-arguments* (command-line-arguments))) 276 277 (defun restore-image (&key 278 (lisp-interaction *lisp-interaction*) 279 (restore-hook *image-restore-hook*) 280 (prelude *image-prelude*) 281 (entry-point *image-entry-point*) 282 (if-already-restored '(cerror "RUN RESTORE-IMAGE ANYWAY"))) 283 "From a freshly restarted Lisp image, restore the saved Lisp environment 284 by setting appropriate variables, running various hooks, and calling any specified entry point. 285 286 If the image has already been restored or is already being restored, as per *IMAGE-RESTORED-P*, 287 call the IF-ALREADY-RESTORED error handler (by default, a continuable error), and do return 288 immediately to the surrounding restore process if allowed to continue. 289 290 Then, comes the restore process itself: 291 First, call each function in the RESTORE-HOOK, 292 in the order they were registered with REGISTER-IMAGE-RESTORE-HOOK. 293 Second, evaluate the prelude, which is often Lisp text that is read, 294 as per EVAL-INPUT. 295 Third, call the ENTRY-POINT function, if any is specified, with no argument. 296 297 The restore process happens in a WITH-FATAL-CONDITION-HANDLER, so that if LISP-INTERACTION is NIL, 298 any unhandled error leads to a backtrace and an exit with an error status. 299 If LISP-INTERACTION is NIL, the process also exits when no error occurs: 300 if neither restart nor entry function is provided, the program will exit with status 0 (success); 301 if a function was provided, the program will exit after the function returns (if it returns), 302 with status 0 if and only if the primary return value of result is generalized boolean true, 303 and with status 1 if this value is NIL. 304 305 If LISP-INTERACTION is true, unhandled errors will take you to the debugger, and the result 306 of the function will be returned rather than interpreted as a boolean designating an exit code." 307 (when *image-restored-p* 308 (if if-already-restored 309 (call-function if-already-restored "Image already ~:[being ~;~]restored" 310 (eq *image-restored-p* t)) 311 (return-from restore-image))) 312 (with-fatal-condition-handler () 313 (setf *lisp-interaction* lisp-interaction) 314 (setf *image-restore-hook* restore-hook) 315 (setf *image-prelude* prelude) 316 (setf *image-restored-p* :in-progress) 317 (call-image-restore-hook) 318 (standard-eval-thunk prelude) 319 (setf *image-restored-p* t) 320 (let ((results (multiple-value-list 321 (if entry-point 322 (call-function entry-point) 323 t)))) 324 (if lisp-interaction 325 (values-list results) 326 (shell-boolean-exit (first results))))))) 327 328 329 ;;; Dumping an image 330 331 (with-upgradability () 332 (defun dump-image (filename &key output-name executable 333 (postlude *image-postlude*) 334 (dump-hook *image-dump-hook*) 335 #+clozure prepend-symbols #+clozure (purify t) 336 #+sbcl compression 337 #+(and sbcl os-windows) application-type) 338 "Dump an image of the current Lisp environment at pathname FILENAME, with various options. 339 340 First, finalize the image, by evaluating the POSTLUDE as per EVAL-INPUT, then calling each of 341 the functions in DUMP-HOOK, in reverse order of registration by REGISTER-DUMP-HOOK. 342 343 If EXECUTABLE is true, create an standalone executable program that calls RESTORE-IMAGE on startup. 344 345 Pass various implementation-defined options, such as PREPEND-SYMBOLS and PURITY on CCL, 346 or COMPRESSION on SBCL, and APPLICATION-TYPE on SBCL/Windows." 347 ;; Note: at least SBCL saves only global values of variables in the heap image, 348 ;; so make sure things you want to dump are NOT just local bindings shadowing the global values. 349 (declare (ignorable filename output-name executable)) 350 (setf *image-dumped-p* (if executable :executable t)) 351 (setf *image-restored-p* :in-regress) 352 (setf *image-postlude* postlude) 353 (standard-eval-thunk *image-postlude*) 354 (setf *image-dump-hook* dump-hook) 355 (call-image-dump-hook) 356 (setf *image-restored-p* nil) 357 #-(or clisp clozure (and cmucl executable) lispworks sbcl scl) 358 (when executable 359 (not-implemented-error 'dump-image "dumping an executable")) 360 #+allegro 361 (progn 362 (sys:resize-areas :global-gc t :pack-heap t :sift-old-areas t :tenure t) ; :new 5000000 363 (excl:dumplisp :name filename :suppress-allegro-cl-banner t)) 364 #+clisp 365 (apply #'ext:saveinitmem filename 366 :quiet t 367 :start-package *package* 368 :keep-global-handlers nil 369 :executable (if executable 0 t) ;--- requires clisp 2.48 or later, still catches --clisp-x 370 (when executable 371 (list 372 ;; :parse-options nil ;--- requires a non-standard patch to clisp. 373 :norc t :script nil :init-function #'restore-image))) 374 #+clozure 375 (flet ((dump (prepend-kernel) 376 (ccl:save-application filename :prepend-kernel prepend-kernel :purify purify 377 :toplevel-function (when executable #'restore-image)))) 378 ;;(setf ccl::*application* (make-instance 'ccl::lisp-development-system)) 379 (if prepend-symbols 380 (with-temporary-file (:prefix "ccl-symbols-" :direction :output :pathname path) 381 (require 'elf) 382 (funcall (fdefinition 'ccl::write-elf-symbols-to-file) path) 383 (dump path)) 384 (dump t))) 385 #+(or cmucl scl) 386 (progn 387 (ext:gc :full t) 388 (setf ext:*batch-mode* nil) 389 (setf ext::*gc-run-time* 0) 390 (apply 'ext:save-lisp filename 391 :allow-other-keys t ;; hush SCL and old versions of CMUCL 392 #+(and cmucl executable) :executable #+(and cmucl executable) t 393 (when executable '(:init-function restore-image :process-command-line nil 394 :quiet t :load-init-file nil :site-init nil)))) 395 #+gcl 396 (progn 397 (si::set-hole-size 500) (si::gbc nil) (si::sgc-on t) 398 (si::save-system filename)) 399 #+lispworks 400 (if executable 401 (lispworks:deliver 'restore-image filename 0 :interface nil) 402 (hcl:save-image filename :environment nil)) 403 #+sbcl 404 (progn 405 ;;(sb-pcl::precompile-random-code-segments) ;--- it is ugly slow at compile-time (!) when the initial core is a big CLOS program. If you want it, do it yourself 406 (setf sb-ext::*gc-run-time* 0) 407 (apply 'sb-ext:save-lisp-and-die filename 408 :executable t ;--- always include the runtime that goes with the core 409 (append 410 (when compression (list :compression compression)) 411 ;;--- only save runtime-options for standalone executables 412 (when executable (list :toplevel #'restore-image :save-runtime-options t)) 413 #+(and sbcl os-windows) ;; passing :application-type :gui will disable the console window. 414 ;; the default is :console - only works with SBCL 1.1.15 or later. 415 (when application-type (list :application-type application-type))))) 416 #-(or allegro clisp clozure cmucl gcl lispworks sbcl scl) 417 (not-implemented-error 'dump-image)) 418 419 (defun create-image (destination lisp-object-files 420 &key kind output-name prologue-code epilogue-code extra-object-files 421 (prelude () preludep) (postlude () postludep) 422 (entry-point () entry-point-p) build-args no-uiop) 423 (declare (ignorable destination lisp-object-files extra-object-files kind output-name 424 prologue-code epilogue-code prelude preludep postlude postludep 425 entry-point entry-point-p build-args no-uiop)) 426 "On ECL, create an executable at pathname DESTINATION from the specified OBJECT-FILES and options" 427 ;; Is it meaningful to run these in the current environment? 428 ;; only if we also track the object files that constitute the "current" image, 429 ;; and otherwise simulate dump-image, including quitting at the end. 430 #-(or clasp ecl mkcl) (not-implemented-error 'create-image) 431 #+(or clasp ecl mkcl) 432 (let ((epilogue-code 433 (if no-uiop 434 epilogue-code 435 (let ((forms 436 (append 437 (when epilogue-code `(,epilogue-code)) 438 (when postludep `((setf *image-postlude* ',postlude))) 439 (when preludep `((setf *image-prelude* ',prelude))) 440 (when entry-point-p `((setf *image-entry-point* ',entry-point))) 441 (case kind 442 ((:image) 443 (setf kind :program) ;; to ECL, it's just another program. 444 `((setf *image-dumped-p* t) 445 (si::top-level #+(or clasp ecl) t) (quit))) 446 ((:program) 447 `((setf *image-dumped-p* :executable) 448 (shell-boolean-exit 449 (restore-image)))))))) 450 (when forms `(progn ,@forms)))))) 451 (check-type kind (member :dll :shared-library :lib :static-library 452 :fasl :fasb :program)) 453 (apply #+clasp 'cmp:builder #+clasp kind 454 #+(or ecl mkcl) 455 (ecase kind 456 ((:dll :shared-library) 457 #+ecl 'c::build-shared-library #+mkcl 'compiler:build-shared-library) 458 ((:lib :static-library) 459 #+ecl 'c::build-static-library #+mkcl 'compiler:build-static-library) 460 ((:fasl #+ecl :fasb) 461 #+ecl 'c::build-fasl #+mkcl 'compiler:build-fasl) 462 #+mkcl ((:fasb) 'compiler:build-bundle) 463 ((:program) 464 #+ecl 'c::build-program #+mkcl 'compiler:build-program)) 465 (pathname destination) 466 #+(or clasp ecl) :lisp-files #+mkcl :lisp-object-files 467 (append lisp-object-files #+(or clasp ecl) extra-object-files) 468 #+ecl :init-name 469 #+ecl (getf build-args :init-name) 470 (append 471 (when prologue-code `(:prologue-code ,prologue-code)) 472 (when epilogue-code `(:epilogue-code ,epilogue-code)) 473 #+mkcl (when extra-object-files `(:object-files ,extra-object-files)) 474 build-args))))) 475 476 477 ;;; Some universal image restore hooks 478 (with-upgradability () 479 (map () 'register-image-restore-hook 480 '(setup-stdin setup-stdout setup-stderr 481 setup-command-line-arguments setup-temporary-directory 482 #+abcl detect-os)))