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 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871
|
;;; proof-shell.el --- Proof General shell mode.
;;
;; Copyright (C) 1994-2011 LFCS Edinburgh.
;; Authors: David Aspinall, Yves Bertot, Healfdene Goguen,
;; Thomas Kleymann and Dilip Sequeira
;; License: GPL (GNU GENERAL PUBLIC LICENSE)
;;
;; $Id: proof-shell.el,v 12.11 2012/05/24 21:11:06 tews Exp $
;;
;;; Commentary:
;;
;; Mode for buffer which interacts with proof assistant.
;; Includes management of a queue of commands waiting to be sent.
;;
;;; Code:
(require 'cl) ; set-difference, every
(eval-when-compile
(require 'span)
(require 'proof-utils))
(require 'scomint)
(require 'pg-response)
(require 'pg-goals)
(require 'pg-user) ; proof-script, new-command-advance
(require 'proof-tree)
;;
;; Internal variables used by proof shell
;;
(defvar proof-marker nil
"Marker in proof shell buffer pointing to previous command input.")
(defvar proof-action-list nil
"The main queue of things to do: spans, commands and actions.
The value is a list of lists of the form
(SPAN COMMANDS ACTION [DISPLAYFLAGS])
which is the queue of things to do.
SPAN is a region in the sources, where COMMANDS come from. Often,
additional properties are recorded as properties of SPAN.
COMMANDS is a list of strings, holding the text to be send to the
prover. It might be the empty list if nothing needs to be sent to
the prover, such as, for comments. Usually COMMANDS
contains just 1 string, but it might also contains more elements.
The text should be obtained with
`(mapconcat 'identity COMMANDS \" \")', where the last argument
is a space.
ACTION is the callback to be invoked when this item has been
processed by the prover. For normal scripting items it is
`proof-done-advancing', for retract items
`proof-done-retracting', but there are more possibilities (e.g.
`proof-done-invisible', `proof-shell-set-silent',
`proof-shell-clear-silent' and `proof-tree-show-goal-callback').
The DISPLAYFLAGS are set
for non-scripting commands or for when scripting should not
bother the user. They may include
'invisible non-script command (`proof-shell-invisible-command')
'no-response-display do not display messages in *response* buffer
'no-error-display do not display errors/take error action
'no-goals-display do not goals in *goals* buffer
'proof-tree-show-subgoal item inserted by the proof-tree package
Note that 'invisible does not imply any of the others. If flags
are non-empty, interactive cues will be surpressed. (E.g.,
printing hints).
See the functions `proof-start-queue' and `proof-shell-exec-loop'.")
(defsubst proof-shell-invoke-callback (listitem)
"From `proof-action-list' LISTITEM, invoke the callback on the span."
(condition-case nil
(funcall (nth 2 listitem) (car listitem))
(error nil)))
;; We record the last output from the prover and a flag indicating its
;; type, as well as a previous ("delayed") version for when the end
;; of the queue is reached or an error or interrupt occurs.
;;
;; See `proof-shell-last-output', `proof-shell-last-prompt' in
;; pg-vars.el
(defvar proof-shell-last-goals-output ""
"The last displayed goals string.")
(defvar proof-shell-last-response-output ""
"The last displayed response message.")
(defvar proof-shell-delayed-output-start nil
"A record of the start of the previous output in the shell buffer.
The previous output is held back for processing at end of queue.")
(defvar proof-shell-delayed-output-end nil
"A record of the start of the previous output in the shell buffer.
The previous output is held back for processing at end of queue.")
(defvar proof-shell-delayed-output-flags nil
"A copy of the `proof-action-list' flags for `proof-shell-delayed-output'.")
(defvar proof-shell-interrupt-pending nil
"A flag indicating an interrupt is pending.
This ensures that the proof queue will be interrupted even if no
interrupt message is printed from the prover after the last output.")
(defvar proof-shell-exit-in-progress nil
"A flag indicating that the current proof process is about to exit.
This flag is set for the duration of `proof-shell-kill-function'
to tell hooks in `proof-deactivate-scripting-hook' to refrain
from calling `proof-shell-exit'.")
;;
;; Indicator and fake minor mode for active scripting buffer
;;
(defcustom proof-shell-active-scripting-indicator
'(:eval (propertize
" Scripting " 'face
(cond
(proof-shell-busy 'proof-queue-face)
((eq proof-shell-last-output-kind 'error) 'proof-script-sticky-error-face)
((proof-with-current-buffer-if-exists proof-script-buffer
(proof-locked-region-full-p))
'font-lock-type-face)
(t 'proof-locked-face))))
"Modeline indicator for active scripting buffer.
Changes colour to indicate whether the shell is busy, etc."
:type 'sexp
:group 'proof-general-internals)
(unless
(assq 'proof-active-buffer-fake-minor-mode minor-mode-alist)
(setq minor-mode-alist
(append minor-mode-alist
(list
(list
'proof-active-buffer-fake-minor-mode
proof-shell-active-scripting-indicator)))))
;;
;; Implementing the process lock
;;
;; Note that because Emacs Lisp code is single-threaded, there are no
;; concurrency issues here (a loop parsing process output cannot get
;; pre-empted by the user trying to send more input to the process, or
;; by the process filter trying to deal with more output). But the
;; lock allows for clear management of the queue.
;;
;; Three relevant functions:
;;
;; proof-shell-ready-prover
;; starts proof shell, gives error if it's busy.
;;
;; proof-activate-scripting (in proof-script.el)
;; calls proof-shell-ready-prover, and turns on scripting minor
;; mode for current (scripting) buffer.
;;
;; Also, an enabler predicate:
;;
;; proof-shell-available-p
;; returns non-nil if a proof shell is active and not locked.
;;
;;;###autoload
(defun proof-shell-ready-prover (&optional queuemode)
"Make sure the proof assistant is ready for a command.
If QUEUEMODE is set, succeed if the proof shell is busy but
has mode QUEUEMODE, which is a symbol or list of symbols.
Otherwise, if the shell is busy, give an error.
No change to current buffer or point."
(proof-shell-start)
(unless (or (not proof-shell-busy)
(eq queuemode proof-shell-busy)
(and (listp queuemode)
(member proof-shell-busy queuemode)))
(error "Proof process busy!")))
;;;###autoload
(defsubst proof-shell-live-buffer ()
"Return non-nil if proof-shell-buffer is live."
(and proof-shell-buffer
(buffer-live-p proof-shell-buffer)
(scomint-check-proc proof-shell-buffer)))
;;;###autoload
(defun proof-shell-available-p ()
"Return non-nil if there is a proof shell active and available.
No error messages. Useful as menu or toolbar enabler."
(and (proof-shell-live-buffer)
(not proof-shell-busy)))
(defun proof-grab-lock (&optional queuemode)
"Grab the proof shell lock, starting the proof assistant if need be.
Runs `proof-state-change-hook' to notify state change.
If QUEUEMODE is supplied, set the lock to that value."
(proof-shell-ready-prover queuemode)
(setq proof-shell-interrupt-pending nil
proof-shell-busy (or queuemode t)
proof-shell-last-queuemode proof-shell-busy)
(run-hooks 'proof-state-change-hook))
(defun proof-release-lock ()
"Release the proof shell lock. Clear `proof-shell-busy'."
(setq proof-shell-busy nil))
;;
;; Starting and stopping the proof shell
;;
(defcustom proof-shell-fiddle-frames t
"Non-nil if proof-shell functions should fire-up/delete frames like crazy."
:type 'boolean
:group 'proof-shell)
(defun proof-shell-set-text-representation ()
"Adjust representation for current buffer, to match `proof-shell-unicode'."
(unless proof-shell-unicode
;; Prevent interpretation of multi-byte characters.
;; Otherwise, chars 128-255 get remapped higher, breaking regexps
(toggle-enable-multibyte-characters -1)))
(defun proof-shell-start ()
"Initialise a shell-like buffer for a proof assistant.
Does nothing if proof assistant is already running.
Also generates goal and response buffers.
If `proof-prog-name-ask' is set, query the user for the
process command."
(interactive)
(unless (proof-shell-live-buffer)
(setq proof-included-files-list nil) ; clear some state
(let ((name (buffer-file-name (current-buffer))))
(if (and name proof-prog-name-guess proof-guess-command-line)
(setq proof-prog-name
(apply proof-guess-command-line (list name)))))
(if proof-prog-name-ask
(setq proof-prog-name (read-shell-command "Run process: "
proof-prog-name)))
(let
((proc (downcase proof-assistant)))
;; Starting the inferior process (asynchronous)
(let* ((prog-name-list1
(if (functionp (proof-ass-sym prog-args))
;; complex assistants define <PA>-prog-args as function
;; that computes the argument list.
(cons proof-prog-name (funcall (proof-ass-sym prog-args)))
(if (proof-ass prog-args)
;; Intermediate complex assistants set the value
;; of <PA>-prog-args to the argument list.
(cons proof-prog-name (proof-ass prog-args))
;; Trivial assistants simply set proof-prog-name
(split-string proof-prog-name))))
(prog-name-list
;; Splice in proof-rsh-command if it's non-nil
(if (and proof-rsh-command
(> (length proof-rsh-command) 0))
(append (split-string proof-rsh-command)
prog-name-list1)
prog-name-list1))
(prog-command-line (mapconcat 'identity prog-name-list " "))
(process-connection-type
proof-shell-process-connection-type)
;; Trac #324, Trac #284: default with Emacs 23 variants
;; is t. nil gives marginally better results with "make
;; profile.isar" on homogenous test input. Top-level
;; Emacs loop causes slow down on Mac and Windows ports.
(process-adaptive-read-buffering nil)
;; The next few settings control the proof assistant encoding.
;; See Elisp manual for recommendations for coding systems.
;; Modern versions of proof systems should be Unicode
;; clean, i.e., outputing only ASCII characters or using a
;; representation such as UTF-8. Old versions of PG
;; relied on control sequences using 8-bit characters with
;; codes between 127 and 255, this is now deprecated.
;; Backward compatibility: remove UTF-8 encoding if not
;; wanted; it conflicts with using chars 128-255 for
;; markup and results in blocking in C libraries.
(process-environment
(append (proof-ass prog-env) ; custom environment
(if proof-shell-unicode ; if specials not used,
process-environment ; leave it alone
(cons
(if (getenv "LANG")
(format "LANG=%s"
(replace-regexp-in-string
"\\.UTF-8" ""
(getenv "LANG")))
"LANG=C")
(delete
(concat "LANG=" (getenv "LANG"))
process-environment)))))
(normal-coding-system-for-read coding-system-for-read)
(coding-system-for-read
(if proof-shell-unicode
(or (condition-case nil
(check-coding-system 'utf-8)
(error nil))
normal-coding-system-for-read)
(if (string-match "Linux"
(shell-command-to-string "uname"))
'raw-text
normal-coding-system-for-read)))
(coding-system-for-write coding-system-for-read))
(message "Starting: %s" prog-command-line)
(apply 'scomint-make (append (list proc (car prog-name-list) nil)
(cdr prog-name-list)))
(setq proof-shell-buffer (get-buffer (concat "*" proc "*")))
(unless (proof-shell-live-buffer)
;; Give error now if shell buffer isn't live (process exited)
(setq proof-shell-buffer nil)
(error "Starting process: %s..failed" prog-command-line)))
;; Create the associated buffers and set buffer variables
(let ((goals "*goals*")
(resp "*response*")
(trace "*trace*")
(thms "*thms*"))
(setq proof-goals-buffer (get-buffer-create goals))
(setq proof-response-buffer (get-buffer-create resp))
(if proof-shell-trace-output-regexp
(setq proof-trace-buffer (get-buffer-create trace)))
(if proof-shell-thms-output-regexp
(setq proof-thms-buffer (get-buffer-create thms)))
;; Set the special-display-regexps now we have the buffer names
(setq pg-response-special-display-regexp
(proof-regexp-alt goals resp trace thms)))
(with-current-buffer proof-shell-buffer
;; Clear and set text representation (see CVS history for comments)
(erase-buffer)
(proof-shell-set-text-representation)
;; Initialise associated buffers
(with-current-buffer proof-response-buffer
(erase-buffer)
(proof-shell-set-text-representation)
(funcall proof-mode-for-response))
(with-current-buffer proof-goals-buffer
(erase-buffer)
(proof-shell-set-text-representation)
(funcall proof-mode-for-goals))
(proof-with-current-buffer-if-exists proof-trace-buffer
(erase-buffer)
(proof-shell-set-text-representation)
(funcall proof-mode-for-response)
(setq pg-response-eagerly-raise nil))
;; Initialise shell mode (calls hook function, after process started)
(funcall proof-mode-for-shell)
;; Check to see that the process is still going. If not,
;; switch buffer to display the error messages to the user.
(unless (proof-shell-live-buffer)
(switch-to-buffer proof-shell-buffer)
(error "%s process exited!" proc))
;; Setting modes initialises local variables which
;; may affect frame/buffer appearance: so we fire up frames
;; once this has been done.
(if proof-shell-fiddle-frames
;; Call multiple-frames-enable in case we need to fire up
;; new frames (NB: sets specifiers to remove modeline)
(save-selected-window
(save-selected-frame
(proof-multiple-frames-enable)))))
(message "Starting %s process... done." proc))))
;;
;; Shutting down proof shell and associated buffers
;;
;; Hooks here are handy for liaising with prover config stuff.
(defvar proof-shell-kill-function-hooks nil
"Functions run from `proof-shell-kill-function'.")
(defun proof-shell-kill-function ()
"Function run when a proof-shell buffer is killed.
Try to shut down the proof process nicely and clear locked
regions and state variables. Value for `kill-buffer-hook' in
shell buffer, called by `proof-shell-bail-out' if process exits."
(let* ((alive (scomint-check-proc (current-buffer)))
(proc (get-buffer-process (current-buffer)))
(bufname (buffer-name)))
(message "%s, cleaning up and exiting..." bufname)
(redisplay t)
(when (and alive proc)
(catch 'exited
(setq proof-shell-exit-in-progress t)
(set-process-sentinel
proc
(lambda (p m) (throw 'exited t)))
;; Turn off scripting (ensure buffers completely processed/undone)
(proof-deactivate-scripting-auto)
(proof-shell-wait (proof-ass quit-timeout))
;; Try to shut down politely.
(if proof-shell-quit-cmd
(scomint-send-string proc
(concat proof-shell-quit-cmd "\n"))
(scomint-send-eof))
;; Wait for it to die
(let ((timecount (proof-ass quit-timeout))
(proc (get-buffer-process proof-shell-buffer)))
(while (and (> timecount 0)
(scomint-check-proc proof-shell-buffer))
(accept-process-output proc 1 nil 1)
(decf timecount)))
;; Still there, kill it rudely.
(when (memq (process-status proc) '(open run stop))
(message "%s, cleaning up and exiting...killing process" bufname)
(kill-process proc)))
(set-process-sentinel proc nil))
;; Clear all state
(proof-script-remove-all-spans-and-deactivate)
(proof-shell-clear-state)
(run-hooks 'proof-shell-kill-function-hooks)
;; Remove auxiliary windows, trying to stop proliferation of
;; frames (NB: loses if user has switched buffer in special frame)
(if (and proof-multiple-frames-enable
proof-shell-fiddle-frames)
(proof-delete-other-frames))
;; Kill associated buffer
(let ((proof-shell-buffer nil)) ;; fool kill buffer hooks
(dolist (buf '(proof-goals-buffer proof-response-buffer
proof-trace-buffer))
(when (buffer-live-p (symbol-value buf))
(delete-windows-on (symbol-value buf))
(kill-buffer (symbol-value buf))
(set buf nil))))
(setq proof-shell-exit-in-progress nil)
(message "%s exited." bufname)))
(defun proof-shell-clear-state ()
"Clear internal state of proof shell."
(setq proof-action-list nil
proof-included-files-list nil
proof-shell-busy nil
proof-shell-last-queuemode nil
proof-shell-proof-completed nil
proof-nesting-depth 0
proof-shell-silent nil
proof-shell-last-output ""
proof-shell-last-prompt ""
proof-shell-last-output-kind nil
proof-shell-delayed-output-start nil
proof-shell-delayed-output-end nil
proof-shell-delayed-output-flags nil))
(defun proof-shell-exit (&optional dont-ask)
"Query the user and exit the proof process.
This simply kills the `proof-shell-buffer' relying on the hook function
`proof-shell-kill-function' to do the hard work. If optional
argument DONT-ASK is non-nil, the proof process is terminated
without confirmation.
The kill function uses `<PA>-quit-timeout' as a timeout to wait
after sending `proof-shell-quit-cmd' before rudely killing the process.
This function should not be called if
`proof-shell-exit-in-progress' is t, because a recursive call of
`proof-shell-kill-function' will give strange errors."
(interactive "P")
(if (buffer-live-p proof-shell-buffer)
(when (or dont-ask
(yes-or-no-p (format "Exit %s process? " proof-assistant)))
(let ((kill-buffer-query-functions nil)) ; avoid extra dialog
(kill-buffer proof-shell-buffer))
(setq proof-shell-buffer nil))
(error "No proof shell buffer to kill!")))
(defun proof-shell-bail-out (process event)
"Value for the process sentinel for the proof assistant PROCESS.
If the proof assistant dies, run `proof-shell-kill-function' to
cleanup and remove the associated buffers. The shell buffer is
left around so the user may discover what killed the process.
EVENT is the string describing the change."
(message "Process %s %s, shutting down scripting..." process event)
(proof-shell-kill-function)
(message "Process %s %s, shutting down scripting...done." process event))
(defun proof-shell-restart ()
"Clear script buffers and send `proof-shell-restart-cmd'.
All locked regions are cleared and the active scripting buffer
deactivated.
If the proof shell is busy, an interrupt is sent with
`proof-interrupt-process' and we wait until the process is ready.
The restart command should re-synchronize Proof General with the proof
assistant, without actually exiting and restarting the proof assistant
process.
It is up to the proof assistant how much context is cleared: for
example, theories already loaded may be \"cached\" in some way,
so that loading them the next time round only performs a re-linking
operation, not full re-processing. (One way of caching is via
object files, used by Lego and Coq)."
(interactive)
(when proof-shell-busy
(proof-interrupt-process)
(proof-shell-wait))
(if (not (proof-shell-live-buffer))
(proof-shell-start) ;; start if not running
;; otherwise clear context
(proof-script-remove-all-spans-and-deactivate)
(proof-shell-clear-state)
(with-current-buffer proof-shell-buffer
(delete-region (point-min) (point-max)))
(if (and (buffer-live-p proof-shell-buffer)
proof-shell-restart-cmd)
(proof-shell-invisible-command
proof-shell-restart-cmd))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Response buffer processing
;;
(defvar proof-shell-urgent-message-marker nil
"Marker in proof shell buffer pointing to end of last urgent message.")
(defvar proof-shell-urgent-message-scanner nil
"Marker in proof shell buffer pointing to scan start for urgent messages.
This is only used in `proof-shell-process-urgent-message'.")
(defun proof-shell-handle-error-output (start-regexp append-face)
"Displays output from process in `proof-response-buffer'.
The output is taken from `proof-shell-last-output' and begins
the first match for START-REGEXP.
If START-REGEXP is nil or no match can be found (which can happen
if output has been garbled somehow), begin from the start of
the output for this command.
This is a subroutine of `proof-shell-handle-error'."
(let ((string proof-shell-last-output) pos)
(if (and start-regexp
(setq pos (string-match start-regexp string)))
(setq string (substring string pos)))
;; Erase if need be, and erase next time round too.
(pg-response-maybe-erase t nil)
(pg-response-display-with-face string append-face)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Processing error output
;;
(defun proof-shell-handle-error-or-interrupt (err-or-int flags)
"React on an error or interrupt message triggered by the prover.
The argument ERR-OR-INT should be set to 'error or 'interrupt
which affects the action taken.
For errors, we first flush unprocessed output (usually goals).
The error message is the (usually) displayed in the response buffer.
For interrupts, a warning message is displayed.
In both cases we then sound a beep, clear the queue and spans and
finally we call `proof-shell-handle-error-or-interrupt-hook'.
Commands which are not part of regular script management (with
non-empty flags='no-error-display) will not cause any display action.
This is called in two places: (1) from the output processing
functions, in case we find an error or interrupt message output,
and (2) from the exec loop, in case of a pending interrupt which
didn't cause prover output."
(unless (memq 'no-error-display flags)
(cond
((eq err-or-int 'interrupt)
(pg-response-maybe-erase t t t) ; force cleaned now & next
(proof-shell-handle-error-output
(if proof-shell-truncate-before-error proof-shell-interrupt-regexp)
'proof-error-face)
(pg-response-warning
"Interrupt: script management may be in an inconsistent state
(but it's probably okay)"))
(t ; error
(if proof-shell-delayed-output-start
(save-excursion
(proof-shell-handle-delayed-output)))
(proof-shell-handle-error-output
(if proof-shell-truncate-before-error proof-shell-error-regexp)
'proof-error-face)
(proof-display-and-keep-buffer proof-response-buffer))))
(proof-with-current-buffer-if-exists proof-shell-buffer
(proof-shell-error-or-interrupt-action err-or-int)))
(defun proof-shell-error-or-interrupt-action (err-or-int)
"Take action on errors or interrupts.
ERR-OR-INT is a flag, 'error or 'interrupt.
This is a subroutine of `proof-shell-handle-error-or-interrupt'.
Must be called with proof shell buffer current.
This function invokes `proof-shell-handle-error-or-interrupt-hook'
unless the FLAGS for the command are non-nil (see `proof-action-list')."
(unless proof-shell-quiet-errors
(beep))
(let* ((fatalitem (car-safe proof-action-list))
(badspan (car-safe fatalitem))
(flags (if fatalitem (nth 3 fatalitem))))
(proof-with-current-buffer-if-exists proof-script-buffer
(save-excursion
(proof-script-clear-queue-spans-on-error badspan
(eq err-or-int 'interrupt))))
(setq proof-action-list nil)
(proof-release-lock)
(unless flags
;; Give a hint about C-c C-`. (NB: approximate test)
(if (pg-response-has-error-location)
(pg-next-error-hint))
;; Run hooks for additional effects, e.g. highlight or moving pointer
(run-hooks 'proof-shell-handle-error-or-interrupt-hook))))
(defun proof-goals-pos (span maparg)
"Given a span, return the start of it if corresponds to a goal, nil otherwise."
(and (eq 'goal (car (span-property span 'proof-top-element)))
(span-start span)))
(defun proof-pbp-focus-on-first-goal ()
"If the `proof-goals-buffer' contains goals, bring the first one into view.
This is a hook function for proof-shell-handle-delayed-output-hook."
)
;; PG 4.0 FIXME
; (let
; ((pos (map-extents 'proof-goals-pos proof-goals-buffer
; nil nil nil nil 'proof-top-element)))
; (and pos (set-window-point
; (get-buffer-window proof-goals-buffer t) pos)))))
(defsubst proof-shell-string-match-safe (regexp string)
"Like string-match except returns nil if REGEXP is nil."
(and regexp (string-match regexp string)))
(defun proof-shell-handle-immediate-output (cmd start end flags)
"See if the output between START and END must be dealt with immediately.
To speed up processing, PG tries to avoid displaying output that
the user will not have a chance to see. Some output must be
handled immediately, however: these are errors, interrupts,
goals and loopbacks (proof step hints/proof by pointing results).
In this function we check, in turn:
`proof-shell-interrupt-regexp'
`proof-shell-error-regexp'
`proof-shell-proof-completed-regexp'
`proof-shell-result-start'
Other kinds of output are essentially display only, so only
dealt with if necessary.
To extend this, set `proof-shell-handle-output-system-specific',
which is a hook to take particular additional actions.
This function sets variables: `proof-shell-last-output-kind',
and the counter `proof-shell-proof-completed' which counts commands
after a completed proof."
(setq proof-shell-last-output-kind nil) ; unclassified
(goto-char start)
(cond
;; TODO: Isabelle has changed (since 2009) and is now amalgamating
;; output between prompts, and does e.g.,
;; GOALS
;; ERROR
;; we need to override delayed output from the previous
;; command with delayed output from this command to handle that!
((proof-re-search-forward-safe proof-shell-interrupt-regexp end t)
(setq proof-shell-last-output-kind 'interrupt)
(proof-shell-handle-error-or-interrupt 'interrupt flags))
((proof-re-search-forward-safe proof-shell-error-regexp end t)
(setq proof-shell-last-output-kind 'error)
(proof-shell-handle-error-or-interrupt 'error flags))
((proof-re-search-forward-safe proof-shell-result-start end t)
;; NB: usually the action list is empty, strange results likely if
;; more commands follow. Therefore, this case might be delayed.
(let (pstart pend)
(setq pstart (+ 1 (match-end 0)))
(re-search-forward proof-shell-result-end end t)
(setq pend (- (match-beginning 0) 1))
(proof-shell-insert-loopback-cmd
(buffer-substring-no-properties pstart pend)))
(setq proof-shell-last-output-kind 'loopback)
(proof-shell-exec-loop))
((proof-re-search-forward-safe proof-shell-proof-completed-regexp end t)
(setq proof-shell-proof-completed 0))) ; commands since complete
;; PG4.0 change: simplify and run earlier
(if proof-shell-handle-output-system-specific
(funcall proof-shell-handle-output-system-specific
cmd proof-shell-last-output)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Interrupts
;;
(defun proof-interrupt-process ()
"Interrupt the proof assistant. Warning! This may confuse Proof General.
This sends an interrupt signal to the proof assistant, if Proof General
thinks it is busy.
This command is risky because we don't know whether the last command
succeeded or not. The assumption is that it didn't, which should be true
most of the time, and all of the time if the proof assistant has a careful
handling of interrupt signals.
Some provers may ignore (and lose) interrupt signals, or fail to indicate
that they have been acted upon yet stop in the middle of output.
In the first case, PG will terminate the queue of commands at the first
available point. In the second case, you may need to press enter inside
the prover command buffer (e.g., with Isabelle2009 press RET inside *isabelle*)."
(interactive)
(unless (proof-shell-live-buffer)
(error "Proof process not started!"))
(unless proof-shell-busy
(error "Proof process not active!"))
(setq proof-shell-interrupt-pending t)
(with-current-buffer proof-shell-buffer
(interrupt-process)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Low-level commands for shell communication
;;
;;;###autoload
(defun proof-shell-insert (strings action &optional scriptspan)
"Insert STRINGS at the end of the proof shell, call `scomint-send-input'.
STRINGS is a list of strings (which will be concatenated), or a
single string.
The ACTION argument is a symbol which is typically the name of a
callback for when each string has been processed.
This calls `proof-shell-insert-hook'. The arguments `action' and
`scriptspan' may be examined by the hook to determine how to modify
the `string' variable (exploiting dynamic scoping) which will be
the command actually sent to the shell.
Note that the hook is not called for the empty (null) string
or a carriage return.
We strip the string of carriage returns before inserting it
and updating `proof-marker' to point to the end of the newly
inserted text.
Do not use this function directly, or output will be lost. It is only
used in `proof-add-to-queue' when we start processing a queue, and in
`proof-shell-exec-loop', to process the next item."
(assert (or (stringp strings)
(listp strings))
nil "proof-shell-insert: expected string list argument")
(with-current-buffer proof-shell-buffer
(goto-char (point-max))
;; TEMP: next step: preprocess list of strings directly
(let ((string (if (stringp strings) strings
(apply 'concat strings))))
;; Hook for munging `string' and other dirty hacks.
(run-hooks 'proof-shell-insert-hook)
;; Replace CRs from string with spaces to avoid spurious prompts.
(if proof-shell-strip-crs-from-input
(setq string (subst-char-in-string ?\n ?\ string t)))
(insert string)
;; Advance the proof-marker, if synchronization has been gained.
;; Null marker => no yet synced; output is ignored.
(unless (null (marker-position proof-marker))
(set-marker proof-marker (point)))
(scomint-send-input))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Code for manipulating proof queue
;;
(defun proof-shell-action-list-item (cmd callback &optional flags)
"Return action list entry run CMD with callback CALLBACK and FLAGS.
The queue entry does not refer to a span in the script buffer."
(list nil (list cmd) callback flags))
(defun proof-shell-set-silent (span)
"Callback for `proof-shell-start-silent'.
Very simple function but it's important to give it a name to help
track what happens in the proof queue."
(setq proof-shell-silent t))
(defun proof-shell-start-silent-item ()
"Return proof queue entry for starting silent mode."
(proof-shell-action-list-item
proof-shell-start-silent-cmd
'proof-shell-set-silent))
(defun proof-shell-clear-silent (span)
"Callback for `proof-shell-stop-silent'.
Very simple function but it's important to give it a name to help
track what happens in the proof queue."
(setq proof-shell-silent nil))
(defun proof-shell-stop-silent-item ()
"Return proof queue entry for stopping silent mode."
(proof-shell-action-list-item
proof-shell-stop-silent-cmd
'proof-shell-clear-silent))
(defsubst proof-shell-should-be-silent ()
"Non-nil if we should switch to silent mode based on size of queue."
(if (and proof-shell-start-silent-cmd ; configured
(not proof-full-annotation) ; always noisy
(not proof-tree-external-display) ; no proof-tree display
(not proof-shell-silent)) ; already silent
;; NB: to be more accurate we should only count number
;; of scripting items in the list (not e.g. invisibles).
;; More efficient: keep track of size of queue as modified.
(>= (length proof-action-list) proof-shell-silent-threshold)))
(defsubst proof-shell-insert-action-item (item)
"Insert ITEM from `proof-action-list' into the proof shell."
(proof-shell-insert (nth 1 item) (nth 2 item) (nth 0 item)))
(defsubst proof-shell-slurp-comments ()
"Strip comments at front of `proof-action-list', returning items stripped.
Comments are not sent to the prover."
(let (cbitems nextitem)
(while (and proof-action-list
(not (nth 1 (setq nextitem
(car proof-action-list)))))
(setq cbitems (cons nextitem cbitems))
(setq proof-action-list (cdr proof-action-list)))
(nreverse cbitems)))
(defun proof-add-to-queue (queueitems &optional queuemode)
"Chop off the vacuous prefix of the QUEUEITEMS and queue them.
For each item with a nil command at the head of the list, invoke its
callback and remove it from the list.
Append the result onto `proof-action-list', and if the proof
shell isn't already busy, grab the lock with QUEUEMODE and
start processing the queue.
If the proof shell is busy when this function is called,
then QUEUEMODE must match the mode of the queue currently
being processed."
(when (and queueitems proof-action-list)
;; internal check: correct queuemode in force if busy
;; (should have proof-action-list<>nil -> busy)
(and proof-shell-busy queuemode
(unless (eq proof-shell-busy queuemode)
(proof-debug
"proof-append-alist: wrong queuemode detected for busy shell")
(assert (eq proof-shell-busy queuemode)))))
(let ((nothingthere (null proof-action-list)))
;; Now extend or start the queue.
(setq proof-action-list
(nconc proof-action-list queueitems))
(when nothingthere ; process comments immediately
(let ((cbitems (proof-shell-slurp-comments)))
(mapc 'proof-shell-invoke-callback cbitems)))
(if proof-action-list ;; something to do
(progn
(if (proof-shell-should-be-silent)
;; do this ASAP, either first or just after current command
(setq proof-action-list
(if nothingthere ; the first thing
(cons (proof-shell-start-silent-item)
proof-action-list)
(cons (car proof-action-list) ; after current
(cons (proof-shell-start-silent-item)
(cdr proof-action-list))))))
(when nothingthere ; start sending commands
(proof-grab-lock queuemode)
(setq proof-shell-last-output-kind nil)
(proof-shell-insert-action-item (car proof-action-list))))
;; nothing to do: maybe we completed a list of comments without sending them
(proof-detach-queue))))
;;;###autoload
(defun proof-start-queue (start end queueitems &optional queuemode)
"Begin processing a queue of commands in QUEUEITEMS.
If START is non-nil, START and END are buffer positions in the
active scripting buffer for the queue region.
This function calls `proof-add-to-queue'."
(if start
(proof-set-queue-endpoints start end))
(proof-add-to-queue queueitems queuemode))
;;;###autoload
(defun proof-extend-queue (end queueitems)
"Extend the current queue with QUEUEITEMS, queue end END.
To make sense, the commands should correspond to processing actions
for processing a region from (buffer-queue-or-locked-end) to END.
The queue mode is set to 'advancing"
(proof-set-queue-endpoints (proof-unprocessed-begin) end)
(condition-case err
(run-hooks 'proof-shell-extend-queue-hook)
((error quit)
(proof-detach-queue)
(signal (car err) (cdr err))))
(proof-add-to-queue queueitems 'advancing))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; MAIN LOOP
;;
(defun proof-shell-exec-loop ()
"Main loop processing the `proof-action-list', called from shell filter.
`proof-action-list' contains a list of (SPAN COMMAND ACTION [FLAGS]) lists.
If this function is called with a non-empty `proof-action-list', the
head of the list is the previously executed command which succeeded.
We execute the callback (ACTION SPAN) on the first item,
then (ACTION SPAN) on any following items which have null as
their cmd components.
If a there is a next command after that, send it to the process.
If the action list becomes empty, unlock the process and remove
the queue region.
The return value is non-nil if the action list is now empty or
contains only invisible elements for Prooftree synchronization."
(unless (null proof-action-list)
(save-excursion
(if proof-script-buffer ; switch to active script
(set-buffer proof-script-buffer))
(let* ((item (car proof-action-list))
(flags (nth 3 item))
cbitems)
;; now we should invoke callback on just processed command,
;; but we delay this until sending the next command, attempting
;; to parallelize prover and Emacs somewhat. (PG 4.0 change)
(setq proof-action-list (cdr proof-action-list))
(setq cbitems (cons item
(proof-shell-slurp-comments)))
;; This is the point where old items have been removed from
;; proof-action-list and where the next item has not yet been
;; sent to the proof assistant. This is therefore one of the
;; few points where it is safe to manipulate
;; proof-action-list. The urgent proof-tree display actions
;; must therefore be called here, because they might add some
;; Show actions at the front of proof-action-list.
(if proof-tree-external-display
(proof-tree-urgent-action flags))
;; if action list is (nearly) empty, ensure prover is noisy.
(if (and proof-shell-silent
(not (eq (nth 2 item) 'proof-shell-clear-silent))
(or (null proof-action-list)
(null (cdr proof-action-list))))
;; Insert the quieten command on head of queue
(setq proof-action-list
(cons (proof-shell-stop-silent-item)
proof-action-list)))
;; pending interrupts: we want to stop the queue here
(when proof-shell-interrupt-pending
(mapc 'proof-shell-invoke-callback cbitems)
(setq cbitems nil)
(proof-shell-handle-error-or-interrupt 'interrupt flags))
(if proof-action-list
;; send the next command to the process.
(proof-shell-insert-action-item (car proof-action-list)))
;; process the delayed callbacks now
(mapc 'proof-shell-invoke-callback cbitems)
(unless proof-action-list ; release lock, cleanup
(proof-release-lock)
(proof-detach-queue)
(unless flags ; hint after a batch of scripting
(pg-processing-complete-hint))
(pg-finish-tracing-display))
(or (null proof-action-list)
(every
(lambda (item) (memq 'proof-tree-show-subgoal (nth 3 item)))
proof-action-list))))))
(defun proof-shell-insert-loopback-cmd (cmd)
"Insert command string CMD sent from prover into script buffer.
String is inserted at the end of locked region, after a newline
and indentation. Assumes `proof-script-buffer' is active."
(unless (string-match "^\\s-*$" cmd) ; FIXME: assumes cmd is single line
(with-current-buffer proof-script-buffer
(let (span)
(proof-goto-end-of-locked)
(let ((proof-one-command-per-line t)) ; because pbp several commands
(proof-script-new-command-advance))
(insert cmd)
;; NB: difference between ordinary commands and pbp is that
;; pbp can return *several* commands, that are treated as
;; a unit, i.e. sent to the proof assistant together.
;; FIXME da: this seems very similar to proof-insert-pbp-command
;; in proof-script.el. Should be unified, I suspect.
(setq span (span-make (proof-unprocessed-begin) (point)))
(span-set-property span 'type 'pbp)
(span-set-property span 'cmd cmd)
(proof-set-queue-endpoints (proof-unprocessed-begin) (point))
(setq proof-action-list
(cons (car proof-action-list)
(cons (list span cmd 'proof-done-advancing)
(cdr proof-action-list))))))))
(defun proof-shell-process-urgent-message (start end)
"Analyse urgent message between START and END for various cases.
Cases are: *trace* output, included/retracted files, cleared
goals/response buffer, variable setting, xml-encoded PGIP response,
theorem dependency message or interactive output indicator.
If none of these apply, display the text between START and END.
The text between START and END should be a string that starts with
text matching `proof-shell-eager-annotation-start' and
ends with text matching `proof-shell-eager-annotation-end'."
(goto-char start)
(cond
((proof-looking-at-safe proof-shell-trace-output-regexp)
(proof-shell-process-urgent-message-trace start end))
((proof-looking-at-safe (car-safe proof-shell-process-file))
(let ((file (funcall (cdr proof-shell-process-file))))
(if (and file (not (string= file "")))
(proof-register-possibly-new-processed-file file))))
((proof-looking-at-safe proof-shell-retract-files-regexp)
(proof-shell-process-urgent-message-retract start end))
((proof-looking-at-safe proof-shell-clear-response-regexp)
(pg-response-maybe-erase nil t t))
((proof-looking-at-safe proof-shell-clear-goals-regexp)
(proof-clean-buffer proof-goals-buffer))
((proof-looking-at-safe proof-shell-set-elisp-variable-regexp)
(proof-shell-process-urgent-message-elisp))
((proof-looking-at-safe proof-shell-match-pgip-cmd)
(pg-pgip-process-packet
;; NB: xml-parse-region ignores junk before XML
(xml-parse-region start end)))
((proof-looking-at-safe proof-shell-theorem-dependency-list-regexp)
(proof-shell-process-urgent-message-thmdeps))
((proof-looking-at-safe proof-shell-theorem-dependency-list-regexp)
(proof-shell-process-urgent-message-thmdeps))
((proof-looking-at-safe proof-shell-interactive-prompt-regexp)
(proof-shell-process-interactive-prompt-regexp))
(t
(proof-shell-process-urgent-message-default start end))))
;;
;; urgent message subroutines
;;
(defun proof-shell-process-urgent-message-default (start end)
"A subroutine of `proof-shell-process-urgent-message'."
;; Clear the response buffer this time, but not next, leave window.
(pg-response-maybe-erase nil nil)
(proof-minibuffer-message
(buffer-substring-no-properties
(save-excursion
(re-search-forward proof-shell-eager-annotation-start end nil)
(point))
(min end
(save-excursion (end-of-line) (point))
(+ start 75))))
(pg-response-display-with-face
(proof-shell-strip-eager-annotations start end)
'proof-eager-annotation-face))
(defun proof-shell-process-urgent-message-trace (start end)
"Display a message in the tracing buffer.
A subroutine of `proof-shell-process-urgent-message'."
(proof-trace-buffer-display start end)
(unless (and proof-trace-output-slow-catchup
(pg-tracing-tight-loop))
(proof-display-and-keep-buffer proof-trace-buffer))
;; If user quits during tracing output, send an interrupt
;; to the prover. Helps when Emacs is "choking".
(if (and quit-flag proof-action-list)
(proof-interrupt-process)))
(defun proof-shell-process-urgent-message-retract (start end)
"A subroutine of `proof-shell-process-urgent-message'.
Takes files off `proof-included-files-list' and calls
`proof-restart-buffers' to do the necessary clean-up on those
buffers visting a file that disappears from
`proof-included-files-list'. So in some respect this function is
inverse to `proof-register-possibly-new-processed-file'."
(let ((current-included proof-included-files-list))
(setq proof-included-files-list
(funcall proof-shell-compute-new-files-list))
(let ((scrbuf proof-script-buffer))
;; NB: we assume that no new buffers are *added* by
;; the proof-shell-compute-new-files-list
(proof-restart-buffers
(proof-files-to-buffers
(set-difference current-included
proof-included-files-list)))
(cond
;; Do nothing if there was no active scripting buffer
((not scrbuf))
;; Do nothing if active buffer hasn't changed (may be nuked)
((eq scrbuf proof-script-buffer))
;; Otherwise, active scripting buffer has been retracted.
(t
(setq proof-script-buffer nil))))))
(defun proof-shell-process-urgent-message-elisp ()
"A subroutine of `proof-shell-process-urgent-message'."
(let
((variable (match-string 1))
(expr (match-string 2)))
(condition-case nil
(with-temp-buffer
(insert expr) ; massive risk from malicious provers!!
(set (intern variable) (eval-last-sexp t)))
(t (proof-debug
(concat
"lisp error when obeying proof-shell-set-elisp-variable: \n"
"setting `" variable "'\n to: \n"
expr "\n"))))))
(defun proof-shell-process-urgent-message-thmdeps ()
"A subroutine of `proof-shell-process-urgent-message'."
(let ((names (match-string 1))
(deps (match-string 2))
(sep proof-shell-theorem-dependency-list-split))
(setq
proof-last-theorem-dependencies
(cons (split-string names sep)
(split-string deps sep)))))
(defun proof-shell-process-interactive-prompt-regexp ()
"Action taken when `proof-shell-interactive-prompt-regexp' is observed."
(when (and (proof-shell-live-buffer)
; not already visible
t)
(switch-to-buffer proof-shell-buffer)
(message "Prover expects input in %s buffer" proof-shell-buffer)))
;;
;; urgent message utilities
;;
(defun proof-shell-strip-eager-annotations (start end)
"Strip `proof-shell-eager-annotation-{start,end}' from region."
(goto-char start)
(if (re-search-forward proof-shell-eager-annotation-start end nil)
(setq start (point)))
(if (re-search-forward proof-shell-eager-annotation-end end nil)
(setq end (match-beginning 0)))
(buffer-substring-no-properties start end))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; The proof shell process filter
;;
(defun proof-shell-filter (str)
"Master filter for the proof assistant shell-process.
A function for `scomint-output-filter-functions'.
Deal with output STR and issue new input from the queue. This is
an important internal function.
Handle urgent messages first. As many as possible are processed,
using the function `proof-shell-process-urgent-messages'.
If a prompt is seen, run `proof-shell-filter-manage-output' on the
output between the new prompt and the last input (position of
`proof-marker') or the last urgent message (position of
`proof-shell-urgent-message-marker'), whichever is later. For
example, in this case:
PROMPT> INPUT
OUTPUT-1
URGENT-MESSAGE-1
OUTPUT-2
URGENT-MESSAGE-2
OUTPUT-3
PROMPT>
`proof-marker' points after INPUT.
`proof-shell-urgent-message-marker' points after URGENT-MESSAGE-2,
after both urgent messages have been processed by
`proof-shell-process-urgent-messages'. Urgent messages always
processed; they are intended to correspond to informational
notes that the prover makes to inform the user or interface on
progress.
In this case, the ordinary outputs OUTPUT-1 and OUTPUT-2 are
ignored; only OUTPUT-3 will be processed by
`proof-shell-filter-manage-output'.
Error or interrupt messages are expected to terminate an
interactive output and appear last before a prompt and will
always be processed. Error messages and interrupt messages are
therefore *not* considered as urgent messages.
The first time that a prompt is seen, `proof-marker' is
initialised to the end of the prompt. This should correspond
with initializing the process. After that, `proof-marker'
is only changed when input is sent in `proof-shell-insert'."
(save-excursion
;; Process urgent messages.
(and proof-shell-eager-annotation-start
(proof-shell-process-urgent-messages))
(let ((pos (marker-position proof-marker)))
(if (not pos)
(proof-shell-filter-first-command)
(if proof-action-list
;; We were expecting some output. Wait until output is
;; complete. Only one piece of output is dealt with at a
;; time; we loose sync if there's more than one bit there.
(let ((urgnt (marker-position
proof-shell-urgent-message-marker))
(prev-prompt pos)
(startpos pos)
endpos)
;; Ignore any urgent messages that have already been dealt
;; with. This loses in the case mentioned above. Instead
;; might try to delete/filter out old urgent messages.
(goto-char pos)
(if (and urgnt (< startpos urgnt))
(setq startpos (goto-char urgnt))
;; Otherwise, skip possibly a (fudge) space and new line
(if (eq (char-after startpos) ?\ )
(setq startpos (goto-char (+ 2 startpos)))
(setq startpos (goto-char (1+ startpos)))))
;; Find next prompt.
(if (re-search-forward
proof-shell-annotated-prompt-regexp nil t)
(progn
(setq endpos (match-beginning 0))
(setq proof-shell-last-prompt
(buffer-substring-no-properties
endpos (match-end 0)))
(goto-char (point-max))
;; Process output string.
(proof-shell-filter-manage-output startpos endpos))))
;; Not expecting output, ignore it. Busy flag should be clear.
(if proof-shell-busy
(progn
(proof-debug
"proof-shell-filter found empty action list yet proof shell busy.")
(proof-release-lock))))))))
(defun proof-shell-filter-first-command ()
"Deal with initial output. A subroutine of `proof-shell-filter'.
This initialises `proof-marker': we set marker to after the first
prompt in the output buffer if one can be found now.
The first time a prompt is seen we ignore any output that occurred
before it, assuming that corresponds to uninteresting startup
messages."
(goto-char (point-min))
(if (re-search-forward proof-shell-annotated-prompt-regexp nil t)
(progn
(set-marker proof-marker (point))
(proof-shell-exec-loop))))
(defun proof-shell-process-urgent-messages ()
"Scan the shell buffer for urgent messages.
Scanning starts from `proof-shell-urgent-message-scanner' or
`scomint-last-input-end', which ever is later. We deal with strings
between regexps `proof-shell-eager-annotation-start' and
`proof-shell-eager-annotation-end'.
We update `proof-shell-urgent-message-marker' to point to last message found.
This is a subroutine of `proof-shell-filter'."
(let ((pt (point)) (end t)
lastend laststart
(initstart (max (marker-position proof-shell-urgent-message-scanner)
(marker-position scomint-last-input-end))))
(goto-char initstart)
(while (and end
(re-search-forward proof-shell-eager-annotation-start
nil 'limit))
(setq laststart (match-beginning 0))
(if (setq end
(re-search-forward proof-shell-eager-annotation-end nil t))
(save-excursion
(setq lastend end)
;; Process the region including the annotations
(proof-shell-process-urgent-message laststart lastend))))
(set-marker
proof-shell-urgent-message-scanner
(if end ;; couldn't find message start; move forward to avoid rescanning
(max initstart
(- (point)
(1+ proof-shell-eager-annotation-start-length)))
;; incomplete message; leave marker at start of message
laststart))
;; Set position of last urgent message found
(if lastend
(set-marker proof-shell-urgent-message-marker lastend))
(goto-char pt)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Despatching output
;;
(defun proof-shell-filter-manage-output (start end)
"Subroutine of `proof-shell-filter' for output between START and END.
First, we invoke `proof-shell-handle-immediate-output' which classifies
and handles output that must be dealt with immediately.
Other output (user display) is only displayed when the proof
action list becomes empty, to avoid a confusing rapidly changing
output that slows down processing.
After processing the current output, the last step undertaken
by the filter is to send the next command from the queue."
(let ((span (caar proof-action-list))
(cmd (nth 1 (car proof-action-list)))
(flags (nth 3 (car proof-action-list))))
;; A copy of the last message, verbatim, never modified.
(setq proof-shell-last-output
(buffer-substring-no-properties start end))
;; sets proof-shell-last-output-kind
(proof-shell-handle-immediate-output cmd start end flags)
(unless proof-shell-last-output-kind ; dealt with already
(setq proof-shell-delayed-output-start start)
(setq proof-shell-delayed-output-end end)
(setq proof-shell-delayed-output-flags flags)
(if (proof-shell-exec-loop)
(setq proof-shell-last-output-kind
;; only display result for last output
(proof-shell-handle-delayed-output)))
;; send output to the proof tree visualizer
(if proof-tree-external-display
(proof-tree-handle-delayed-output cmd flags span)))))
(defsubst proof-shell-display-output-as-response (flags str)
"If FLAGS permit, display response STR; set `proof-shell-last-response-output'."
(setq proof-shell-last-response-output str) ; set even if not displayed
(unless (memq 'no-response-display flags)
(pg-response-display str)))
(defun proof-shell-handle-delayed-output ()
"Display delayed goals/responses, when queue is stopped or completed.
This function handles the cases of `proof-shell-output-kind' which
are not dealt with eagerly during script processing, namely
'response and 'goals types.
This is useful even with empty delayed output as it will empty
the buffers.
The delayed output is in the region
\[proof-shell-delayed-output-start,proof-shell-delayed-output-end].
If no goals classified output is found, the whole output is
displayed in the response buffer.
If goals output is found, the last matching instance, possibly
bounded by `proof-shell-end-goals-regexp', will be displayed
in the goals buffer (and may be further analysed by Proof General).
Any output that appears *before* the last goals output (but
after messages classified as urgent, see `proof-shell-filter')
will also be displayed in the response buffer.
For example, if OUTPUT has this form:
MESSSAGE-1
GOALS-1
MESSAGE-2
GOALS-2
JUNK
then GOALS-2 will be displayed in the goals buffer, and MESSAGE-2
in the response buffer. JUNK will be ignored.
Notice that the above alternation (and separation of JUNK) can
only be distinguished if both `proof-shell-start-goals-regexp'
and `proof-shell-end-goals-regexp' are set. With just the start
goals regexp set, GOALS-2 JUNK will appear in the goals buffer
and no response output would occur.
The goals and response outputs are copied into
`proof-shell-last-goals-output' and
`proof-shell-last-response-output' respectively.
The value returned is the value for `proof-shell-last-output-kind',
i.e., 'goals or 'response."
(let ((start proof-shell-delayed-output-start)
(end proof-shell-delayed-output-end)
(flags proof-shell-delayed-output-flags))
(goto-char start)
(cond
((and proof-shell-start-goals-regexp
(proof-re-search-forward proof-shell-start-goals-regexp end t))
(let* ((gmark (match-beginning 0)) ; start of goals message
(gstart (or (match-end 1) ; start of actual display
gmark))
(rstart start) ; possible response before goals
(gend end)
both) ; flag for response+goals
(goto-char gstart)
(while (re-search-forward proof-shell-start-goals-regexp end t)
(setq gmark (match-beginning 0))
(setq gstart (or (match-end 1) gmark))
(setq gend
(if (and proof-shell-end-goals-regexp
(re-search-forward proof-shell-end-goals-regexp end t))
(progn
(setq rstart (match-end 0))
(match-beginning 0))
end)))
(setq proof-shell-last-goals-output
(buffer-substring-no-properties gstart gend))
;; FIXME heuristic: 4 allows for annotation in end-goals-regexp [is it needed?]
(setq both
(> (- gmark rstart) 4))
(if both
(proof-shell-display-output-as-response
flags
(buffer-substring-no-properties rstart gmark)))
;; display goals output second so it persists in 2-pane mode
(unless (memq 'no-goals-display flags)
(pg-goals-display proof-shell-last-goals-output both))
;; indicate a goals output has been given
'goals))
(t
(proof-shell-display-output-as-response flags proof-shell-last-output)
;; indicate that (only) a response output has been given
'response))
(run-hooks 'proof-shell-handle-delayed-output-hook)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Tracing slow down: prevent Emacs-consumes-all-CPU-displaying phenomenon
;;
;; Possible improvement: add user-controlled flag to turn on/off display
(defvar pg-last-tracing-output-time (float-time)
"Time of last tracing output, as recorded by (float-time).")
(defvar pg-last-trace-output-count 0
"Count up to `pg-slow-mode-trigger-count'.")
(defconst pg-slow-mode-trigger-count 20
"Number of fast trace messages before turning on slow mode.")
(defconst pg-slow-mode-duration 3
"Maximum duration of slow mode in seconds.")
(defconst pg-fast-tracing-mode-threshold 500000
"Minimum microsecond delay between tracing outputs that triggers slow mode.")
(defun pg-tracing-tight-loop ()
"Return non-nil in case it seems like prover is dumping a lot of output.
This is a performance hack to avoid Emacs consuming CPU when prover is output
tracing information.
Only works when system timer has microsecond count available."
(let ((tm (float-time))
(dontprint pg-tracing-slow-mode))
(if pg-tracing-slow-mode
(when ;; seconds differs by more than slow mode max duration
(> (- tm pg-last-tracing-output-time) pg-slow-mode-duration)
(setq dontprint nil))
(when ;; time since last tracing output less than threshold
(and
(< (- tm pg-last-tracing-output-time)
(/ pg-fast-tracing-mode-threshold 1000000.0))
(>= (incf pg-last-trace-output-count)
pg-slow-mode-trigger-count))
;; quickly consecutive tracing outputs: go into slow mode
(setq dontprint t)
(pg-slow-fontify-tracing-hint)))
;; return flag for non-printing is new value of slow mode
(setq pg-last-tracing-output-time tm)
(setq pg-tracing-slow-mode dontprint)))
(defun pg-finish-tracing-display ()
(proof-trace-buffer-finish)
(when pg-tracing-slow-mode
(proof-display-and-keep-buffer proof-trace-buffer)
(setq pg-tracing-slow-mode nil))
(setq pg-last-trace-output-count 0))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; proof-shell-invisible-command: for user-level commands.
;;
;;;###autoload
(defun proof-shell-wait (&optional interrupt-on-input timeoutsecs)
"Busy wait for `proof-shell-busy' to become nil, reading from prover.
Needed between sequences of commands to maintain synchronization,
because Proof General does not allow for the action list to be extended
in some cases. Also is considerably faster than leaving the Emacs
top-level command loop to read from the prover.
Called by `proof-shell-invisible-command' and `proof-process-buffer'
when setting `proof-fast-process-buffer' is enabled.
If INTERRUPT-ON-INPUT is non-nil, return if input is received.
If TIMEOUTSECS is a number, time out after that many seconds."
(let* ((proverproc (get-buffer-process proof-shell-buffer))
(accepttime 0.01)
(timecount (if (numberp timeoutsecs)
(/ timeoutsecs accepttime))))
(when proverproc
(while (and proof-shell-busy (not quit-flag)
(if timecount
(> (setq timecount (1- timecount)) 0)
t)
(not (and interrupt-on-input (input-pending-p))))
;; TODO: check below OK on GE 22/23.1. See Trac #324
(accept-process-output proverproc accepttime nil 1))
(redisplay)
(if quit-flag
(error "Proof General: quit in proof-shell-wait")))))
(defun proof-done-invisible (span)
"Callback for proof-shell-invisible-command.
Calls proof-state-change-hook."
(run-hooks 'proof-state-change-hook))
;;;###autoload
(defun proof-shell-invisible-command (cmd &optional wait invisiblecallback
&rest flags)
"Send CMD to the proof process.
The CMD is `invisible' in the sense that it is not recorded in buffer.
CMD may be a string or a string-yielding expression.
Automatically add `proof-terminal-string' if necessary, examining
`proof-shell-no-auto-terminate-commands'.
By default, let the command be processed asynchronously.
But if optional WAIT command is non-nil, wait for processing to finish
before and after sending the command.
In case CMD is (or yields) nil, do nothing.
INVISIBLECALLBACK will be invoked after the command has finished,
if it is set. It should probably run the hook variables
`proof-state-change-hook'.
FLAGS are additional flags to put onto the `proof-action-list'.
The flag 'invisible is always added to FLAGS."
(unless (stringp cmd)
(setq cmd (eval cmd)))
(if cmd
(progn
(unless (or (null proof-terminal-string)
(not proof-shell-auto-terminate-commands)
(string-match (concat
(regexp-quote proof-terminal-string)
"[ \t]*$") cmd))
(setq cmd (concat cmd proof-terminal-string)))
(if wait (proof-shell-wait))
(proof-shell-ready-prover) ; start proof assistant; set vars.
(let* ((callback
(if invisiblecallback
(lexical-let ((icb invisiblecallback))
(lambda (span)
(funcall icb span)))
'proof-done-invisible)))
(proof-start-queue nil nil
(list (proof-shell-action-list-item
cmd
callback
(cons 'invisible flags)))))
(if wait (proof-shell-wait)))))
;;;###autoload
(defun proof-shell-invisible-cmd-get-result (cmd)
"Execute CMD and return result as a string.
This expects CMD to result in some theorem prover output.
Ordinary output (and error handling) is disabled, and the result
\(contents of `proof-shell-last-output') is returned as a string."
(proof-shell-invisible-command cmd 'waitforit
nil
'no-response-display
'no-error-display)
proof-shell-last-output)
;;;###autoload
(defun proof-shell-invisible-command-invisible-result (cmd)
"Execute CMD for side effect in the theorem prover, waiting before and after.
Error messages are displayed as usual."
(proof-shell-invisible-command cmd 'waitforit
nil
'no-response-display))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; User-level functions depending on shell commands
;;
;;
;; Function to insert last prover output in comment.
;; Requested/suggested by Christophe Raffalli
;;
(defun pg-insert-last-output-as-comment ()
"Insert the last output from the proof system as a comment in the proof script."
(interactive)
(if proof-shell-last-output
(let ((beg (point)))
(insert (proof-shell-strip-output-markup proof-shell-last-output))
(comment-region beg (point)))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;
;; Proof General shell mode definition
;;
;(eval-and-compile ; to define vars
;;;###autoload
(define-derived-mode proof-shell-mode scomint-mode
"proof-shell" "Proof General shell mode class for proof assistant processes"
(setq proof-buffer-type 'shell)
(proof-shell-clear-state)
(buffer-disable-undo)
;; scomint customisation.
(setq scomint-output-filter-functions
(append
(if proof-shell-strip-crs-from-output 'scomint-strip-ctrl-m)
(list 'proof-shell-filter)))
(setq proof-marker ; follows prompt
(make-marker)
proof-shell-urgent-message-marker
(make-marker) ; follows urgent messages
proof-shell-urgent-message-scanner
(make-marker)) ; last scan point
(set-marker proof-shell-urgent-message-scanner (point-min))
;; Make cut functions work with proof shell output
(add-hook 'buffer-substring-filters 'proof-shell-strip-output-markup)
;; Note: before entering proof assistant specific code, we could
;; check that process is up and running. If not, could call the
;; sentinel to display the buffer, and give error.
)
;;
;; Sanity checks on important settings
;;
(defconst proof-shell-important-settings
'(proof-shell-annotated-prompt-regexp ; crucial
))
;;;###autoload
(defun proof-shell-config-done ()
"Initialise the specific prover after the child has been configured.
Every derived shell mode should call this function at the end of
processing."
(with-current-buffer proof-shell-buffer
;; Give warnings if some crucial settings haven't been made
(dolist (sym proof-shell-important-settings)
(proof-warn-if-unset "proof-shell-config-done" sym))
;; Set font lock keywords, but turn off by default to save cycles.
(setq font-lock-defaults '(proof-shell-font-lock-keywords))
(set (make-local-variable 'font-lock-global-modes)
(list 'not proof-mode-for-shell))
(let ((proc (get-buffer-process proof-shell-buffer)))
;; Add the kill buffer function and process sentinel
(add-hook 'kill-buffer-hook 'proof-shell-kill-function t t)
(set-process-sentinel proc 'proof-shell-bail-out)
;; Pre-sync initialization command. Necessary for provers which
;; change output modes only after some initializations.
(if proof-shell-pre-sync-init-cmd
(proof-shell-insert proof-shell-pre-sync-init-cmd 'init-cmd))
;; Flush pending output from startup (it gets hidden from the user)
;; while waiting for the prompt to appear
(while (and (memq (process-status proc) '(open run))
(null (marker-position proof-marker)))
(accept-process-output proc 1))
(if (memq (process-status proc) '(open run))
(progn
;; Also ensure that proof-action-list is initialised.
(setq proof-action-list nil)
;; Send main intitialization command and wait for it to be
;; processed.
;; First, configure PGIP preferences (even before init cmd)
;; available: this allows setting them after the init cmd.
(proof-maybe-askprefs)
;; Now send the initialisation commands.
(unwind-protect
(progn
(run-hooks 'proof-shell-init-hook)
(when proof-shell-init-cmd
(if (listp proof-shell-init-cmd)
(mapc 'proof-shell-invisible-command-invisible-result
proof-shell-init-cmd)
(proof-shell-invisible-command-invisible-result
proof-shell-init-cmd)))
(proof-shell-wait)
(if proof-assistant-settings
(mapcar (lambda (c)
(proof-shell-invisible-command c t))
(proof-assistant-settings-cmds))))))))))
(provide 'proof-shell)
;;; proof-shell.el ends here
|