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 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917 1918 1919 1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982 1983 1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035 2036 2037 2038 2039 2040 2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054 2055 2056 2057 2058 2059 2060 2061 2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077 2078 2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114 2115 2116 2117 2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178 2179 2180 2181 2182 2183 2184 2185 2186 2187 2188 2189 2190 2191 2192 2193 2194 2195 2196 2197 2198 2199 2200 2201 2202 2203 2204 2205 2206 2207 2208 2209 2210 2211 2212 2213 2214 2215 2216 2217 2218 2219 2220 2221 2222 2223 2224 2225 2226 2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288 2289 2290 2291 2292 2293 2294 2295 2296 2297 2298 2299 2300 2301 2302 2303 2304 2305 2306 2307 2308 2309 2310 2311 2312 2313 2314 2315 2316 2317 2318 2319 2320 2321 2322 2323 2324 2325 2326 2327 2328 2329 2330 2331 2332 2333 2334 2335 2336 2337 2338 2339 2340 2341 2342 2343 2344 2345 2346 2347 2348 2349 2350 2351 2352 2353 2354 2355 2356 2357 2358 2359 2360 2361 2362 2363 2364 2365 2366 2367 2368 2369 2370 2371 2372 2373 2374 2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392 2393 2394 2395 2396 2397 2398 2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425 2426 2427 2428 2429 2430 2431 2432 2433 2434 2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448 2449 2450 2451 2452 2453 2454 2455 2456 2457 2458 2459 2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478 2479 2480 2481 2482 2483 2484 2485 2486 2487 2488 2489 2490 2491 2492 2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505 2506 2507 2508 2509 2510 2511 2512 2513 2514 2515 2516 2517 2518 2519 2520 2521 2522 2523 2524 2525 2526 2527 2528 2529 2530 2531 2532 2533 2534 2535 2536 2537 2538 2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586 2587 2588 2589 2590 2591 2592 2593 2594 2595 2596 2597 2598 2599 2600 2601 2602 2603 2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625 2626 2627 2628 2629 2630 2631 2632 2633 2634 2635 2636 2637 2638 2639 2640 2641 2642 2643 2644 2645 2646 2647 2648 2649 2650 2651 2652 2653 2654 2655 2656 2657 2658 2659 2660 2661 2662 2663 2664 2665 2666 2667 2668 2669 2670 2671 2672 2673 2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694 2695 2696 2697 2698 2699 2700 2701 2702 2703 2704 2705 2706 2707 2708 2709 2710 2711 2712 2713 2714 2715 2716 2717 2718 2719 2720 2721 2722 2723 2724 2725 2726 2727 2728 2729 2730 2731 2732 2733 2734 2735 2736 2737 2738 2739 2740 2741 2742 2743 2744 2745 2746 2747 2748 2749 2750 2751 2752 2753 2754 2755 2756 2757 2758 2759 2760 2761 2762 2763 2764 2765 2766 2767 2768 2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786 2787 2788 2789 2790 2791 2792 2793 2794 2795 2796 2797 2798 2799 2800 2801 2802 2803 2804 2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831 2832 2833 2834 2835 2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912 2913 2914 2915 2916 2917 2918 2919 2920 2921 2922 2923 2924 2925 2926 2927 2928 2929 2930 2931 2932 2933 2934 2935 2936 2937 2938 2939 2940 2941 2942 2943 2944 2945 2946 2947 2948 2949 2950 2951 2952 2953 2954 2955 2956 2957 2958 2959 2960 2961 2962 2963 2964 2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985 2986 2987 2988 2989 2990 2991 2992 2993 2994 2995 2996 2997 2998 2999 3000 3001 3002 3003 3004 3005 3006 3007 3008 3009 3010 3011 3012 3013 3014 3015 3016 3017 3018 3019 3020 3021 3022 3023 3024 3025 3026 3027 3028 3029 3030 3031 3032 3033 3034 3035 3036 3037 3038 3039 3040 3041 3042 3043 3044 3045 3046 3047 3048 3049 3050 3051 3052 3053 3054 3055 3056 3057 3058 3059 3060 3061 3062 3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091 3092 3093 3094 3095 3096 3097 3098 3099 3100 3101 3102 3103 3104 3105 3106 3107 3108 3109 3110 3111 3112 3113 3114 3115 3116 3117 3118 3119 3120 3121 3122 3123 3124 3125 3126 3127 3128 3129 3130 3131 3132 3133 3134 3135 3136 3137 3138 3139 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 3150 3151 3152 3153 3154 3155 3156 3157 3158 3159 3160 3161 3162 3163 3164 3165 3166 3167 3168 3169 3170 3171 3172 3173 3174 3175 3176 3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202 3203 3204 3205 3206 3207 3208 3209 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241 3242 3243 3244 3245 3246 3247 3248 3249 3250 3251 3252 3253 3254 3255 3256 3257 3258 3259 3260 3261 3262 3263 3264 3265 3266 3267 3268 3269 3270 3271 3272 3273 3274 3275 3276 3277 3278 3279 3280 3281 3282 3283 3284 3285 3286 3287 3288 3289 3290 3291 3292 3293 3294 3295 3296 3297 3298 3299 3300 3301 3302 3303 3304 3305 3306 3307 3308 3309 3310 3311 3312 3313 3314 3315 3316 3317 3318 3319 3320 3321 3322 3323 3324 3325 3326 3327 3328 3329 3330 3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342 3343 3344 3345 3346 3347 3348 3349 3350 3351 3352 3353 3354 3355 3356 3357 3358 3359 3360 3361 3362 3363 3364 3365 3366 3367 3368 3369 3370 3371 3372 3373 3374 3375 3376 3377 3378 3379 3380 3381 3382 3383 3384 3385 3386 3387 3388 3389 3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404 3405 3406 3407 3408 3409 3410 3411 3412 3413 3414 3415 3416 3417 3418 3419 3420 3421 3422 3423 3424 3425 3426 3427 3428 3429 3430 3431 3432 3433 3434 3435 3436 3437 3438 3439 3440 3441 3442 3443 3444 3445 3446 3447 3448 3449 3450 3451 3452 3453 3454 3455 3456 3457 3458 3459 3460 3461 3462 3463 3464 3465 3466 3467 3468 3469 3470 3471 3472 3473 3474 3475 3476 3477 3478 3479 3480 3481 3482 3483 3484 3485 3486 3487 3488 3489 3490 3491 3492 3493 3494 3495 3496 3497 3498 3499 3500 3501 3502 3503 3504 3505 3506 3507 3508 3509 3510 3511 3512 3513 3514 3515 3516 3517 3518 3519 3520 3521 3522 3523 3524 3525 3526 3527 3528 3529 3530 3531 3532 3533 3534 3535 3536 3537 3538 3539 3540 3541 3542 3543 3544 3545 3546 3547 3548 3549 3550 3551 3552 3553 3554 3555 3556 3557 3558 3559 3560 3561 3562 3563 3564 3565 3566 3567 3568 3569 3570 3571 3572 3573 3574 3575 3576 3577 3578 3579 3580 3581 3582 3583 3584 3585 3586 3587 3588 3589 3590 3591 3592 3593 3594 3595 3596 3597 3598 3599 3600 3601 3602 3603 3604 3605 3606 3607 3608 3609 3610 3611 3612 3613 3614 3615 3616 3617 3618 3619 3620 3621 3622 3623 3624 3625 3626 3627 3628 3629 3630 3631 3632 3633 3634 3635 3636 3637 3638 3639 3640 3641 3642 3643 3644 3645 3646 3647 3648 3649 3650 3651 3652 3653 3654 3655 3656 3657 3658 3659 3660 3661 3662 3663 3664 3665 3666 3667 3668 3669 3670 3671 3672 3673 3674 3675 3676 3677 3678 3679 3680 3681 3682 3683 3684 3685 3686 3687 3688 3689 3690 3691 3692 3693 3694 3695 3696 3697 3698 3699 3700 3701 3702 3703 3704 3705 3706 3707 3708 3709 3710 3711 3712 3713 3714 3715 3716 3717 3718 3719 3720 3721 3722 3723 3724 3725 3726 3727 3728 3729 3730 3731 3732 3733 3734 3735 3736 3737 3738 3739 3740 3741 3742 3743 3744 3745 3746 3747 3748 3749 3750 3751 3752 3753 3754 3755 3756 3757 3758 3759 3760 3761 3762 3763 3764 3765 3766 3767 3768 3769 3770 3771 3772 3773 3774 3775 3776 3777 3778 3779 3780 3781 3782 3783 3784 3785 3786 3787 3788 3789 3790 3791 3792 3793 3794 3795 3796 3797 3798 3799 3800 3801 3802 3803 3804 3805 3806 3807 3808 3809 3810 3811 3812 3813 3814 3815 3816 3817 3818 3819 3820 3821 3822 3823 3824 3825 3826 3827 3828 3829 3830 3831 3832 3833 3834 3835 3836 3837 3838 3839 3840 3841 3842 3843 3844 3845 3846 3847 3848 3849 3850 3851 3852 3853 3854 3855 3856 3857 3858 3859 3860 3861 3862 3863 3864 3865 3866 3867 3868 3869 3870 3871 3872 3873 3874 3875 3876 3877 3878 3879 3880 3881 3882 3883 3884 3885 3886 3887 3888 3889 3890 3891 3892 3893 3894 3895 3896 3897 3898 3899 3900 3901 3902 3903 3904 3905 3906 3907 3908 3909 3910 3911 3912 3913 3914 3915 3916 3917 3918 3919 3920 3921 3922 3923 3924 3925 3926 3927 3928 3929 3930 3931 3932 3933 3934 3935 3936 3937 3938 3939 3940 3941 3942 3943 3944 3945 3946 3947 3948 3949 3950 3951 3952 3953 3954 3955 3956 3957 3958 3959 3960 3961 3962 3963 3964 3965 3966 3967 3968 3969 3970 3971 3972 3973 3974 3975 3976 3977 3978 3979 3980 3981 3982 3983 3984 3985 3986 3987 3988 3989 3990 3991 3992 3993 3994 3995 3996 3997 3998 3999 4000 4001 4002 4003 4004 4005 4006 4007 4008 4009 4010 4011 4012 4013 4014 4015 4016 4017 4018 4019 4020 4021 4022 4023 4024 4025 4026 4027 4028 4029 4030 4031 4032 4033 4034 4035 4036 4037 4038 4039 4040 4041 4042 4043 4044 4045 4046 4047 4048 4049 4050 4051 4052 4053 4054 4055 4056 4057 4058 4059 4060 4061 4062 4063 4064 4065 4066 4067 4068 4069 4070 4071 4072 4073 4074 4075 4076 4077 4078 4079 4080 4081 4082 4083 4084 4085 4086 4087 4088 4089 4090 4091 4092 4093 4094 4095 4096 4097 4098 4099 4100 4101 4102 4103 4104 4105 4106 4107 4108 4109 4110 4111 4112 4113 4114 4115 4116 4117 4118 4119 4120 4121 4122 4123 4124 4125 4126 4127 4128 4129 4130 4131 4132 4133 4134 4135 4136 4137 4138 4139 4140 4141 4142 4143 4144 4145 4146 4147 4148 4149 4150 4151 4152 4153 4154 4155 4156 4157 4158 4159 4160 4161 4162 4163 4164 4165 4166 4167 4168 4169 4170 4171 4172 4173 4174 4175 4176 4177 4178 4179 4180 4181 4182 4183 4184 4185 4186 4187 4188 4189 4190 4191 4192 4193 4194 4195 4196 4197 4198 4199 4200 4201 4202 4203 4204 4205 4206 4207 4208 4209 4210 4211 4212 4213 4214 4215 4216 4217 4218 4219 4220 4221 4222 4223 4224 4225 4226 4227 4228 4229 4230 4231 4232 4233 4234 4235 4236 4237 4238 4239 4240 4241 4242 4243 4244 4245 4246 4247 4248 4249 4250 4251 4252 4253 4254 4255 4256 4257 4258 4259 4260 4261 4262 4263 4264 4265 4266 4267 4268 4269 4270 4271 4272 4273 4274 4275 4276 4277 4278 4279 4280 4281 4282 4283 4284 4285 4286 4287 4288 4289 4290 4291 4292 4293 4294 4295 4296 4297 4298 4299 4300 4301 4302 4303 4304 4305 4306 4307 4308 4309 4310 4311 4312 4313 4314 4315 4316 4317 4318 4319 4320 4321 4322 4323 4324 4325 4326 4327 4328 4329 4330 4331 4332 4333 4334 4335 4336 4337 4338 4339 4340 4341 4342 4343 4344 4345 4346 4347 4348 4349 4350 4351 4352 4353 4354 4355 4356 4357 4358 4359 4360 4361 4362 4363 4364 4365 4366 4367 4368 4369 4370 4371 4372 4373 4374 4375 4376 4377 4378 4379 4380 4381 4382 4383 4384 4385 4386 4387 4388 4389 4390 4391 4392 4393 4394 4395 4396 4397 4398 4399 4400 4401 4402 4403 4404 4405 4406 4407 4408 4409 4410 4411 4412 4413 4414 4415 4416 4417 4418 4419 4420 4421 4422 4423 4424 4425 4426 4427 4428 4429 4430 4431 4432 4433 4434 4435 4436 4437 4438 4439 4440 4441 4442 4443 4444 4445 4446 4447 4448 4449 4450 4451 4452 4453 4454 4455 4456 4457 4458 4459 4460 4461 4462 4463 4464 4465 4466 4467 4468 4469 4470 4471 4472 4473 4474 4475 4476 4477 4478 4479 4480 4481 4482 4483 4484 4485 4486 4487 4488 4489 4490 4491 4492 4493 4494 4495 4496 4497 4498 4499 4500 4501 4502 4503 4504 4505 4506 4507 4508 4509 4510 4511 4512 4513 4514 4515 4516 4517 4518 4519 4520 4521 4522 4523 4524 4525 4526 4527 4528 4529 4530 4531 4532 4533 4534 4535 4536 4537 4538 4539 4540 4541 4542 4543 4544 4545 4546 4547 4548 4549 4550 4551 4552 4553 4554 4555 4556 4557 4558 4559 4560 4561 4562 4563 4564 4565 4566 4567 4568 4569 4570 4571 4572 4573 4574 4575 4576 4577 4578 4579 4580 4581 4582 4583 4584 4585 4586 4587 4588 4589 4590 4591 4592 4593 4594 4595 4596 4597 4598 4599 4600 4601 4602 4603 4604 4605 4606 4607 4608 4609 4610 4611 4612 4613 4614 4615 4616 4617 4618 4619 4620 4621 4622 4623 4624 4625 4626 4627 4628 4629 4630 4631 4632 4633 4634 4635 4636 4637 4638 4639 4640 4641 4642 4643 4644 4645 4646 4647 4648 4649 4650 4651 4652 4653 4654 4655 4656 4657 4658 4659 4660 4661 4662 4663 4664 4665 4666 4667 4668 4669 4670 4671 4672 4673 4674 4675 4676 4677 4678 4679 4680 4681 4682 4683 4684 4685 4686 4687 4688 4689 4690 4691 4692 4693 4694 4695 4696 4697 4698 4699 4700 4701 4702 4703 4704 4705 4706 4707 4708 4709 4710 4711 4712 4713 4714 4715 4716 4717 4718 4719 4720 4721 4722 4723 4724 4725 4726 4727 4728 4729 4730 4731 4732 4733 4734 4735 4736 4737 4738 4739 4740 4741 4742 4743 4744 4745 4746 4747 4748 4749 4750 4751 4752 4753 4754 4755 4756 4757 4758 4759 4760 4761 4762 4763 4764 4765 4766 4767 4768 4769 4770 4771 4772 4773 4774 4775 4776 4777 4778 4779 4780 4781 4782 4783 4784 4785 4786 4787 4788 4789 4790 4791 4792 4793 4794 4795 4796 4797 4798 4799 4800 4801 4802 4803 4804 4805 4806 4807 4808 4809 4810 4811 4812 4813 4814 4815 4816 4817 4818 4819 4820 4821 4822 4823 4824 4825 4826 4827 4828 4829 4830 4831 4832 4833 4834 4835 4836 4837 4838 4839 4840 4841 4842 4843 4844 4845 4846 4847 4848 4849 4850 4851 4852 4853 4854 4855 4856 4857 4858 4859 4860 4861 4862 4863 4864 4865 4866 4867 4868 4869 4870 4871 4872 4873 4874 4875 4876 4877 4878 4879 4880 4881 4882 4883 4884 4885 4886 4887 4888 4889 4890 4891 4892 4893 4894 4895 4896 4897 4898 4899 4900 4901 4902 4903 4904 4905 4906 4907 4908 4909 4910 4911 4912 4913 4914 4915 4916 4917 4918 4919 4920 4921 4922 4923 4924 4925 4926 4927 4928 4929 4930 4931 4932 4933 4934 4935 4936 4937 4938 4939 4940 4941 4942 4943 4944 4945 4946 4947 4948 4949 4950 4951 4952 4953 4954 4955 4956 4957 4958 4959 4960 4961 4962 4963 4964 4965 4966 4967 4968 4969 4970 4971 4972 4973 4974 4975 4976 4977 4978 4979 4980 4981 4982 4983 4984 4985 4986 4987 4988 4989 4990 4991 4992 4993 4994 4995 4996 4997 4998 4999 5000 5001 5002 5003 5004 5005 5006 5007 5008 5009 5010 5011 5012 5013 5014 5015 5016 5017 5018 5019 5020 5021 5022 5023 5024 5025 5026 5027 5028 5029 5030 5031 5032 5033 5034 5035 5036 5037 5038 5039 5040 5041 5042 5043 5044 5045 5046 5047 5048 5049 5050 5051 5052 5053 5054 5055 5056 5057 5058 5059 5060 5061 5062 5063 5064 5065 5066 5067 5068 5069 5070 5071 5072 5073 5074 5075 5076 5077 5078 5079 5080 5081 5082 5083 5084 5085 5086 5087 5088 5089 5090 5091 5092 5093 5094 5095 5096 5097 5098 5099 5100 5101 5102 5103 5104 5105 5106 5107 5108 5109 5110 5111 5112 5113 5114 5115 5116 5117 5118 5119 5120 5121 5122 5123 5124 5125 5126 5127 5128 5129 5130 5131 5132 5133 5134 5135 5136 5137 5138 5139 5140 5141 5142 5143 5144 5145 5146 5147 5148 5149 5150 5151 5152 5153 5154 5155 5156 5157 5158 5159 5160 5161 5162 5163 5164 5165 5166 5167 5168 5169 5170 5171 5172 5173 5174 5175 5176 5177 5178 5179 5180 5181 5182 5183 5184 5185 5186 5187 5188 5189 5190 5191 5192 5193 5194 5195 5196 5197 5198 5199 5200 5201 5202 5203 5204 5205 5206 5207 5208 5209 5210 5211 5212 5213 5214 5215 5216 5217 5218 5219 5220 5221 5222 5223 5224 5225 5226 5227 5228 5229 5230 5231 5232 5233 5234 5235 5236 5237 5238 5239 5240 5241 5242 5243 5244 5245 5246 5247 5248 5249 5250 5251 5252 5253 5254 5255 5256 5257 5258 5259 5260 5261 5262 5263 5264 5265 5266 5267 5268 5269 5270 5271 5272 5273 5274 5275 5276 5277 5278 5279 5280 5281 5282 5283 5284 5285 5286 5287 5288 5289 5290 5291 5292 5293 5294 5295 5296 5297 5298 5299 5300 5301 5302 5303 5304 5305 5306 5307 5308 5309 5310 5311 5312 5313 5314 5315 5316 5317 5318 5319 5320 5321 5322 5323 5324 5325 5326 5327 5328 5329 5330 5331 5332 5333 5334 5335 5336 5337 5338 5339 5340 5341 5342 5343 5344 5345 5346 5347 5348 5349 5350 5351 5352 5353 5354 5355 5356 5357 5358 5359 5360 5361 5362 5363 5364 5365 5366 5367 5368 5369 5370 5371 5372 5373 5374 5375 5376 5377 5378 5379 5380 5381 5382 5383 5384 5385 5386 5387 5388 5389 5390 5391 5392 5393 5394 5395 5396 5397 5398 5399 5400 5401 5402 5403 5404 5405 5406 5407 5408 5409 5410 5411 5412 5413 5414 5415 5416 5417 5418 5419 5420 5421 5422 5423 5424 5425 5426 5427 5428 5429 5430 5431 5432 5433 5434 5435 5436 5437 5438 5439 5440 5441 5442 5443 5444 5445 5446 5447 5448 5449 5450 5451 5452 5453 5454 5455 5456 5457 5458 5459 5460 5461 5462 5463 5464 5465 5466 5467 5468 5469 5470 5471 5472 5473 5474 5475 5476 5477 5478 5479 5480 5481 5482 5483 5484 5485 5486 5487 5488 5489 5490 5491 5492 5493 5494 5495 5496 5497 5498 5499 5500 5501 5502 5503 5504 5505 5506 5507 5508 5509 5510 5511 5512 5513 5514 5515 5516 5517 5518 5519 5520 5521 5522 5523 5524 5525 5526 5527 5528 5529 5530 5531 5532 5533 5534 5535 5536 5537 5538 5539 5540 5541 5542 5543 5544 5545 5546 5547 5548 5549 5550 5551 5552 5553 5554 5555 5556 5557 5558 5559 5560 5561 5562 5563 5564 5565 5566 5567 5568 5569 5570 5571 5572 5573 5574 5575 5576 5577 5578 5579 5580 5581 5582 5583 5584 5585 5586 5587 5588 5589 5590 5591 5592 5593 5594 5595 5596 5597 5598 5599 5600 5601 5602 5603 5604 5605 5606 5607 5608 5609 5610 5611 5612 5613 5614 5615 5616 5617 5618 5619 5620 5621 5622 5623 5624 5625 5626 5627 5628 5629 5630 5631 5632 5633 5634 5635 5636 5637 5638 5639 5640 5641 5642 5643 5644 5645 5646 5647 5648 5649 5650 5651 5652 5653 5654 5655 5656 5657 5658 5659 5660 5661 5662 5663 5664 5665 5666 5667 5668 5669 5670 5671 5672 5673 5674 5675 5676 5677 5678 5679 5680 5681 5682 5683 5684 5685 5686 5687 5688 5689 5690 5691 5692 5693 5694 5695 5696 5697 5698 5699 5700 5701 5702 5703 5704 5705 5706 5707 5708 5709 5710 5711 5712 5713 5714 5715 5716 5717 5718 5719 5720 5721 5722 5723 5724 5725 5726 5727 5728 5729 5730 5731 5732 5733 5734 5735 5736 5737 5738 5739 5740 5741 5742 5743 5744 5745 5746 5747 5748 5749 5750 5751 5752 5753 5754 5755 5756 5757 5758 5759 5760 5761 5762 5763 5764 5765 5766 5767 5768 5769 5770 5771 5772 5773 5774 5775 5776 5777 5778 5779 5780 5781 5782 5783 5784 5785 5786 5787 5788 5789 5790 5791 5792 5793 5794 5795 5796 5797 5798 5799 5800 5801 5802 5803 5804 5805 5806 5807 5808 5809 5810 5811 5812 5813 5814 5815 5816 5817 5818 5819 5820 5821 5822 5823 5824 5825 5826 5827 5828 5829 5830 5831 5832 5833 5834 5835 5836 5837 5838 5839 5840 5841 5842 5843 5844 5845 5846 5847 5848 5849 5850 5851 5852 5853 5854 5855 5856 5857 5858 5859 5860 5861 5862 5863 5864 5865 5866 5867 5868 5869 5870 5871 5872 5873 5874 5875 5876 5877 5878 5879 5880 5881 5882 5883 5884 5885 5886 5887 5888 5889 5890 5891 5892 5893 5894 5895 5896 5897 5898 5899 5900 5901 5902 5903 5904 5905 5906 5907 5908 5909 5910 5911 5912 5913 5914 5915 5916 5917 5918 5919 5920 5921 5922 5923 5924 5925 5926 5927 5928 5929 5930 5931 5932 5933 5934 5935 5936 5937 5938 5939 5940 5941 5942 5943 5944 5945 5946 5947 5948 5949 5950 5951 5952 5953 5954 5955 5956 5957 5958 5959 5960 5961 5962 5963 5964 5965 5966 5967 5968 5969 5970 5971 5972 5973 5974 5975 5976 5977 5978 5979 5980 5981 5982 5983 5984 5985 5986 5987 5988 5989 5990 5991 5992 5993 5994 5995 5996 5997 5998 5999 6000 6001 6002 6003 6004 6005 6006 6007 6008 6009 6010 6011 6012 6013 6014 6015 6016 6017 6018 6019 6020 6021 6022 6023 6024 6025 6026 6027 6028 6029 6030 6031 6032 6033 6034 6035 6036 6037 6038 6039 6040 6041 6042 6043 6044 6045 6046 6047 6048 6049 6050 6051 6052 6053 6054 6055 6056 6057 6058 6059 6060 6061 6062 6063 6064 6065 6066 6067 6068 6069 6070 6071 6072 6073 6074 6075 6076 6077 6078 6079 6080 6081 6082 6083 6084 6085 6086 6087 6088 6089 6090 6091 6092 6093 6094 6095 6096 6097 6098 6099 6100 6101 6102 6103 6104 6105 6106 6107 6108 6109 6110 6111 6112 6113 6114 6115 6116 6117 6118 6119 6120 6121 6122 6123 6124 6125 6126 6127 6128 6129 6130 6131 6132 6133 6134 6135 6136 6137 6138 6139 6140 6141 6142 6143 6144 6145 6146 6147 6148 6149 6150 6151 6152 6153 6154 6155 6156 6157 6158 6159 6160 6161 6162 6163 6164 6165 6166 6167 6168 6169 6170 6171 6172 6173 6174 6175 6176 6177 6178 6179 6180 6181 6182 6183 6184 6185 6186 6187 6188 6189 6190 6191 6192 6193 6194 6195 6196 6197 6198 6199 6200 6201 6202 6203 6204 6205 6206 6207 6208 6209 6210 6211 6212 6213 6214 6215 6216 6217 6218 6219 6220 6221 6222 6223 6224 6225 6226 6227 6228 6229 6230 6231 6232 6233 6234 6235 6236 6237 6238 6239 6240 6241 6242 6243 6244 6245 6246 6247 6248 6249 6250 6251 6252 6253 6254 6255 6256 6257 6258 6259 6260 6261 6262 6263 6264 6265 6266 6267 6268 6269 6270 6271 6272 6273 6274 6275 6276 6277 6278 6279 6280 6281 6282 6283 6284 6285 6286 6287 6288 6289 6290 6291 6292 6293 6294 6295 6296 6297 6298 6299 6300 6301 6302 6303 6304 6305 6306 6307 6308 6309 6310 6311 6312 6313 6314 6315 6316 6317 6318 6319 6320 6321 6322 6323 6324 6325 6326 6327 6328 6329 6330 6331 6332 6333 6334 6335 6336 6337 6338 6339 6340 6341 6342 6343 6344 6345 6346 6347 6348 6349 6350 6351 6352 6353 6354 6355 6356 6357 6358 6359 6360 6361 6362 6363 6364 6365 6366 6367 6368 6369 6370 6371 6372 6373 6374 6375 6376 6377 6378 6379 6380 6381 6382 6383 6384 6385 6386 6387 6388 6389 6390 6391 6392 6393 6394 6395 6396 6397 6398 6399 6400 6401 6402 6403 6404 6405 6406 6407 6408 6409 6410 6411 6412 6413 6414 6415 6416 6417 6418 6419 6420 6421 6422 6423 6424 6425 6426 6427 6428 6429 6430 6431 6432 6433 6434 6435 6436 6437 6438 6439 6440 6441 6442 6443 6444 6445 6446 6447 6448 6449 6450 6451 6452 6453 6454 6455 6456 6457 6458 6459 6460 6461 6462 6463 6464 6465 6466 6467 6468 6469 6470 6471 6472 6473 6474 6475 6476 6477 6478 6479 6480 6481 6482 6483 6484 6485 6486 6487 6488 6489 6490 6491 6492 6493 6494 6495 6496 6497 6498 6499 6500 6501 6502 6503 6504 6505 6506 6507 6508 6509 6510 6511 6512 6513 6514 6515 6516 6517 6518 6519 6520 6521 6522 6523 6524 6525 6526 6527 6528 6529 6530 6531 6532 6533 6534 6535 6536 6537 6538 6539 6540 6541 6542 6543 6544 6545 6546 6547 6548 6549 6550 6551 6552 6553 6554 6555 6556 6557 6558 6559 6560 6561 6562 6563 6564 6565 6566 6567 6568 6569 6570 6571 6572 6573 6574 6575 6576 6577 6578 6579 6580 6581 6582 6583 6584 6585 6586 6587 6588 6589 6590 6591 6592 6593 6594 6595 6596 6597 6598 6599 6600 6601 6602 6603 6604 6605 6606 6607 6608 6609 6610 6611 6612 6613 6614 6615 6616 6617 6618 6619 6620 6621 6622 6623 6624 6625 6626 6627 6628 6629 6630 6631 6632 6633 6634 6635 6636 6637 6638 6639 6640 6641 6642 6643 6644 6645 6646 6647 6648 6649 6650 6651 6652 6653 6654 6655 6656 6657 6658 6659 6660 6661 6662 6663 6664 6665 6666 6667 6668 6669 6670 6671 6672 6673 6674 6675 6676 6677 6678 6679 6680 6681 6682 6683 6684 6685 6686 6687 6688 6689 6690 6691 6692 6693 6694 6695 6696 6697 6698 6699 6700 6701 6702 6703 6704 6705 6706 6707 6708 6709 6710 6711 6712 6713 6714 6715 6716 6717 6718 6719 6720 6721 6722 6723 6724 6725 6726 6727 6728 6729 6730 6731 6732 6733 6734 6735 6736 6737 6738 6739 6740 6741 6742 6743 6744 6745 6746 6747 6748 6749 6750 6751 6752 6753 6754 6755 6756 6757 6758 6759 6760 6761 6762 6763 6764 6765 6766 6767 6768 6769 6770 6771 6772 6773 6774 6775 6776 6777 6778 6779 6780 6781 6782 6783 6784 6785 6786 6787 6788 6789 6790 6791 6792 6793 6794 6795 6796 6797 6798 6799 6800 6801 6802 6803 6804 6805 6806 6807 6808 6809 6810 6811 6812 6813 6814 6815 6816 6817 6818 6819 6820 6821 6822 6823 6824 6825 6826 6827 6828 6829 6830 6831 6832 6833 6834 6835 6836 6837 6838 6839 6840 6841 6842 6843 6844 6845 6846 6847 6848 6849 6850 6851 6852 6853 6854 6855 6856 6857 6858 6859 6860 6861 6862 6863 6864 6865 6866 6867 6868 6869 6870 6871 6872 6873 6874 6875 6876 6877 6878 6879 6880 6881 6882 6883 6884 6885 6886 6887 6888 6889 6890 6891 6892 6893 6894 6895 6896 6897 6898 6899 6900 6901 6902 6903 6904 6905 6906 6907 6908 6909 6910 6911 6912 6913 6914 6915 6916 6917 6918 6919 6920 6921 6922 6923 6924 6925 6926 6927 6928 6929 6930 6931 6932 6933 6934 6935 6936 6937 6938 6939 6940 6941 6942 6943 6944 6945 6946 6947 6948 6949 6950 6951 6952 6953 6954 6955 6956 6957 6958 6959 6960 6961 6962 6963 6964 6965 6966 6967 6968 6969 6970 6971 6972 6973 6974 6975 6976 6977 6978 6979 6980 6981 6982 6983 6984 6985 6986 6987 6988 6989 6990 6991 6992 6993 6994 6995 6996 6997 6998 6999 7000 7001 7002 7003 7004 7005 7006 7007 7008 7009 7010 7011 7012 7013 7014 7015 7016 7017 7018 7019 7020 7021 7022 7023 7024 7025 7026 7027 7028 7029 7030 7031 7032 7033 7034 7035 7036 7037 7038 7039 7040 7041 7042 7043 7044 7045 7046 7047 7048 7049 7050 7051 7052 7053 7054 7055 7056 7057 7058 7059 7060 7061 7062 7063 7064 7065 7066 7067 7068 7069 7070 7071 7072 7073 7074 7075 7076 7077 7078 7079 7080 7081 7082 7083 7084 7085 7086 7087 7088 7089 7090 7091 7092 7093 7094 7095 7096 7097 7098 7099 7100 7101 7102 7103 7104 7105 7106 7107 7108 7109 7110 7111 7112 7113 7114 7115 7116 7117 7118 7119 7120 7121 7122 7123 7124 7125 7126 7127 7128 7129 7130 7131 7132 7133 7134 7135 7136 7137 7138 7139 7140 7141 7142 7143 7144 7145 7146 7147 7148 7149 7150 7151 7152 7153 7154 7155 7156 7157 7158 7159 7160 7161 7162 7163 7164 7165 7166 7167 7168 7169 7170 7171 7172 7173 7174 7175 7176 7177 7178 7179 7180 7181 7182 7183 7184 7185 7186 7187 7188 7189 7190 7191 7192 7193 7194 7195 7196 7197 7198 7199 7200 7201 7202 7203 7204 7205 7206 7207 7208 7209 7210 7211 7212 7213 7214 7215 7216 7217 7218 7219 7220 7221 7222 7223 7224 7225 7226 7227 7228 7229 7230 7231 7232 7233 7234 7235 7236 7237 7238 7239 7240 7241 7242 7243 7244 7245 7246 7247 7248 7249 7250 7251 7252 7253 7254 7255 7256 7257 7258 7259 7260 7261 7262 7263 7264 7265 7266 7267 7268 7269 7270 7271 7272 7273 7274 7275 7276 7277 7278 7279 7280 7281 7282 7283 7284 7285 7286 7287 7288 7289 7290 7291 7292 7293 7294 7295 7296 7297 7298 7299 7300 7301 7302 7303 7304 7305 7306 7307 7308 7309 7310 7311 7312 7313 7314 7315 7316 7317 7318 7319 7320 7321 7322 7323 7324 7325 7326 7327 7328 7329 7330 7331 7332 7333 7334 7335 7336 7337 7338 7339 7340 7341 7342 7343 7344 7345 7346 7347 7348 7349 7350 7351 7352 7353 7354 7355 7356 7357 7358 7359 7360 7361 7362 7363 7364 7365 7366 7367 7368 7369 7370 7371 7372 7373 7374 7375 7376 7377 7378 7379 7380 7381 7382 7383 7384 7385 7386 7387 7388 7389 7390 7391 7392 7393 7394 7395 7396 7397 7398 7399 7400 7401 7402 7403 7404 7405 7406 7407 7408 7409 7410 7411 7412 7413 7414 7415 7416 7417 7418 7419 7420 7421 7422 7423 7424 7425 7426 7427 7428 7429 7430 7431 7432 7433 7434 7435 7436 7437 7438 7439 7440 7441 7442 7443 7444 7445 7446 7447 7448 7449 7450 7451 7452 7453 7454 7455 7456 7457 7458 7459 7460 7461 7462 7463 7464 7465 7466 7467 7468 7469 7470 7471 7472 7473 7474 7475 7476 7477 7478 7479 7480 7481 7482 7483 7484 7485 7486 7487 7488 7489 7490 7491 7492 7493 7494 7495 7496 7497 7498 7499 7500 7501 7502 7503 7504 7505 7506 7507 7508 7509 7510 7511 7512 7513 7514 7515 7516 7517 7518 7519 7520 7521 7522 7523 7524 7525 7526 7527 7528 7529 7530 7531 7532 7533 7534 7535 7536 7537 7538 7539 7540 7541 7542 7543 7544 7545 7546 7547 7548 7549 7550 7551 7552 7553 7554 7555 7556 7557 7558 7559 7560 7561 7562 7563 7564 7565 7566 7567 7568 7569 7570 7571 7572 7573 7574 7575 7576 7577 7578 7579 7580 7581 7582 7583 7584 7585 7586 7587 7588 7589 7590 7591 7592 7593 7594 7595 7596 7597 7598 7599 7600 7601 7602 7603 7604 7605 7606 7607 7608 7609 7610 7611 7612 7613 7614 7615 7616 7617 7618 7619 7620 7621 7622 7623 7624 7625 7626 7627 7628 7629 7630 7631 7632 7633 7634 7635 7636 7637 7638 7639 7640 7641 7642 7643 7644 7645 7646 7647 7648 7649 7650 7651 7652 7653 7654 7655 7656 7657 7658 7659 7660 7661 7662 7663 7664 7665 7666 7667 7668 7669 7670 7671 7672 7673 7674 7675 7676 7677 7678 7679 7680 7681 7682 7683 7684 7685 7686 7687 7688 7689 7690 7691 7692 7693 7694 7695 7696 7697 7698 7699 7700 7701 7702 7703 7704 7705 7706 7707 7708 7709 7710 7711 7712 7713 7714 7715 7716 7717 7718 7719 7720 7721 7722 7723 7724 7725 7726 7727 7728 7729 7730 7731 7732 7733 7734 7735 7736 7737 7738 7739 7740 7741 7742 7743 7744 7745 7746 7747 7748 7749 7750 7751 7752 7753 7754 7755 7756 7757 7758 7759 7760 7761 7762 7763 7764 7765 7766 7767 7768 7769 7770 7771 7772 7773 7774 7775 7776 7777 7778 7779 7780 7781 7782 7783 7784 7785 7786 7787 7788 7789 7790 7791 7792 7793 7794 7795 7796 7797 7798 7799 7800 7801 7802 7803 7804 7805 7806 7807 7808 7809 7810 7811 7812 7813 7814 7815 7816 7817 7818 7819 7820 7821 7822 7823 7824 7825 7826 7827 7828 7829 7830 7831 7832 7833 7834 7835 7836 7837 7838 7839 7840 7841 7842 7843 7844 7845 7846 7847 7848 7849 7850 7851 7852 7853 7854 7855 7856 7857 7858 7859 7860 7861 7862 7863 7864 7865 7866 7867 7868 7869 7870 7871 7872 7873 7874 7875 7876 7877 7878 7879 7880 7881 7882 7883 7884 7885 7886 7887 7888 7889 7890 7891 7892 7893 7894 7895 7896 7897 7898 7899 7900 7901 7902 7903 7904 7905 7906 7907 7908 7909 7910 7911 7912 7913 7914 7915 7916 7917 7918 7919 7920 7921 7922 7923 7924 7925 7926 7927 7928 7929 7930 7931 7932 7933 7934 7935 7936 7937 7938 7939 7940 7941 7942 7943 7944 7945 7946 7947 7948 7949 7950 7951 7952 7953 7954 7955 7956 7957 7958 7959 7960 7961 7962 7963 7964 7965 7966 7967 7968 7969 7970 7971 7972 7973 7974 7975 7976 7977 7978 7979 7980 7981 7982 7983 7984 7985 7986 7987 7988 7989 7990 7991 7992 7993 7994 7995 7996 7997 7998 7999 8000 8001 8002 8003 8004 8005 8006 8007 8008 8009 8010 8011 8012 8013 8014 8015 8016 8017 8018 8019 8020 8021 8022 8023 8024 8025 8026 8027 8028 8029 8030 8031 8032 8033 8034 8035 8036 8037 8038 8039 8040 8041 8042 8043 8044 8045 8046 8047 8048 8049 8050 8051 8052 8053 8054 8055 8056 8057 8058 8059 8060 8061 8062 8063 8064 8065 8066 8067 8068 8069 8070 8071 8072 8073 8074 8075 8076 8077 8078 8079 8080 8081 8082 8083 8084 8085 8086 8087 8088 8089 8090 8091 8092 8093 8094 8095 8096 8097 8098 8099 8100 8101 8102 8103 8104 8105 8106 8107 8108 8109 8110 8111 8112 8113 8114 8115 8116 8117 8118 8119 8120 8121 8122 8123 8124 8125 8126 8127 8128 8129 8130 8131 8132 8133 8134 8135 8136 8137 8138 8139 8140 8141 8142 8143 8144 8145 8146 8147 8148 8149 8150 8151 8152 8153 8154 8155 8156 8157 8158 8159 8160 8161 8162 8163 8164 8165 8166 8167 8168 8169 8170 8171 8172 8173 8174 8175 8176 8177 8178 8179 8180 8181 8182 8183 8184 8185 8186 8187 8188 8189 8190 8191 8192 8193 8194 8195 8196 8197 8198 8199 8200 8201 8202 8203 8204 8205 8206 8207 8208 8209 8210 8211 8212 8213 8214 8215 8216 8217 8218 8219 8220 8221 8222 8223 8224 8225 8226 8227 8228 8229 8230 8231 8232 8233 8234 8235 8236 8237 8238 8239 8240 8241 8242 8243 8244 8245 8246 8247 8248 8249 8250 8251 8252 8253 8254 8255 8256 8257 8258 8259 8260 8261 8262 8263 8264 8265 8266 8267 8268 8269 8270 8271 8272 8273 8274 8275 8276 8277 8278 8279 8280 8281 8282 8283 8284 8285 8286 8287 8288 8289 8290 8291 8292 8293 8294 8295 8296 8297 8298 8299 8300 8301 8302 8303 8304 8305 8306 8307 8308 8309 8310 8311 8312 8313 8314 8315 8316 8317 8318 8319 8320 8321 8322 8323 8324 8325 8326 8327 8328 8329 8330 8331 8332 8333 8334 8335 8336 8337 8338 8339 8340 8341 8342 8343 8344 8345 8346 8347 8348 8349 8350 8351 8352 8353 8354 8355 8356 8357 8358 8359 8360 8361 8362 8363 8364 8365 8366 8367 8368 8369 8370 8371 8372 8373 8374 8375 8376 8377 8378 8379 8380 8381 8382 8383 8384 8385 8386 8387 8388 8389 8390 8391 8392 8393 8394 8395 8396 8397 8398 8399 8400 8401 8402 8403 8404 8405 8406 8407 8408 8409 8410 8411 8412 8413 8414 8415 8416 8417 8418 8419 8420 8421 8422 8423 8424 8425 8426 8427 8428 8429 8430 8431 8432 8433 8434 8435 8436 8437 8438 8439 8440 8441 8442 8443 8444 8445 8446 8447 8448 8449 8450 8451 8452 8453 8454 8455 8456 8457 8458 8459 8460 8461 8462 8463 8464 8465 8466 8467 8468 8469 8470 8471 8472 8473 8474 8475 8476 8477 8478 8479 8480 8481 8482 8483 8484 8485 8486 8487 8488 8489 8490 8491 8492 8493 8494 8495 8496 8497 8498 8499 8500 8501 8502 8503 8504 8505 8506 8507 8508 8509 8510 8511 8512 8513 8514 8515 8516 8517 8518 8519 8520 8521 8522 8523 8524 8525 8526 8527 8528 8529 8530 8531 8532 8533 8534 8535 8536 8537 8538 8539 8540 8541 8542 8543 8544 8545 8546 8547 8548 8549 8550 8551 8552 8553 8554 8555 8556 8557 8558 8559 8560 8561 8562 8563 8564 8565 8566 8567 8568 8569 8570 8571 8572 8573 8574 8575 8576 8577 8578 8579 8580 8581 8582 8583 8584 8585 8586 8587 8588 8589 8590 8591 8592 8593 8594 8595 8596 8597 8598 8599 8600 8601 8602 8603 8604 8605 8606 8607 8608 8609 8610 8611 8612 8613 8614 8615 8616 8617 8618 8619 8620 8621 8622 8623 8624 8625 8626 8627 8628 8629 8630 8631 8632 8633 8634 8635 8636 8637 8638 8639 8640 8641 8642 8643 8644 8645 8646 8647 8648 8649 8650 8651 8652 8653 8654 8655 8656 8657 8658 8659 8660 8661 8662 8663 8664 8665 8666 8667 8668 8669 8670 8671 8672 8673 8674 8675 8676 8677 8678 8679 8680 8681 8682 8683 8684 8685 8686 8687 8688 8689 8690 8691 8692 8693 8694 8695 8696 8697 8698 8699 8700 8701 8702 8703 8704 8705 8706 8707 8708 8709 8710 8711 8712 8713 8714 8715 8716 8717 8718 8719 8720 8721 8722 8723 8724 8725 8726 8727 8728 8729 8730 8731 8732 8733 8734 8735 8736 8737 8738 8739 8740 8741 8742 8743 8744 8745 8746 8747 8748 8749 8750 8751 8752 8753 8754 8755 8756 8757 8758 8759 8760 8761 8762 8763 8764 8765 8766 8767 8768 8769 8770 8771 8772 8773 8774 8775 8776 8777 8778 8779 8780 8781 8782 8783 8784 8785 8786 8787 8788 8789 8790 8791 8792 8793 8794 8795 8796 8797 8798 8799 8800 8801 8802 8803 8804 8805 8806 8807 8808 8809 8810 8811 8812 8813 8814 8815 8816 8817 8818 8819 8820 8821 8822 8823 8824 8825 8826 8827 8828 8829 8830 8831 8832 8833 8834 8835 8836 8837 8838 8839 8840 8841 8842 8843 8844 8845 8846 8847 8848 8849 8850 8851 8852 8853 8854 8855 8856 8857 8858 8859 8860 8861 8862 8863 8864 8865 8866 8867 8868 8869 8870 8871 8872 8873 8874 8875 8876 8877 8878 8879 8880 8881 8882 8883 8884 8885 8886 8887 8888 8889 8890 8891 8892 8893 8894 8895 8896 8897 8898 8899 8900 8901 8902 8903 8904 8905 8906 8907 8908 8909 8910 8911 8912 8913 8914 8915 8916 8917 8918 8919 8920 8921 8922 8923 8924 8925 8926 8927 8928 8929 8930 8931 8932 8933 8934 8935 8936 8937 8938 8939 8940 8941 8942 8943 8944 8945 8946 8947 8948 8949 8950 8951 8952 8953 8954 8955 8956 8957 8958 8959 8960 8961 8962 8963 8964 8965 8966 8967 8968 8969 8970 8971 8972 8973 8974 8975 8976 8977 8978 8979 8980 8981 8982 8983 8984 8985 8986 8987 8988 8989 8990 8991 8992 8993 8994 8995 8996 8997 8998 8999 9000 9001 9002 9003 9004 9005 9006 9007 9008 9009 9010 9011 9012 9013 9014 9015 9016 9017 9018 9019 9020 9021 9022 9023 9024 9025 9026 9027 9028 9029 9030 9031 9032 9033 9034 9035 9036 9037 9038 9039 9040 9041 9042 9043 9044 9045 9046 9047 9048 9049 9050 9051 9052 9053 9054 9055 9056 9057 9058 9059 9060 9061 9062 9063 9064 9065 9066 9067 9068 9069 9070 9071 9072 9073 9074 9075 9076 9077 9078 9079 9080 9081 9082 9083 9084 9085 9086 9087 9088 9089 9090 9091 9092 9093 9094 9095 9096 9097 9098 9099 9100 9101 9102 9103 9104 9105 9106 9107 9108 9109 9110 9111 9112 9113 9114 9115 9116 9117 9118 9119 9120 9121 9122 9123 9124 9125 9126 9127 9128 9129 9130 9131 9132 9133 9134 9135 9136 9137 9138 9139 9140 9141 9142 9143 9144 9145 9146 9147 9148 9149 9150 9151 9152 9153 9154 9155 9156 9157 9158 9159 9160 9161 9162 9163 9164 9165 9166 9167 9168 9169 9170 9171 9172 9173 9174 9175 9176 9177 9178 9179 9180 9181 9182 9183 9184 9185 9186 9187 9188 9189 9190 9191 9192 9193 9194 9195 9196 9197 9198 9199 9200 9201 9202 9203 9204 9205 9206 9207 9208 9209 9210 9211 9212 9213 9214 9215 9216 9217 9218 9219 9220 9221 9222 9223 9224 9225 9226 9227 9228 9229 9230 9231 9232 9233 9234 9235 9236 9237 9238 9239 9240 9241 9242 9243 9244 9245 9246 9247 9248 9249 9250 9251 9252 9253 9254 9255 9256 9257 9258 9259 9260 9261 9262 9263 9264 9265 9266 9267 9268 9269 9270 9271 9272 9273 9274 9275 9276 9277 9278 9279 9280 9281 9282 9283 9284 9285 9286 9287 9288 9289 9290 9291 9292 9293 9294 9295 9296 9297 9298 9299 9300 9301 9302 9303 9304 9305 9306 9307 9308 9309 9310 9311 9312 9313 9314 9315 9316 9317 9318 9319 9320 9321 9322 9323 9324 9325 9326 9327 9328 9329 9330 9331 9332 9333 9334 9335 9336 9337 9338 9339 9340 9341 9342 9343 9344 9345 9346 9347 9348 9349 9350 9351 9352 9353 9354 9355 9356 9357 9358 9359 9360 9361 9362 9363 9364 9365 9366 9367 9368 9369 9370 9371 9372 9373 9374 9375 9376 9377 9378 9379 9380 9381 9382 9383 9384 9385 9386 9387 9388 9389 9390 9391 9392 9393 9394 9395 9396 9397 9398 9399 9400 9401 9402 9403 9404 9405 9406 9407 9408 9409 9410 9411 9412 9413 9414 9415 9416 9417 9418 9419 9420 9421 9422 9423 9424 9425 9426 9427 9428 9429 9430 9431 9432 9433 9434 9435 9436 9437 9438 9439 9440 9441 9442 9443 9444 9445 9446 9447 9448 9449 9450 9451 9452 9453 9454 9455 9456 9457 9458 9459 9460 9461 9462 9463 9464 9465 9466 9467 9468 9469 9470 9471 9472 9473 9474 9475 9476 9477 9478 9479 9480 9481 9482 9483 9484 9485 9486 9487 9488 9489 9490 9491 9492 9493 9494 9495 9496 9497 9498 9499 9500 9501 9502 9503 9504 9505 9506 9507 9508 9509 9510 9511 9512 9513 9514 9515 9516 9517 9518 9519 9520 9521 9522 9523 9524 9525 9526 9527 9528 9529 9530 9531 9532 9533 9534 9535 9536 9537 9538 9539 9540 9541 9542 9543 9544 9545 9546 9547 9548 9549 9550 9551 9552 9553 9554 9555 9556 9557 9558 9559 9560 9561 9562 9563 9564 9565 9566 9567 9568 9569 9570 9571 9572 9573 9574 9575 9576 9577 9578 9579 9580 9581 9582 9583 9584 9585 9586 9587 9588 9589 9590 9591 9592 9593 9594 9595 9596 9597 9598 9599 9600 9601 9602 9603 9604 9605 9606 9607 9608 9609 9610 9611 9612 9613 9614 9615 9616 9617 9618 9619 9620 9621 9622 9623 9624 9625 9626 9627 9628 9629 9630 9631 9632 9633 9634 9635 9636 9637 9638 9639 9640 9641 9642 9643 9644 9645 9646 9647 9648 9649 9650 9651 9652 9653 9654 9655 9656 9657 9658 9659 9660 9661 9662 9663 9664 9665 9666 9667 9668 9669 9670 9671 9672 9673 9674 9675 9676 9677 9678 9679 9680 9681 9682 9683 9684 9685 9686 9687 9688 9689 9690 9691 9692 9693 9694 9695 9696 9697 9698 9699 9700 9701 9702 9703 9704 9705 9706 9707 9708 9709 9710 9711 9712 9713 9714 9715 9716 9717 9718 9719 9720 9721 9722 9723 9724 9725 9726 9727 9728 9729 9730 9731 9732 9733 9734 9735 9736 9737 9738 9739 9740 9741 9742 9743 9744 9745 9746 9747 9748 9749 9750 9751 9752 9753 9754 9755 9756 9757 9758 9759 9760 9761 9762 9763 9764 9765 9766 9767 9768 9769 9770 9771 9772 9773 9774 9775 9776 9777 9778 9779 9780 9781 9782 9783 9784 9785 9786 9787 9788 9789 9790 9791 9792 9793 9794 9795 9796 9797 9798 9799 9800 9801 9802 9803 9804 9805 9806 9807 9808 9809 9810 9811 9812 9813 9814 9815 9816 9817 9818 9819 9820 9821 9822 9823 9824 9825 9826 9827 9828 9829 9830 9831 9832 9833 9834 9835 9836 9837 9838 9839 9840 9841 9842 9843 9844 9845 9846 9847 9848 9849 9850 9851 9852 9853 9854 9855 9856 9857 9858 9859 9860 9861 9862 9863 9864 9865 9866 9867 9868 9869 9870 9871 9872 9873 9874 9875 9876 9877 9878 9879 9880 9881 9882 9883 9884 9885 9886 9887 9888 9889 9890 9891 9892 9893 9894 9895 9896 9897 9898 9899 9900 9901 9902 9903 9904 9905 9906 9907 9908 9909 9910 9911 9912 9913 9914 9915 9916 9917 9918 9919 9920 9921 9922 9923 9924 9925 9926 9927 9928 9929 9930 9931 9932 9933 9934 9935 9936 9937 9938 9939 9940 9941 9942 9943 9944 9945 9946 9947 9948 9949 9950 9951 9952 9953 9954 9955 9956 9957 9958 9959 9960 9961 9962 9963 9964 9965 9966 9967 9968 9969 9970 9971 9972 9973 9974 9975 9976 9977 9978 9979 9980 9981 9982 9983 9984 9985 9986 9987 9988 9989 9990 9991 9992 9993 9994 9995 9996 9997 9998 9999 10000 10001 10002 10003 10004 10005 10006 10007 10008 10009 10010 10011 10012 10013 10014 10015 10016 10017 10018 10019 10020 10021 10022 10023 10024 10025 10026 10027 10028 10029 10030 10031 10032 10033 10034 10035 10036 10037 10038 10039 10040 10041 10042 10043 10044 10045 10046 10047 10048 10049 10050 10051 10052 10053 10054 10055 10056 10057 10058 10059 10060 10061 10062 10063 10064 10065 10066 10067 10068 10069 10070 10071 10072 10073 10074 10075 10076 10077 10078 10079 10080 10081 10082 10083 10084 10085 10086 10087 10088 10089 10090 10091 10092 10093 10094 10095 10096 10097 10098 10099 10100 10101 10102 10103 10104 10105 10106 10107 10108 10109 10110 10111 10112 10113 10114 10115 10116 10117 10118 10119 10120 10121 10122 10123 10124 10125 10126 10127 10128 10129 10130 10131 10132 10133 10134 10135 10136 10137 10138 10139 10140 10141 10142 10143 10144 10145 10146 10147 10148 10149 10150 10151 10152 10153 10154 10155 10156 10157 10158 10159 10160 10161 10162 10163 10164 10165 10166 10167 10168 10169 10170 10171 10172 10173 10174 10175 10176 10177 10178 10179 10180 10181 10182 10183 10184 10185 10186 10187 10188 10189 10190 10191 10192 10193 10194 10195 10196 10197 10198 10199 10200 10201 10202 10203 10204 10205 10206 10207 10208 10209 10210 10211 10212 10213 10214 10215 10216 10217 10218 10219 10220 10221 10222 10223 10224 10225 10226 10227 10228 10229 10230 10231 10232 10233 10234 10235 10236 10237 10238 10239 10240 10241 10242 10243 10244 10245 10246 10247 10248 10249 10250 10251 10252 10253 10254 10255 10256 10257 10258 10259 10260 10261 10262 10263 10264 10265 10266 10267 10268 10269 10270 10271 10272 10273 10274 10275 10276 10277 10278 10279 10280 10281 10282 10283 10284 10285 10286 10287 10288 10289 10290 10291 10292 10293 10294 10295 10296 10297 10298 10299 10300 10301 10302 10303 10304 10305 10306 10307 10308 10309 10310 10311 10312 10313 10314 10315 10316 10317 10318 10319 10320 10321 10322 10323 10324 10325 10326 10327 10328 10329 10330 10331 10332 10333 10334 10335 10336 10337 10338 10339 10340 10341 10342 10343 10344 10345 10346 10347 10348 10349 10350 10351 10352 10353 10354 10355 10356 10357 10358 10359 10360 10361 10362 10363 10364 10365 10366 10367 10368 10369 10370 10371 10372 10373 10374 10375 10376 10377 10378 10379 10380 10381 10382 10383 10384 10385 10386 10387 10388 10389 10390 10391 10392 10393 10394 10395 10396 10397 10398 10399 10400 10401 10402 10403 10404 10405 10406 10407 10408 10409 10410 10411 10412 10413 10414 10415 10416 10417 10418 10419 10420 10421 10422 10423 10424 10425 10426 10427 10428 10429 10430 10431 10432 10433 10434 10435 10436 10437 10438 10439 10440 10441 10442 10443 10444 10445 10446 10447 10448 10449 10450 10451 10452 10453 10454 10455 10456 10457 10458 10459 10460 10461 10462 10463 10464 10465 10466 10467 10468 10469 10470 10471 10472 10473 10474 10475 10476 10477 10478 10479 10480 10481 10482 10483 10484 10485 10486 10487 10488 10489 10490 10491 10492 10493 10494 10495 10496 10497 10498 10499 10500 10501 10502 10503 10504 10505 10506 10507 10508 10509 10510 10511 10512 10513 10514 10515 10516 10517 10518 10519 10520 10521 10522 10523 10524 10525 10526 10527 10528 10529 10530 10531 10532 10533 10534 10535 10536 10537 10538 10539 10540 10541 10542 10543 10544 10545 10546 10547 10548 10549 10550 10551 10552 10553 10554 10555 10556 10557 10558 10559 10560 10561 10562 10563 10564 10565 10566 10567 10568 10569 10570 10571 10572 10573 10574 10575 10576 10577 10578 10579 10580 10581 10582 10583 10584 10585 10586 10587 10588 10589 10590 10591 10592 10593 10594 10595 10596 10597 10598 10599 10600 10601 10602 10603 10604 10605 10606 10607 10608 10609 10610 10611 10612 10613 10614 10615 10616 10617 10618 10619 10620 10621 10622 10623 10624 10625 10626 10627 10628 10629 10630 10631 10632 10633 10634 10635 10636 10637 10638 10639 10640 10641 10642 10643 10644 10645 10646 10647 10648 10649 10650 10651 10652 10653 10654 10655 10656 10657 10658 10659 10660 10661 10662 10663 10664 10665 10666 10667 10668 10669 10670 10671 10672 10673 10674 10675 10676 10677 10678 10679 10680 10681 10682 10683 10684 10685 10686 10687 10688 10689 10690 10691 10692 10693 10694 10695 10696 10697 10698 10699 10700 10701 10702 10703 10704 10705 10706 10707 10708 10709 10710 10711 10712 10713 10714 10715 10716 10717 10718 10719 10720 10721 10722 10723 10724 10725 10726 10727 10728 10729 10730 10731 10732 10733 10734 10735 10736 10737 10738 10739 10740 10741 10742 10743 10744 10745 10746 10747 10748 10749 10750 10751 10752 10753 10754 10755 10756 10757 10758 10759 10760 10761 10762 10763 10764 10765 10766 10767 10768 10769 10770 10771 10772 10773 10774 10775 10776 10777 10778 10779 10780 10781 10782 10783 10784 10785 10786 10787 10788 10789 10790 10791 10792 10793 10794 10795 10796 10797 10798 10799 10800 10801 10802 10803 10804 10805 10806 10807 10808 10809 10810 10811 10812 10813 10814 10815 10816 10817 10818 10819 10820 10821 10822 10823 10824 10825 10826 10827 10828 10829 10830 10831 10832 10833 10834 10835 10836 10837 10838 10839 10840 10841 10842 10843 10844 10845 10846 10847 10848 10849 10850 10851 10852 10853 10854 10855 10856 10857 10858 10859 10860 10861 10862 10863 10864 10865 10866 10867 10868 10869 10870 10871 10872 10873 10874 10875 10876 10877 10878 10879 10880 10881 10882 10883 10884 10885 10886 10887 10888 10889 10890 10891 10892 10893 10894 10895 10896 10897 10898 10899 10900 10901 10902 10903 10904 10905 10906 10907 10908 10909 10910 10911 10912 10913 10914 10915 10916 10917 10918 10919 10920 10921 10922 10923 10924 10925 10926 10927 10928 10929 10930 10931 10932 10933 10934 10935 10936 10937 10938 10939 10940 10941 10942 10943 10944 10945 10946 10947 10948 10949 10950 10951 10952 10953 10954 10955 10956 10957 10958 10959 10960 10961 10962 10963 10964 10965 10966 10967 10968 10969 10970 10971 10972 10973 10974 10975 10976 10977 10978 10979 10980 10981 10982 10983 10984 10985 10986 10987 10988 10989 10990 10991 10992 10993 10994 10995 10996 10997 10998 10999 11000 11001 11002 11003 11004 11005 11006 11007 11008 11009 11010 11011 11012 11013 11014 11015 11016 11017 11018 11019 11020 11021 11022 11023 11024 11025 11026 11027 11028 11029 11030 11031 11032 11033 11034 11035 11036 11037 11038 11039 11040 11041 11042 11043 11044 11045 11046 11047 11048 11049 11050 11051 11052 11053 11054 11055 11056 11057 11058 11059 11060 11061 11062 11063 11064 11065 11066 11067 11068 11069 11070 11071 11072 11073 11074 11075 11076 11077 11078 11079 11080 11081 11082 11083 11084 11085 11086 11087 11088 11089 11090 11091 11092 11093 11094 11095 11096 11097 11098 11099 11100 11101 11102 11103 11104 11105 11106 11107 11108 11109 11110 11111 11112 11113 11114 11115 11116 11117 11118 11119 11120 11121 11122 11123 11124 11125 11126 11127 11128 11129 11130 11131 11132 11133 11134 11135 11136 11137 11138 11139 11140 11141 11142 11143 11144 11145 11146 11147 11148 11149 11150 11151 11152 11153 11154 11155 11156 11157 11158 11159 11160 11161 11162 11163 11164 11165 11166 11167 11168 11169 11170 11171 11172 11173 11174 11175 11176 11177 11178 11179 11180 11181 11182 11183 11184 11185 11186 11187 11188 11189 11190 11191 11192 11193 11194 11195 11196 11197 11198 11199 11200 11201 11202 11203 11204 11205 11206 11207 11208 11209 11210 11211 11212 11213 11214 11215 11216 11217 11218 11219 11220 11221 11222 11223 11224 11225 11226 11227 11228 11229 11230 11231 11232 11233 11234 11235 11236 11237 11238 11239 11240 11241 11242 11243 11244 11245 11246 11247 11248 11249 11250 11251 11252 11253 11254 11255 11256 11257 11258 11259 11260 11261 11262 11263 11264 11265 11266 11267 11268 11269 11270 11271 11272 11273 11274 11275 11276 11277 11278 11279 11280 11281 11282 11283 11284 11285 11286 11287 11288 11289 11290 11291 11292 11293 11294 11295 11296 11297 11298 11299 11300 11301 11302 11303 11304 11305 11306 11307 11308 11309 11310 11311 11312 11313 11314 11315 11316 11317 11318 11319 11320 11321 11322 11323 11324 11325 11326 11327 11328 11329 11330 11331 11332 11333 11334 11335 11336 11337 11338 11339 11340 11341 11342 11343 11344 11345 11346 11347 11348 11349 11350 11351 11352 11353 11354 11355 11356 11357 11358 11359 11360 11361 11362 11363 11364 11365 11366 11367 11368 11369 11370 11371 11372 11373 11374 11375 11376 11377 11378 11379 11380 11381 11382 11383 11384 11385 11386 11387 11388 11389 11390 11391 11392 11393 11394 11395 11396 11397 11398 11399 11400 11401 11402 11403 11404 11405 11406 11407 11408 11409 11410 11411 11412 11413 11414 11415 11416 11417 11418 11419 11420 11421 11422 11423 11424 11425 11426 11427 11428 11429 11430 11431 11432 11433 11434 11435 11436 11437 11438 11439 11440 11441 11442 11443 11444 11445 11446 11447 11448 11449 11450 11451 11452 11453 11454 11455 11456 11457 11458 11459 11460 11461 11462 11463 11464 11465 11466 11467 11468 11469 11470 11471 11472 11473 11474 11475 11476 11477 11478 11479 11480 11481 11482 11483 11484 11485 11486 11487 11488 11489 11490 11491 11492 11493 11494 11495 11496 11497 11498 11499 11500 11501 11502 11503 11504 11505 11506 11507 11508 11509 11510 11511 11512 11513 11514 11515 11516 11517 11518 11519 11520 11521 11522 11523 11524 11525 11526 11527 11528 11529 11530 11531 11532 11533 11534 11535 11536 11537 11538 11539 11540 11541 11542 11543 11544 11545 11546 11547 11548 11549 11550 11551 11552 11553 11554 11555 11556 11557 11558 11559 11560 11561 11562 11563 11564 11565 11566 11567 11568 11569 11570 11571 11572 11573 11574 11575 11576 11577 11578 11579 11580 11581 11582 11583 11584 11585 11586 11587 11588 11589 11590 11591 11592 11593 11594 11595 11596 11597 11598 11599 11600 11601 11602 11603 11604 11605 11606 11607 11608 11609 11610 11611 11612 11613 11614 11615 11616 11617 11618 11619 11620 11621 11622 11623 11624 11625 11626 11627 11628 11629 11630 11631 11632 11633 11634 11635 11636 11637 11638 11639 11640 11641 11642 11643 11644 11645 11646 11647 11648 11649 11650 11651 11652 11653 11654 11655 11656 11657 11658 11659 11660 11661 11662 11663 11664 11665 11666 11667 11668 11669 11670 11671 11672 11673 11674 11675 11676 11677 11678 11679 11680 11681 11682 11683 11684 11685 11686 11687 11688 11689 11690 11691 11692 11693 11694 11695 11696 11697 11698 11699 11700 11701 11702 11703 11704 11705 11706 11707 11708 11709 11710 11711 11712 11713 11714 11715 11716 11717 11718 11719 11720 11721 11722 11723 11724 11725 11726 11727 11728 11729 11730 11731 11732 11733 11734 11735 11736 11737 11738 11739 11740 11741 11742 11743 11744 11745 11746 11747 11748 11749 11750 11751 11752 11753 11754 11755 11756 11757 11758 11759 11760 11761 11762 11763 11764 11765 11766 11767 11768 11769 11770 11771 11772 11773 11774 11775 11776 11777 11778 11779 11780 11781 11782 11783 11784 11785 11786 11787 11788 11789 11790 11791 11792 11793 11794 11795 11796 11797 11798 11799 11800 11801 11802 11803 11804 11805 11806 11807 11808 11809 11810 11811 11812 11813 11814 11815 11816 11817 11818 11819 11820 11821 11822 11823 11824 11825 11826 11827 11828 11829 11830 11831 11832 11833 11834 11835 11836 11837 11838 11839 11840 11841 11842 11843 11844 11845 11846 11847 11848 11849 11850 11851 11852 11853 11854 11855 11856 11857 11858 11859 11860 11861 11862 11863 11864 11865 11866 11867 11868 11869 11870 11871 11872 11873 11874 11875 11876 11877 11878 11879 11880 11881 11882 11883 11884 11885 11886 11887 11888 11889 11890 11891 11892 11893 11894 11895 11896 11897 11898 11899 11900 11901 11902 11903 11904 11905 11906 11907 11908 11909 11910 11911 11912 11913 11914 11915 11916 11917 11918 11919 11920 11921 11922 11923 11924 11925 11926 11927 11928 11929 11930 11931 11932 11933 11934 11935 11936 11937 11938 11939 11940 11941 11942 11943 11944 11945 11946 11947 11948 11949 11950 11951 11952 11953 11954 11955 11956 11957 11958 11959 11960 11961 11962 11963 11964 11965 11966 11967 11968 11969 11970 11971 11972 11973 11974 11975 11976 11977 11978 11979 11980 11981 11982 11983 11984 11985 11986 11987 11988 11989 11990 11991 11992 11993 11994 11995 11996 11997 11998 11999 12000 12001 12002 12003 12004 12005 12006 12007 12008 12009 12010 12011 12012 12013 12014 12015 12016 12017 12018 12019 12020 12021 12022 12023 12024 12025 12026 12027 12028 12029 12030 12031 12032 12033 12034 12035 12036 12037 12038 12039 12040 12041 12042 12043 12044 12045 12046 12047 12048 12049 12050 12051 12052 12053 12054 12055 12056 12057 12058 12059 12060 12061 12062 12063 12064 12065 12066 12067 12068 12069 12070 12071 12072 12073 12074 12075 12076 12077 12078 12079 12080 12081 12082 12083 12084 12085 12086 12087 12088 12089 12090 12091 12092 12093 12094 12095 12096 12097 12098 12099 12100 12101 12102 12103 12104 12105 12106 12107 12108 12109 12110 12111 12112 12113 12114 12115 12116 12117 12118 12119 12120 12121 12122 12123 12124 12125 12126 12127 12128 12129 12130 12131 12132 12133 12134 12135 12136 12137 12138 12139 12140 12141 12142 12143 12144 12145 12146 12147 12148 12149 12150 12151 12152 12153 12154 12155 12156 12157 12158 12159 12160 12161 12162 12163 12164 12165 12166 12167 12168 12169 12170 12171 12172 12173 12174 12175 12176 12177 12178 12179 12180 12181 12182 12183 12184 12185 12186 12187 12188 12189 12190 12191 12192 12193 12194 12195 12196 12197 12198 12199 12200 12201 12202 12203 12204 12205 12206 12207 12208 12209 12210 12211 12212 12213 12214 12215 12216 12217 12218 12219 12220 12221 12222 12223 12224 12225 12226 12227 12228 12229 12230 12231 12232 12233 12234 12235 12236 12237 12238 12239 12240 12241 12242 12243 12244 12245 12246 12247 12248 12249 12250 12251 12252 12253 12254 12255 12256 12257 12258 12259 12260 12261 12262 12263 12264 12265 12266 12267 12268 12269 12270 12271 12272 12273 12274 12275 12276 12277 12278 12279 12280 12281 12282 12283 12284 12285 12286 12287 12288 12289 12290 12291 12292 12293 12294 12295 12296 12297 12298 12299 12300 12301 12302 12303 12304 12305 12306 12307 12308 12309 12310 12311 12312 12313 12314 12315 12316 12317 12318 12319 12320 12321 12322 12323 12324 12325 12326 12327 12328 12329 12330 12331 12332 12333 12334 12335 12336 12337 12338 12339 12340 12341 12342 12343 12344 12345 12346 12347 12348 12349 12350 12351 12352 12353 12354 12355 12356 12357 12358 12359 12360 12361 12362 12363 12364 12365 12366 12367 12368 12369 12370 12371 12372 12373 12374 12375 12376 12377 12378 12379 12380 12381 12382 12383 12384 12385 12386 12387 12388 12389 12390 12391 12392 12393 12394 12395 12396 12397 12398 12399 12400 12401 12402 12403 12404 12405 12406 12407 12408 12409 12410 12411 12412 12413 12414 12415 12416 12417 12418 12419 12420 12421 12422 12423 12424 12425 12426 12427 12428 12429 12430 12431 12432 12433 12434 12435 12436 12437 12438 12439 12440 12441 12442 12443 12444 12445 12446 12447 12448 12449 12450 12451 12452 12453 12454 12455 12456 12457 12458 12459 12460 12461 12462 12463 12464 12465 12466 12467 12468 12469 12470 12471 12472 12473 12474 12475 12476 12477 12478 12479 12480 12481 12482 12483 12484 12485 12486 12487 12488 12489 12490 12491 12492 12493 12494 12495 12496 12497 12498 12499 12500 12501 12502 12503 12504 12505 12506 12507 12508 12509 12510 12511 12512 12513 12514 12515 12516 12517 12518 12519 12520 12521 12522 12523 12524 12525 12526 12527 12528 12529 12530 12531 12532 12533 12534 12535 12536 12537 12538 12539 12540 12541 12542 12543 12544 12545 12546 12547 12548 12549 12550 12551 12552 12553 12554 12555 12556 12557 12558 12559 12560 12561 12562 12563 12564 12565 12566 12567 12568 12569 12570 12571 12572 12573 12574 12575 12576 12577 12578 12579 12580 12581 12582 12583 12584 12585 12586 12587 12588 12589 12590 12591 12592 12593 12594 12595 12596 12597 12598 12599 12600 12601 12602 12603 12604 12605 12606 12607 12608 12609 12610 12611 12612 12613 12614 12615 12616 12617 12618 12619 12620 12621 12622 12623 12624 12625 12626 12627 12628 12629 12630 12631 12632 12633 12634 12635 12636 12637 12638 12639 12640 12641 12642 12643 12644 12645 12646 12647 12648 12649 12650 12651 12652 12653 12654 12655 12656 12657 12658 12659 12660 12661 12662 12663 12664 12665 12666 12667 12668 12669 12670 12671 12672 12673 12674 12675 12676 12677 12678 12679 12680 12681 12682 12683 12684 12685 12686 12687 12688 12689 12690 12691 12692 12693 12694 12695 12696 12697 12698 12699 12700 12701 12702 12703 12704 12705 12706 12707 12708 12709 12710 12711 12712 12713 12714 12715 12716 12717 12718 12719
|
; ACL2 Version 8.6 -- A Computational Logic for Applicative Common Lisp
; Copyright (C) 2025, Regents of the University of Texas
; This version of ACL2 is a descendent of ACL2 Version 1.9, Copyright
; (C) 1997 Computational Logic, Inc. See the documentation topic NOTE-2-0.
; This program is free software; you can redistribute it and/or modify
; it under the terms of the LICENSE file distributed with ACL2.
; This program is distributed in the hope that it will be useful,
; but WITHOUT ANY WARRANTY; without even the implied warranty of
; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
; LICENSE for more details.
; Written by: Matt Kaufmann and J Strother Moore
; email: Kaufmann@cs.utexas.edu and Moore@cs.utexas.edu
; Department of Computer Science
; University of Texas at Austin
; Austin, TX 78712 U.S.A.
(in-package "ACL2")
; Essay on Cert-data
; In February 2016, Jared Davis requested on behalf of Centaur Technology
; Inc. that we avoid recomputing runic type-prescriptions when including
; certified books. There were two problems: type-prescription rules from an
; earlier included book were sometimes causing horrendous slowdowns in those
; computations while including a later book; and, we can lose nice
; type-prescriptions that were inferred during the proof (first) pass of
; certify-book using local rules, where a similar problem can occur with
; encapsulate.
; In March 2016 we solved these problems by introducing "cert-data" structures,
; which can be stored in certificates or in the state global variable
; 'cert-data. In January 2020 we added cert-data structures for storing the
; results of translation, specifically, of translating bodies of defun(s) and
; defthm events. The new record structure was introduced as
; translate-cert-data-record; see its defrec form for comments. Thus,
; cert-data stores information on runic type-prescriptions as well as results
; from translating defun and defthm bodies. Some day we might save other
; information that could be expensive to recompute.
; The rest of this Essay is in three parts. First, we introduce some general
; notions. Second, we focus on cert-data structures for computing runic type
; prescriptions. Finally, we discuss cert-data structures for avoiding
; re-translation at include-book time, at least for definitions and theorems.
; Part 1: General Notions
; One of the fields in a .cert file is a :CERT-DATA field, which is an alist
; mapping keys to fast-alists. As of this writing there are two keys:
; :TYPE-PRESCRIPTION and :TRANSLATE. Each of the two fast-alists (for the two
; keys) is called a "cert-data entry". See for example cert-data-fal, which
; takes a :cert-data field value from a certificate and creates an alist
; mapping each of the two keys to a cert-data entry (which is a fast-alist).
; (cert-data-fal is normally the identity function, but if the .cert file is
; written without the serialize writer then cert-data-fal serves to create a
; fast-alist to associate with each of those keys.)
; Each of the two entries is a fast-alist whose keys are symbols. In the case
; of a :type-prescription entry, each key is a function symbol whose associated
; value is a type-prescription record. In the case of a :translate entry, each
; key is an event name (currently, a function symbol or the name of a defthm
; event) that is associated with a list of translate-cert-data-record records.
; Note added 7/2022: We have observed that for a book's portcullis commands,
; :CERT-DATA is included for type-prescriptions but not for translations. This
; distinction may well have been unintentional, but it seems harmless. We see
; no reason why the stored type prescriptions for portcullis commands are
; problematic, and since we do not generally expect many defun or defthm events
; among the portcullis commands (other than those evaluated within include-book
; commands), there is little to be lost by ignoring saved translation
; information for these. This distinction in what is stored as cert-data for
; portcullis commands is a byproduct of our process: translation cert-data is
; determined while processing events under certify-book, while
; type-prescription cert-data is determined at the conclusion of such
; processing; see cert-data-for-certificate. To see this distinction, let
; foo.lisp be any trivial book and execute the following commands.
; (defun f1 (x) x) (local (defun f2 (x) x)) (defun g (x) x)
; (certify-book "foo" ?)
; (read-file "foo.cert" state)
; We see that the :CERT-DATA in foo.cert has :TYPE-PRESCRIPTION entries for f1
; and f3, but has no :TRANSLATE entries.
; Part 2: Type Prescriptions
; The following processes support the effective re-use of runic type
; prescriptions.
; (1) When including a certified book, all functions defined at the top level
; (i.e., not inside sub-books) get their runic type-prescriptions (if any)
; from the .cert file -- more specifically, from the value of state global
; 'cert-data, which is bound to the value from the .cert file.
; (2) Runic type-prescriptions are saved after pass1 of certify-book and
; encapsulate. For each :logic-mode defun processed in the second pass at
; the top level (not in an included sub-book), an "intersection" is taken
; of the runic type-prescription from the first pass and the computed runic
; type-prescription computed in the usual way. Note that since a function
; can be defined locally in a locally included book during the first pass
; but in a later defun at the top level in the second pass (which was
; redundant in the first pass), we even save some runic type-prescriptions
; from functions introduced during the first pass that were not introduced
; at the top level.
; Clearly each such rune is a logical consequence of the first pass; hence,
; by conservativity, it is a logical consequence of the second pass. Note
; that we may need to recompute the :corollary, which otherwise might not
; be a term of the theory produced by the second pass.
; (3) The runic type-prescription written to a .cert file is the one that
; exists after the second pass. We save such a rule for every function
; that could be introduced when including the book, which includes the
; top-level portcullis functions and every function introduced by the book
; (even those not processed during pass 2). Function
; newly-defined-top-level-fns provides this information; happily, even
; before we added support for cert-data, that function was already called
; under certify-book-fn to compute a list of function symbols to pass to
; write-expansion-file.
; Now we elaborate on these processes.
; Note that including an already-certified book gives you the specific type
; prescription from the .cert file. If however a book B1 includes another book
; B2 only locally, then the type-prescription computed for B1 might be stronger
; than that from B2, because of our "intersection" of rules in (2) above.
; Suppose we are to determine the runic type-prescription for a given defun.
; If state global 'cert-data has value nil, then we just do the usual iterative
; calculation. Otherwise we are to use that cert-data; but how do we know that
; we should do the "intersection" operation because we are in the pass 2 case,
; described in (2) above? We bind key :pass1-saved to t in the value of state
; global 'cert-data during the second pass (meaning, cert-data is from pass 1).
; If necessary we could have non-nil values that provide more information than
; t, for example, 'acl2::certify-book or 'acl2::encapsulate. We don't bother
; to bind :pass1-saved to nil for other than pass 2; we simply don't bind
; :pass1-saved.
; In (2) above we describe the saving of runic type-prescriptions to use during
; the second pass. In the case of certify-book, however, we do this only for
; the part of the world not in the retraction after pass 1, since there is no
; need to save information for defuns already processed before starting pass 2.
; Why do we need the "intersection" operation described in (2) above? That is,
; why not just pick either the runic type-prescription from the first pass or
; compute one for the second pass? The answer is that either may be weaker
; than desired, as illustrated by the following two examples. (These examples
; are for encapsulate, but certify-book has the same issue.)
; (2a) First, here is an example showing that the second pass can do a better
; job computing the runic type-prescription than the first pass. (To see the
; empty 'type-prescriptions we would get for FOO on the first pass, change
; (encapsulate () ...) to (progn ...).)
; (encapsulate
; ()
; (local (include-book "rtl/rel9/support/support/lnot" :dir :system))
; ; Because of the following in-theory event, the cert-data from the first pass
; ; associates no runic type-prescription with FOO.
; (local (in-theory nil))
; (defun fl (x)
; (declare (xargs :guard (real/rationalp x)))
; (floor x 1))
; (defun bits (x i j)
; (declare (xargs :guard (and (natp x) (natp i) (natp j))
; :verify-guards nil))
; (mbe :logic (if (or (not (integerp i))
; (not (integerp j)))
; 0
; (fl (/ (mod x (expt 2 (1+ i))) (expt 2 j))))
; :exec (if (< i j)
; 0
; (logand (ash x (- j))
; (1- (ash 1 (1+ (- i j))))))))
; (defun lnot (x n)
; (declare (xargs :guard (and (natp x) (integerp n) (< 0 n))
; :verify-guards nil))
; (if (natp n)
; (+ -1 (expt 2 n) (- (bits x (1- n) 0)))
; 0))
; (defun foo (x n)
; (lnot x n)))
; ; Using cert-data from the first pass only:
; ; (assert-event (null (getpropc 'foo 'type-prescriptions)))
; ; Computing the runic type-prescription in the second pass:
; (assert-event
; (equal (decode-type-set
; (access type-prescription
; (car (last (getpropc 'foo 'type-prescriptions)))
; :basic-ts))
; '*ts-rational*)
; What we actually get, now, is *ts-integer*, because that's the :basic-ts for
; the runic type-prescription of lnot (and also of binary-logand, bits, and fl)
; from the locally included book, saved from the first pass.
; (2b) To obtain an example showing that the first pass can do a better job
; computing the runic type-prescription than the second pass, simply omit
; (in-theory nil) from the example above. The second pass still gives us
; *ts-rational, as above; but if we change (encapsulate () ...) to (progn ...),
; we see:
; (assert-event
; (equal (decode-type-set
; (access type-prescription
; (car (last (getpropc 'foo 'type-prescriptions)))
; :basic-ts))
; '*ts-non-negative-integer*))
; Let's turn now to our handling of a few thorny issues.
; Suppose we are certifying a book with an encapsulate that locally defines a
; function, f, such that later in the book is a different, non-local definition
; of f. At the end of pass 1 of certify-book we will store a runic type
; prescription for the second definition. Then during pass 2 of certify-book,
; might we make a mistake by associating that type-prescription with the first
; (local) definition of f? No, because local definitions are skipped during
; pass 2. But for robustness, we pass a value for cert-data to
; process-embedded-events, which binds state global cert-data to that value.
; Thus, we override the global cert-data when we process an encapsulate.
; The expansion phase of make-event introduces definitions that are in effect
; local. As in the preceding paragraph, we need to avoid applying the global
; cert-data to such definitions. This problem is solved primarily by arranging
; that protected-eval, via protect-system-state-globals, binds state global
; cert-data to nil. An additional binding of cert-data to nil is made before
; calling make-event-fn2-lst, which handles :OR forms and thus can throw away
; earlier values. In such a case, the runic type-prescription will be
; recomputed during include-book, but we think that such an exception is
; tolerable.
; We are careful to not use cert-data for non-trivial encapsulates. It might
; well be possible to do so correctly, but we would need to be very careful to
; track constraints properly; it seems easy to have a soundness bug due to
; recording insufficient constraints in pass 2 to justify deductions made from
; stronger constraints in pass 1.
; A runic type-prescription rule may contain symbols not present in the
; certification world, as shown below. A similar issue applies to the
; :expansion-alist field of the certificate, which is addressed by finding the
; problematic package names with pkg-names introducing hidden defpkg forms. We
; use that same solution for cert-data. The following example shows how this
; works. First, let's look at a couple of books; notice the package definition
; that is to be made in the certification world of the first book.
; $ cat sub.lisp
; ; (defpkg "FOO" nil)
; ; (certify-book "sub" 1)
;
; (in-package "ACL2")
;
; (defun f2 (x) x)
; $ cat top.lisp
; (in-package "ACL2")
;
; (include-book "sub")
;
; (defmacro my-def ()
; `(defun f (,(intern$ "X" "FOO")) ,(intern$ "X" "FOO")))
;
; (my-def)
; $
; After certification of both books, the :CERT-DATA field of top.cert has the
; following value.
; ((:TYPE-PRESCRIPTION (F 0 (3413 F FOO::X)
; (NIL)
; ((FOO::X) :TYPE-PRESCRIPTION F)
; EQUAL (F FOO::X)
; FOO::X)))
; Thus, a hidden defpkg is generated (here we abbreviate the directory as
; <dir>).
; :BEGIN-PORTCULLIS-CMDS
; (DEFPKG "FOO" NIL NIL
; ("<dir>/sub.lisp")
; T)
; :END-PORTCULLIS-CMDS
; Part 3: Translate
; During the first pass of certify-book, an alist is built up as the value of
; the world global, 'translate-cert-data. (See update-translate-cert-data.)
; This alist will ultimately be the value stored in the :translate entry of the
; :cert-data field of the book's certificate, after making it into a
; fast-alist. When including a book, that field will be the value of the
; cert-data given to process-embedded-events in include-book-fn1.
; That fast-alist will be stored in state global 'cert-data during
; include-book. In order for it to be valid (when consulted via the calls of
; get-translate-cert-data-record), we use function store-cert-data to decide
; when to store into the world global, 'translate-cert-data. We must skip
; results of translation computed on behalf of local events, which will of
; course be irrelevant (or even misleading) when later including the book. We
; also need to skip translations computed during make-event expansion, but that
; happens automatically because we are recording the results in a world global
; and the world is reverted after make-event expansion. We also take other
; measures in store-cert-data. In particular, we sometimes avoid storing
; results computed during the first pass of an encapsulate (which is skipped
; during include-book), though that is not necessary since the world is rolled
; back after that pass -- and note that we don't want to use pass1 encapsulate
; results to avoid translating in pass2, because that would avoid local
; incompatibility checking. (For the same reason, we don't retrieve :translate
; cert-data during the include-book phase of certify-book.) Also, we avoid
; worrying about lambda objects. See store-cert-data for details.
; Remarks (in no particular order).
; (1) We considered explicitly avoiding storing translation results from inside
; progn!, simply because of the unbounded flexibility of progn!. But we view
; it unlikely that progn! would cause a problem, since the only truly obvious
; way to build a confusing cert-data entry (alist) for the :translate key seems
; to be to use redefinition, in which case we write an empty :translate entry
; into the certificate (see cert-data-for-certificate); and besides, progn!
; requires a trust tag, so we are sort of off the hook as far as weird end
; cases are concerned.
; (2) One can certify books in community books directory
; books/system/tests/cert-data/ and look at their certificates with (read-file
; "<bookname>.cert" state), to see the :cert-data field for each book. For
; example, the function f1 in top1.lisp and top1a.lisp is redundant with a
; function in an earlier-included book, so its translated body is not stored in
; the corresponding .cert file; but including top1a.lisp does use the
; translated body when including the sub-book, sub1.lisp.
; (3) We are careful not to retrieve translated terms during make-event
; expansion or other uses of protect-system-state-globals, by binding state
; global cert-data to nil in protect-system-state-globals.
; (4) There could be a very few cases where the saved :translate cert-data
; makes include-book a bit more permissive than it would be otherwise,
; especially if trust tags are involved, though we have tried to minimize this.
; An example is that for flet, translate disallows binding a symbol that is
; also bound in the return-last-table. But since translation was OK at
; certify-book time, we don't mind using the same translation at include-book
; time. Another case is when functions called during macroexpansion (within
; translation) are untouchable when including a book on top of a non-trivial
; world; we don't check for that, though we do check for untouchables in the
; translated terms before retrieval. Another case: We don't re-run calls of
; translate-and-test.
; (5) We have considered saving more translation results. But so far,
; profiling has suggested, using (include-book "centaur/sv/top" :dir :system),
; that the only non-trivial benefit might be in saving results of translating
; event forms. This would presumably involve honsing the (untranslated)
; events, as keys; and we haven't thought through what additional work might be
; necessary (for example, we ignored the case (eq stobjs-out :stobjs-out) in
; translate11 because that shouldn't apply for defun or defthm). Profiling
; suggests that we already lose significant potential speed-up simply by having
; to read larger .cert files, and that problem would be even greater if keys
; are events instead of symbols. (Or maybe we can use the cadr of some event
; forms as symbols?... lots to think about.) Another potential issue is added
; complexity from how trans-eval handles IF lazily, rather than translating the
; entire form at once. So our initial work on :translate cert-data in January,
; 2020, is only for translation of defun(s) and defthm bodies.
; (6) Because of our calls of make-fast-alist in the definition of the function
; cert-data-fal, it is unnecessary to store cert-data entries as fast-alists
; in the .cert file. But profiling has suggested that it is harmless to do so.
; The type-prescription cert-data is already a fast-alists anyhow; we'd have to
; free it if we want to avoid storing it as a fast-alist.
; (7) As a sort of optimization, saving space in .cert files at the expense of
; time: we could perhaps loosen the cons-count-bounded restriction in
; store-cert-data by finding all constant symbols' values in the untranslated
; input and doing a sublis to re-insert those symbols in the translated code,
; and then use sublis in the other direction upon retrieval.
; (8) Note that certain tests for ttags in translate11, such as (assoc-eq fn
; *ttag-fns*), aren't a problem. That's because if a trust tag is active at a
; given non-local event during book certification, then it's still active at
; include-book time, since that could only be defeated by inclusion of a
; sub-book, but ttag settings in a sub-book are local to that sub-book.
(defun cert-data-pair (fn cert-data-entry)
; Cert-data-entry is (cdr (assoc-eq key cert-data)) for some key, e.g., for the
; key, :type-prescription.
(and cert-data-entry ; optimization
(hons-get fn cert-data-entry)))
(defun cert-data-val (fn cert-data-entry)
; Cert-data-entry is (cdr (assoc-eq key cert-data)) for some key, e.g., for the
; key, :type-prescription.
(let ((pair (and cert-data-entry ; optimization
(hons-get fn cert-data-entry))))
(cdr pair)))
(defun cert-data-entry-pair (key state)
; Key is :type-prescription or any other keyword that can be associated in a
; cert-data structure with an entry.
(let ((cert-data (f-get-global 'cert-data state)))
(and cert-data ; optimization
(assoc-eq key cert-data))))
(defun cert-data-entry (key state)
; Key is :type-prescription, :defthm, or any other keyword that can be
; associated in a cert-data structure with an entry. We return a valid
; cert-data entry or nil. Note that cert-data entries are not valid in a local
; or make-event context. See the Essay on Cert-data.
(let ((cert-data (f-get-global 'cert-data state)))
(and cert-data ; optimization
(not (f-get-global 'in-local-flg state))
(int= (f-get-global 'make-event-debug-depth state) 0)
(cdr (assoc-eq key cert-data)))))
(mutual-recursion
(defun contains-lambda-objectp (x)
; This function returns true when the input contains a quoted lambda.
(declare (xargs :guard (pseudo-termp x)))
(cond ((atom x) nil)
((eq (car x) 'quote)
(let ((u (unquote x)))
(and (consp u)
(eq (car u) 'lambda))))
(t (or (contains-lambda-object-listp (cdr x))
(and (flambda-applicationp x)
(contains-lambda-objectp (lambda-body (car x))))))))
(defun contains-lambda-object-listp (x)
(declare (xargs :guard (pseudo-term-listp x)))
(cond ((endp x) nil)
(t (or (contains-lambda-objectp (car x))
(contains-lambda-object-listp (cdr x))))))
)
(defun store-cert-data (listp val wrld state)
; Listp is true when val is a list of terms, false when val is a term.
(and (f-get-global 'certify-book-info state)
(not (f-get-global 'in-local-flg state))
(not ; not inside include-book
(global-val 'include-book-path wrld))
; The next conjunct is optional, as explained in the Essay on Cert-data.
(not ; not "obviously" in encapsulate pass1
(and (in-encapsulatep (global-val 'embedded-event-lst wrld) nil)
(not (eq (ld-skip-proofsp state) 'include-book))))
; The following check may be needlessly conservative. It addresses the
; following concern: translation of quoted lambdas is complicated by
; considerations involving apply$ and loop$. For example, untouchable function
; symbols inside such a quoted object might be a concern -- though probably
; not, since we don't seem to allow function symbols to be both badged and
; untouchable. Rather than think through such issues, we simply skip all
; quoted lambdas. If the need arises, one can look into removing this
; restriction.
(not (if listp
(contains-lambda-object-listp val)
(contains-lambda-objectp val)))
; The following heuristic check could be reconsidered. It is intended to keep
; .cert files from being too big.
(< (cons-count-bounded val)
(fn-count-evg-max-val))))
(defrec translate-cert-data-record
; Warning: Keep the fields in sync with update-translate-cert-data and
; cert-data-for-certificate.
; The form of inputs and value depends on type.
; - For translate-bodies: inputs is the names argument to translate-bodies
; and value is (cons tbodies bindings), where tbodies is the list of
; translated bodies and bindings is the corresponding bindings from
; translate.
; - For defthm: inputs is the name
; and value is the translated body (without making an adjustment for
; ACL2(r)).
((type . inputs) . (value . (fns . vars)))
t)
(defun update-translate-cert-data-fn (name installed-wrld wrld type inputs
value fns vars)
(let* ((old-translate-cert-data (global-val 'translate-cert-data
installed-wrld))
(new (make translate-cert-data-record
:type type
:inputs inputs
:value value
:fns fns
:vars vars))
(old-lst (cdr (assoc-eq name old-translate-cert-data))))
(global-set 'translate-cert-data
(acons name
(cons new old-lst)
old-translate-cert-data)
wrld)))
(defmacro update-translate-cert-data (name installed-wrld wrld
&key type inputs value fns vars)
; Warning: Keep the fields in sync with translate-cert-data-record and
; cert-data-for-certificate.
`(update-translate-cert-data-fn ,name ,installed-wrld ,wrld
,type ,inputs ,value ,fns ,vars))
; Rockwell Addition: A major change is the provision of non-executable
; functions. These are typically functions that use stobjs but which
; are translated as though they were theorems rather than definitions.
; This is convenient (necessary?) for specifying some stobj
; properties. These functions will have executable-counterparts that
; just throw. These functions will be marked with the property
; non-executablep.
(defconst *mutual-recursion-ctx-string*
"( MUTUAL-RECURSION ( DEFUN ~x0 ...) ...)")
(defun translate-bodies1 (non-executablep names bodies bindings arglists
stobjs-in-lst ctx wrld state-vars)
; Non-executablep should be t or nil, to indicate whether or not the bodies are
; to be translated for execution. In the case of a function introduced by
; defproxy, non-executablep will be nil.
(cond ((null bodies) (trans-value nil))
(t (mv-let
(erp x bindings2)
(translate1-cmp+ (car bodies)
(if non-executablep t (car names))
(if non-executablep nil bindings)
(collect-non-*-df (car stobjs-in-lst))
(collect-by-position '(:df)
(car stobjs-in-lst)
(car arglists))
(if (and (consp ctx)
(equal (car ctx)
*mutual-recursion-ctx-string*))
(msg "( MUTUAL-RECURSION ... ( DEFUN ~x0 ...) ~
...)"
(car names))
ctx)
wrld state-vars)
(cond
((and erp
(eq bindings2 :UNKNOWN-BINDINGS))
; We try translating in some other order. This attempt isn't complete; for
; example, the following succeeds, but it fails if we switch the first two
; definitions. But it's cheap and better than nothing; without it, the
; unswitched version would fail, too. If this becomes an issue, consider the
; potentially quadratic algorithm of first finding one definition that
; translates successfully, then another, and so on, until all have been
; translated.
; (set-state-ok t)
; (set-bogus-mutual-recursion-ok t)
; (program)
; (mutual-recursion
; (defun f1 (state)
; (let ((state (f-put-global 'last-m 1 state)))
; (f2 state)))
; (defun f2 (state)
; (let ((state (f-put-global 'last-m 1 state)))
; (f3 state)))
; (defun f3 (state)
; state))
(trans-er-let*
((y (translate-bodies1 non-executablep
(cdr names)
(cdr bodies)
bindings
(cdr arglists)
(cdr stobjs-in-lst)
ctx wrld state-vars))
(x (translate1-cmp+ (car bodies)
(if non-executablep t (car names))
(if non-executablep nil bindings)
(collect-non-*-df (car stobjs-in-lst))
(collect-by-position '(:df)
(car stobjs-in-lst)
(car arglists))
(if (and (consp ctx)
(equal (car ctx)
*mutual-recursion-ctx-string*))
(msg "( MUTUAL-RECURSION ... ( DEFUN ~x0 ...) ~
...)"
(car names))
ctx)
wrld state-vars)))
(trans-value (cons x y))))
(erp (mv erp x bindings2))
(t (let ((bindings bindings2))
(trans-er-let*
((y (translate-bodies1 non-executablep
(cdr names)
(cdr bodies)
bindings
(cdr arglists)
(cdr stobjs-in-lst)
ctx wrld state-vars)))
(trans-value (cons x y))))))))))
(defun chk-non-executable-bodies (names arglists bodies non-executablep
mut-rec-p ctx state)
; Note that bodies are in translated form.
(cond ((endp bodies)
(value nil))
(t (let ((name (car names))
(body (car bodies))
(formals (car arglists)))
; The body should generally be a translated form of (prog2$
; (throw-nonexec-error 'name (list . formals)) ...), as laid down by
; defun-nx-fn. But we make an exception for defproxy, i.e. (eq non-executablep
; :program), since it won't be true in that case and we don't care that it be
; true, as we have a program-mode function that does a throw.
(cond ((throw-nonexec-error-p body
(and (not (eq non-executablep
:program))
name)
formals)
(chk-non-executable-bodies
(cdr names) (cdr arglists) (cdr bodies)
non-executablep mut-rec-p ctx state))
(t (er soft ctx
"The body of a defun that is marked :non-executable ~
(perhaps implicitly, by the use of defun-nx~@1) must ~
be of the form (prog2$ (throw-nonexec-error ...) ~
...)~@2. The definition of ~x0 is thus illegal. ~
See :DOC defun-nx."
(car names)
(if mut-rec-p
" in some definition under the mutual-recursion"
"")
(if (eq non-executablep :program)
""
", as is laid down by defun-nx"))))))))
(defun collect-untouchable-fns (syms state)
(let ((temp-touchable-fns (f-get-global 'temp-touchable-fns state)))
(cond ((eq temp-touchable-fns t) nil)
(t (let* ((wrld (w state)) ; installed world
(untouchable-fns (global-val 'untouchable-fns wrld))
(int (intersection-eq syms untouchable-fns)))
(cond (temp-touchable-fns
(set-difference-eq int temp-touchable-fns))
(t int)))))))
(defun collect-untouchable-vars (syms state)
(let ((temp-touchable-vars (f-get-global 'temp-touchable-vars state)))
(cond ((eq temp-touchable-vars t) nil)
(t (let* ((wrld (w state)) ; installed world
(untouchable-vars (global-val 'untouchable-vars wrld))
(int (and syms ; optimization
(intersection-eq syms untouchable-vars))))
(cond (temp-touchable-vars
(set-difference-eq int temp-touchable-vars))
(t int)))))))
(defun get-translate-cert-data-record (type lst state)
; Lst is a list of translate-cert-data-record records associated with a single
; name. We return the unique one associated with type, if any, else nil.
; Reasons for returning nil include:
; (a) two or more relevant records (which would necessarily be distinct; see
; update-translate-cert-data);
; (b) function symbols in the translated term that are now untouchable; or
; (c) state global symbols in the translated term, assigned or made unbound,
; that are now untouchable.
(cond ((endp lst) nil)
((eq type (access translate-cert-data-record (car lst) :type))
(cond ((or (get-translate-cert-data-record type (cdr lst) state) ; (a)
(collect-untouchable-fns
(access translate-cert-data-record (car lst) :fns)
state) ; (b)
(collect-untouchable-vars
(access translate-cert-data-record (car lst) :vars)
state)) ; (c)
nil)
(t (car lst))))
(t (get-translate-cert-data-record type (cdr lst) state))))
(defun get-translate-bodies (names cert-data-entry state)
; Cert-data-entry is a valid cert-data entry for the :translate key. It is
; thus a list of translate-cert-data-record records. We return nil or else the
; unique bodies associated with names, checking for untouchables.
(cond ((null names) ; probably always false, but we check, for robustness
nil)
(t (let ((lst (cert-data-val (car names) cert-data-entry)))
(cond
((null lst) ; optimization
nil)
(t (let ((val (get-translate-cert-data-record :translate-bodies
lst
state)))
(and val
(assert$ (equal (access translate-cert-data-record val
:inputs)
names)
(access translate-cert-data-record val
:value))))))))))
(defun translate-bodies (non-executablep names arglists bodies bindings0
stobjs-in-lst
reclassifying-all-programp
ctx wrld state)
; Translate the bodies given and return a pair consisting of their translations
; and the final bindings from translate. Note that non-executable :program
; mode functions need to be analyzed for stobjs-out, because they are proxies
; (see :DOC defproxy) for encapsulated functions that may replace them later,
; and we need to guarantee to callers that those stobjs-out do not change with
; such replacements.
; Normally, this function is called with bindings0 = (pairlis$ names names),
; which indicates that the output signature of each name must be inferred
; during translation and stored in the ultimate value of bindings. But when
; :loop$-recursion is specified, the caller already knows the output signature
; of the fn being defined and will specify it in the call.
(declare (xargs :guard (true-listp bodies)))
(let ((cert-data-entry (cert-data-entry :translate state)))
(let ((cert-data-tbodies-and-bindings
(if cert-data-entry
; Note that we do not need to rule out make-event expansion explicitly, because
; it is already being ruled out: protect-system-state-globals (called by
; protected-eval, which does the evaluation for make-event expansion) binds
; state global 'cert-data to nil.
(get-translate-bodies names cert-data-entry state)
nil)))
(cond
(cert-data-tbodies-and-bindings (value cert-data-tbodies-and-bindings))
(t
(mv-let (erp lst bindings)
(translate-bodies1 (eq non-executablep t) ; not :program
names bodies
bindings0 arglists
stobjs-in-lst
ctx wrld
(default-state-vars t
; For the application of verify-termination to a function that has already
; been admitted, we avoid failure due to an untouchable function or variable.
:temp-touchable-fns
(or reclassifying-all-programp
(f-get-global 'temp-touchable-fns
state))
:temp-touchable-vars
(or reclassifying-all-programp
(f-get-global 'temp-touchable-vars
state))))
(er-progn
(cond (erp ; erp is a ctx, lst is a msg
(er-soft erp "Translate" "~@0" lst))
(non-executablep
(chk-non-executable-bodies names arglists lst
non-executablep (cdr names)
ctx state))
(t (value nil)))
(value (cons lst
(cond ((eq non-executablep t)
(pairlis-x2 names '(nil)))
(t bindings)))))))))))
; The next section develops our check that mutual recursion is
; sensibly used.
(defun chk-mutual-recursion-bad-names (lst names bodies)
(cond ((null lst) nil)
((ffnnamesp names (car bodies))
(chk-mutual-recursion-bad-names (cdr lst) names (cdr bodies)))
(t
(cons (car lst)
(chk-mutual-recursion-bad-names (cdr lst) names (cdr bodies))))))
(defconst *chk-mutual-recursion-string*
"The definition~#0~[~/s~] of ~&1 ~#0~[does~/do~] not call any of ~
the other functions being defined via ~
mutual recursion. The theorem prover ~
will perform better if you define ~&1 ~
without the appearance of mutual recursion. See ~
:DOC set-bogus-mutual-recursion-ok to get ~
ACL2 to handle this situation differently.")
(defun chk-mutual-recursion1 (names bodies warnp ctx state)
(cond
((and warnp
(warning-disabled-p "mutual-recursion"))
(value nil))
(t
(let ((bad (chk-mutual-recursion-bad-names names names bodies)))
(cond ((null bad) (value nil))
(warnp
(pprogn
(warning$ ctx ("mutual-recursion")
*chk-mutual-recursion-string*
(if (consp (cdr bad)) 1 0)
bad)
(value nil)))
(t (er soft ctx
*chk-mutual-recursion-string*
(if (consp (cdr bad)) 1 0)
bad)))))))
(defun chk-mutual-recursion (names bodies ctx state)
; We check that names has at least 1 element and that if it has
; more than one then every body calls at least one of the fns in
; names. The idea is to ensure that mutual recursion is used only
; when "necessary." This is not necessary for soundness but since
; mutually recursive fns are not handled as well as singly recursive
; ones, it is done as a service to the user. In addition, several
; error messages and other user-interface features exploit the presence
; of this check.
(cond ((null names)
(er soft ctx
"It is illegal to use MUTUAL-RECURSION to define no functions."))
((null (cdr names)) (value nil))
(t
(let ((bogus-mutual-recursion-ok
(cdr (assoc-eq :bogus-mutual-recursion-ok
(table-alist 'acl2-defaults-table (w state))))))
(if (eq bogus-mutual-recursion-ok t)
(value nil)
(chk-mutual-recursion1 names bodies
(eq bogus-mutual-recursion-ok :warn)
ctx state))))))
; We now develop put-induction-info.
(mutual-recursion
(defun ffnnamep-mod-mbe (fn term)
; We determine whether the function symbol fn is called after replacing each
; mbe call in term by its :logic component. Keep this in sync with the
; ffnnamep nest. Unlike ffnnamep, we assume here that fn is a symbolp.
(declare (xargs :guard (and (symbolp fn)
(pseudo-termp term))))
(cond ((variablep term) nil)
((fquotep term) nil)
((flambda-applicationp term)
(or (ffnnamep-mod-mbe fn (lambda-body (ffn-symb term)))
(ffnnamep-mod-mbe-lst fn (fargs term))))
((eq (ffn-symb term) fn) t)
((and (eq (ffn-symb term) 'return-last)
(quotep (fargn term 1))
(eq (unquote (fargn term 1)) 'mbe1-raw))
(ffnnamep-mod-mbe fn (fargn term 3)))
(t (ffnnamep-mod-mbe-lst fn (fargs term)))))
(defun ffnnamep-mod-mbe-lst (fn l)
(declare (xargs :guard (and (symbolp fn)
(pseudo-term-listp l))))
(if (null l)
nil
(or (ffnnamep-mod-mbe fn (car l))
(ffnnamep-mod-mbe-lst fn (cdr l)))))
)
; Here is how we set the recursivep property.
; Rockwell Addition: The recursivep property has changed. Singly
; recursive fns now have the property (fn) instead of fn.
(defun putprop-recursivep-lst (loop$-recursion-checkedp
loop$-recursion
names bodies wrld)
; On the property list of each function symbol is stored the 'recursivep
; property. For nonrecursive functions, the value is implicitly nil but no
; value is stored (see comment below). Otherwise, the value is a true-list of
; fn names in the ``clique.'' Thus, for singly recursive functions, the value
; is a singleton list containing the function name. For mutually recursive
; functions the value is the list of every name in the clique. This function
; stores the property for each name and body in names and bodies.
; When loop$-recursion is t, we know names is a singleton and that the function
; is indeed recursive. Otherwise, we use ffnnamep-mod-mbe to determine whether
; a singly defined function is recursive.
; WARNING: We rely on the fact that this function puts the same names into the
; 'recursivep property of each member of the clique, in our handling of
; being-openedp. Moreover, we rely in function termination-theorem-fn-subst
; (and its supporting functions) that the properties are placed in the order in
; which the names are defined: (mutual-recursion (defun name1 ...) (defun name2
; ... ...)) pushes a property for name1 onto a world with property for name2,
; etc.
(prog2$
(choke-on-loop$-recursion loop$-recursion-checkedp
names
bodies
'putprop-recursivep-lst)
(cond (loop$-recursion
(putprop (car names) 'recursivep names wrld))
((int= (length names) 1)
(cond ((ffnnamep-mod-mbe (car names) (car bodies))
(putprop (car names) 'recursivep names wrld))
(t
; Until we started using the 'def-bodies property to answer most questions
; about recursivep (see macro recursivep), it was a good idea to put a
; 'recursivep property of nil in order to avoid having getprop walk through an
; entire association list looking for 'recursivep. Now, this less-used
; property is just in the way.
wrld)))
(t (putprop-x-lst1 names 'recursivep names wrld)))))
; Formerly, we defined termination-machines and some of its supporting
; functions here. But we moved them to history-management.lisp in order to
; support the definition of termination-theorem-clauses.
; We next develop the function that guesses measures when the user has
; not supplied them.
(defun proper-dumb-occur-as-output (x y)
; We determine whether the term x properly occurs within the term y, insisting
; in addition that if y is an IF expression then x occurs properly within each
; of the two output branches.
; For example, X does not properly occur in X or Z. It does properly occur in
; (CDR X) and (APPEND X Y). It does properly occur in (IF a (CDR X) (CAR X))
; but not in (IF a (CDR X) 0) or (IF a (CDR X) X).
; This function is used in always-tested-and-changedp to identify a formal to
; use as the measured formal in the justification of a recursive definition.
; We seek a formal that is tested on every branch and changed in every
; recursion. But if (IF a (CDR X) X) is the new value of X in some recursion,
; then it is not really changed, since if we distributed the IF out of the
; recursive call we would see a call in which X did not change.
(cond ((equal x y) nil)
((variablep y) nil)
((fquotep y) nil)
((eq (ffn-symb y) 'if)
(and (proper-dumb-occur-as-output x (fargn y 2))
(proper-dumb-occur-as-output x (fargn y 3))))
(t (dumb-occur-lst x (fargs y)))))
(defun always-tested-and-changedp (var pos t-machine)
; Is var involved in every tests component of t-machine and changed
; and involved in every call, in the appropriate argument position?
; In some uses of this function, var may not be a variable symbol
; but an arbitrary term.
(cond ((null t-machine) t)
((and (dumb-occur-lst var
(access tests-and-call
(car t-machine)
:tests))
(let ((argn (nth pos
(fargs (access tests-and-call
(car t-machine)
:call)))))
; If argn is nil then it means there were not enough args to get the one at
; pos. This can happen in a mutually recursive clique where not all clique
; members have the same arity.
(and argn
(proper-dumb-occur-as-output var argn))))
(always-tested-and-changedp var pos (cdr t-machine)))
(t nil)))
(defun guess-measure (name defun-flg args pos t-machine ctx wrld state)
; T-machine is a termination machine, i.e., a lists of tests-and-call. Because
; of mutual recursion, we do not know that the call of a tests-and-call is a
; call of name; it may be a call of a sibling of name. We look for the first
; formal that is (a) somehow tested in every test and (b) somehow changed in
; every call. Upon finding such a var, v, we guess the measure (acl2-count v).
; But what does it mean to say that v is "changed in a call" if we are defining
; (foo x y v) and see a call of bar? We mean that v occurs in an argument to
; bar and is not equal to that argument. Thus, v is not changed in (bar x v)
; and is changed in (bar x (mumble v)). The difficulty here of course is that
; (mumble v) may not be being passed as the new value of v. But since this is
; just a heuristic guess intended to save the user the burden of typing
; (acl2-count x) a lot, it doesn't matter.
; If we fail to find a measure we cause an error.
; Pos is initially 0 and is the position in the formals list of the first
; variable listed in args. Defun-flg is t if we are guessing a measure on
; behalf of a function definition and nil if we are guessing on behalf of a
; :definition rule. It affects only the error message printed.
(cond ((null args)
(cond
((null t-machine)
; Presumably guess-measure was called here with args = NIL, for example if
; :set-bogus-mutual-recursion allowed it. We pick a silly measure that will
; work. If it doesn't work (hard to imagine), well then, we'll find out when
; we try to prove termination.
(value (mcons-term* (default-measure-function wrld) *0*)))
(t
(er soft ctx
"No ~#0~[:MEASURE~/:CONTROLLER-ALIST~] was supplied with the ~
~#0~[definition of~/proposed :DEFINITION rule for~] ~x1. Our ~
heuristics for guessing one have not made any suggestions. ~
No argument of the function is tested along every branch of ~
the relevant IF structure and occurs as a proper subterm at ~
the same argument position in every recursive call. You must ~
specify a ~#0~[:MEASURE. See :DOC defun.~/:CONTROLLER-ALIST. ~
~ See :DOC definition.~@2~] Also see :DOC ruler-extenders ~
for how to affect how much of the IF structure is explored by ~
our heuristics."
(if defun-flg 0 1)
name
(cond
(defun-flg "")
(t " In some cases you may wish to use the :CONTROLLER-ALIST ~
from the original (or any previous) definition; this may ~
be seen by using :PR."))))))
((always-tested-and-changedp (car args) pos t-machine)
(value (mcons-term* (default-measure-function wrld) (car args))))
(t (guess-measure name defun-flg (cdr args) (1+ pos)
t-machine ctx wrld state))))
(defun guess-measure-alist (names arglists measures t-machines ctx wrld state)
; We either cause an error or return an alist mapping the names in
; names to their measures (either user suggested or guessed).
; Warning: The returned alist, a, should have the property that (strip-cars a)
; is equal to names. We rely on that property in put-induction-info.
(cond ((null names) (value nil))
((equal (car measures) *no-measure*)
(er-let* ((m (guess-measure (car names)
t
(car arglists)
0
(car t-machines)
ctx wrld state)))
(er-let* ((alist (guess-measure-alist (cdr names)
(cdr arglists)
(cdr measures)
(cdr t-machines)
ctx wrld state)))
(value (cons (cons (car names) m)
alist)))))
(t (er-let* ((alist (guess-measure-alist (cdr names)
(cdr arglists)
(cdr measures)
(cdr t-machines)
ctx wrld state)))
(value (cons (cons (car names) (car measures))
alist))))))
; We now embark on the development of prove-termination, which must
; prove the justification theorems for each termination machine and
; the measures supplied/guessed.
; Moved remove-built-in-clauses and clean-up-clause-set to
; history-management.lisp.
; Formerly, we defined measure-clauses-for-clique and some of its supporting
; functions here. But we moved them to history-management.lisp in order to
; support the definition of termination-theorem-clauses.
(defun tilde-*-measure-phrase1 (alist wrld)
(cond ((null alist) nil)
(t (cons (msg (cond ((null (cdr alist)) "~p1 for ~x0.")
(t "~p1 for ~x0"))
(caar alist)
(untranslate (cdar alist) nil wrld))
(tilde-*-measure-phrase1 (cdr alist) wrld)))))
(defun tilde-*-measure-phrase (alist wrld)
; Let alist be an alist mapping function symbols, fni, to measure terms, mi.
; The fmt directive ~*0 will print the following, if #\0 is bound to
; the output of this fn:
; "m1 for fn1, m2 for fn2, ..., and mk for fnk."
; provided alist has two or more elements. If alist contains
; only one element, it will print just "m1."
; Note the final period at the end of the phrase! In an earlier version
; we did not add the period and saw a line-break between the ~x1 below
; and its final period.
; Thus, the following fmt directive will print a grammatically correct
; sentence ending with a period: "For the admission of ~&1 we will use
; the measure ~*0"
(list* "" "~@*" "~@* and " "~@*, "
(cond
((null (cdr alist))
(list (cons "~p1."
(list (cons #\1
(untranslate (cdar alist) nil wrld))))))
(t (tilde-*-measure-phrase1 alist wrld)))
nil))
(defun find-?-measure (measure-alist)
(cond ((endp measure-alist) nil)
((let* ((entry (car measure-alist))
(measure (cdr entry)))
(and (consp measure)
(eq (car measure) :?)
entry)))
(t (find-?-measure (cdr measure-alist)))))
(defun prove-termination (names t-machines measure-alist mp rel hints otf-flg
bodies measure-debug ctx ens wrld state ttree)
; Given a list of the functions introduced in a mutually recursive clique,
; their t-machines, the measure-alist for the clique, a domain predicate mp on
; which a certain relation, rel, is known to be well-founded, a list of hints
; (obtained by joining all the hints in the defuns), and a world in which we
; can find the 'formals of each function in the clique, we prove the theorems
; required by the definitional principle. In particular, we prove that each
; measure is an o-p and that in every tests-and-call in the t-machine of each
; function, the measure of the recursive calls is strictly less than that of
; the incoming arguments. If we fail, we cause an error.
; This function produces output describing the proofs. It should be the first
; output-producing function in the defun processing on every branch through
; defun. It always prints something and leaves you in a clean state ready to
; begin a new sentence, but may leave you in the middle of a line (i.e., col >
; 0).
; If we succeed we return two values, consed together as "the" value in this
; error/value/state producing function. The first value is the column produced
; by our output. The second value is a ttree in which we have accumulated all
; of the ttrees associated with each proof done.
; This function is specially coded so that if t-machines is nil then it is a
; signal that there is only one element of names and it is a non-recursive
; function. In that case, we short-circuit all of the proof machinery and
; simply do the associated output. We coded it this way to preserve the
; invariant that prove-termination is THE place the defun output is initiated.
; This function increments timers. Upon entry, any accumulated time is charged
; to 'other-time. The printing done herein is charged to 'print-time and the
; proving is charged to 'prove-time.
(mv-let
(cl-set cl-set-ttree)
(cond ((and (not (ld-skip-proofsp state))
t-machines)
(clean-up-clause-set
(measure-clauses-for-clique names
t-machines
measure-alist
mp rel measure-debug
wrld)
ens
wrld ttree state))
(t (mv nil ttree)))
(cond
((and (not (ld-skip-proofsp state))
(find-?-measure measure-alist))
(let* ((entry (find-?-measure measure-alist))
(fn (car entry))
(measure (cdr entry)))
(er soft ctx
"A :measure of the form (:? v1 ... vk) is only legal when the ~
defun is redundant (see :DOC redundant-events) or when skipping ~
proofs (see :DOC ld-skip-proofsp). The :measure ~x0 supplied for ~
function symbol ~x1 is thus illegal."
measure fn)))
(t
(er-let*
((cl-set-ttree (accumulate-ttree-and-step-limit-into-state
cl-set-ttree :skip state)))
(pprogn
(increment-timer 'other-time state)
(let ((displayed-goal (prettyify-clause-set cl-set
(let*-abstractionp state)
wrld))
(simp-phrase (tilde-*-simp-phrase cl-set-ttree)))
(mv-let
(col state)
(cond
((ld-skip-proofsp state)
(mv 0 state))
((null t-machines)
(io? event nil (mv col state)
(names)
(fmt "Since ~&0 is non-recursive, its admission is trivial."
(list (cons #\0 names))
(proofs-co state)
state
nil)
:default-bindings ((col 0))))
((null cl-set)
(io? event nil (mv col state)
(measure-alist wrld rel names)
(fmt "The admission of ~&0 ~#0~[is~/are~] trivial, using ~@1 ~
and the measure ~*2"
(list (cons #\0 names)
(cons #\1 (tilde-@-well-founded-relation-phrase
rel wrld))
(cons #\2 (tilde-*-measure-phrase
measure-alist wrld)))
(proofs-co state)
state
(term-evisc-tuple nil state))
:default-bindings ((col 0))))
(t
(io? event nil (mv col state)
(cl-set-ttree displayed-goal simp-phrase measure-alist wrld
rel names)
(fmt "For the admission of ~&0 we will use ~@1 and the ~
measure ~*2 The non-trivial part of the measure ~
conjecture~#3~[~/, given ~*4,~] is~@5~%~%Goal~%~Q67."
(list (cons #\0 names)
(cons #\1 (tilde-@-well-founded-relation-phrase
rel wrld))
(cons #\2 (tilde-*-measure-phrase
measure-alist wrld))
(cons #\3 (if (nth 4 simp-phrase) 1 0))
(cons #\4 simp-phrase)
(cons #\5 (if (tagged-objectsp 'sr-limit
cl-set-ttree)
" as follows (where the ~
subsumption/replacement limit ~
affected this analysis; see :DOC ~
case-split-limitations)."
""))
(cons #\6 displayed-goal)
(cons #\7 (term-evisc-tuple nil state)))
(proofs-co state)
state
nil)
:default-bindings ((col 0)))))
(pprogn
(increment-timer 'print-time state)
(cond
((null cl-set)
; If the io? above did not print because 'event is inhibited, then col is nil.
; Just to keep ourselves sane, we will set it to 0.
(value (cons (or col 0) cl-set-ttree)))
(t
(mv-let
(erp ttree state)
(prove (termify-clause-set cl-set)
(make-pspv ens wrld state
:displayed-goal displayed-goal
:otf-flg otf-flg)
hints ens wrld ctx state)
(cond (erp
(let ((erp-msg
(cond
((subsetp-eq
'(summary error)
(f-get-global 'inhibit-output-lst state))
; This case is an optimization, in order to avoid the computations below, in
; particular of termination-machines. Note that erp-msg is potentially used in
; error output -- see the (er soft ...) form below -- and it is also
; potentially used in summary output, when print-summary passes to
; print-failure the first component of the error triple returned below.
nil)
(t
(msg
"The proof of the measure conjecture for ~&0 ~
has failed.~@1"
names
(if (equal
t-machines
(termination-machines
t ; loop$-recursion-checkedp
(if (cdr names) ; loop$-recursion
nil
(getpropc (car names)
'loop$-recursion
nil wrld))
names
(if (cdr names)
nil
(list (formals (car names) wrld)))
bodies
(make-list (length names)
:initial-element
:all)))
""
(msg "~|**NOTE**: The use of declaration ~
~x0 would change the measure ~
conjecture, perhaps making it easier ~
to prove. See :DOC ruler-extenders."
'(xargs :ruler-extenders :all))))))))
(mv-let
(erp val state)
(er soft ctx "~@0~|" erp-msg)
(declare (ignore erp val))
(mv (msg "~@0 " erp-msg) nil state))))
(t
(mv-let (col state)
(io? event nil (mv col state)
(names)
(fmt "That completes the proof of the ~
measure theorem for ~&1. Thus, we ~
admit ~#1~[this function~/these ~
functions~] under the principle of ~
definition."
(list (cons #\1 names))
(proofs-co state)
state
nil)
:default-bindings ((col 0)))
(pprogn
(increment-timer 'print-time state)
(value
(cons
(or col 0)
(cons-tag-trees
cl-set-ttree ttree)))))))))))))))))))
; When we succeed in proving termination, we will store the
; justification properties.
(defun putprop-justification-lst (measure-alist subset-lst mp rel
ruler-extenders-lst
subversive-p wrld)
; Each function has a 'justification property. The value of the property
; is a justification record.
(cond ((null measure-alist) wrld)
(t (putprop-justification-lst
(cdr measure-alist) (cdr subset-lst) mp rel (cdr ruler-extenders-lst)
subversive-p
(putprop (caar measure-alist)
'justification
(make justification
:subset
; The following is equal to (all-vars (cdar measure-alist)), but since we
; already have it available, we use it rather than recomputing this all-vars
; call.
(car subset-lst)
:subversive-p subversive-p
:mp mp
:rel rel
:measure (cdar measure-alist)
:ruler-extenders (car ruler-extenders-lst))
wrld)))))
(defun clean-up-nots (p)
(case-match p
(('IF ('IF q ''NIL ''T) ''NIL ''T) q)
(('NOT ('IF q ''NIL ''T)) q)
(('IF ('NOT q) ''NIL ''T) q)
(('NOT ('NOT q)) q)
(('IF q ''NIL ''T) `(NOT ,q))
(& p)))
(defun clean-up-nots-lst (lst ans)
; Simplify double negations and reverse the order of the terms in lst.
(cond ((endp lst) ans)
(t (clean-up-nots-lst (cdr lst) (cons (clean-up-nots (car lst)) ans)))))
(defun clean-up-conjunction1 (lst ans)
(cond
((endp lst) ans)
((member-complement-term (car lst) (cdr lst))
:contradiction)
((member-equal (car lst) (cdr lst))
(clean-up-conjunction1 (cdr lst) ans))
(t (clean-up-conjunction1 (cdr lst) (cons (car lst) ans)))))
(defun clean-up-conjunction (lst)
; Lst is a list of hypotheses, implicitly conjoined. We return either
; :contradiction or an equivalent list of conjuncts after eliminating
; duplicates and double negations.
(clean-up-conjunction1 (clean-up-nots-lst lst nil) nil))
(defun clean-up-loop$-recursion-induction-machine (tc-list)
; Tc-list is a list of tests-and-calls, i.e., an induction machine. However,
; as of this writing this function is only applied to those induction machines
; generated from loop$ recursion induction function bodies, i.e., inductible
; loop$ recursive functions. This function is spiritually similar to
; simplify-tests-and-calls but just drops cases having contradictory tests and
; eliminates duplicates and double negations among the remaining tests. We
; implement this process as a separate function rather than strengthen
; simplify-tests-and-calls because we wish not to change any existing induction
; machines as we add the ability to induct for loop$ recursive functions.
(cond
((endp tc-list) nil)
(t (let ((tests
(clean-up-conjunction
(access tests-and-calls (car tc-list) :tests))))
(cond
((eq tests :contradiction)
(clean-up-loop$-recursion-induction-machine (cdr tc-list)))
(t (cons
(make tests-and-calls
:tests tests
:calls (access tests-and-calls (car tc-list) :calls))
(clean-up-loop$-recursion-induction-machine (cdr tc-list)))))))))
(defun induction-machines
(loop$-recursion names arglists measure-alist bodies
ruler-extenders-lst wrld)
(declare (ignore arglists measure-alist wrld)) ; See Note 2 below.
; This function builds the induction machine for each function defined
; in names with the corresponding body in bodies. A list of machines
; is returned. See termination-machine.
; Note: If names has more than one element we return nil because we do
; not know how to interpret the induction-machines that would be
; constructed from a non-trivial clique of mutually recursive
; functions. As a matter of fact, as of this writing,
; induction-machine-for-fn constructs the "natural" machine for
; mutually recursive functions, but there's no point in consing them
; up since we can't use them. So all that machinery is
; short-circuited here.
; Note 1: If loop$-recursion has been used (which is only possible if names is
; a singleton) we refuse to generate an induction machine. As of March, 2020,
; we haven't fully understood how to induct appropriately for such functions.
; We currently aim to produce a book containing a utility, perhaps named
; definductor, that takes the name of an admitted loop$ recursive function and
; produces an induction scheme for it. But our current understanding of this
; problem produces plausible schemes only for plain, possibly nested loop$s
; targetting formal variables or car/cdr nests of formal variables.
; Furthermore, these schemes impose additional restrictions on the acceptable
; measures justifying the loop$ recursive functions -- restrictions that are
; not necessary just for admission. Finally, our current understanding has not
; been sufficiently tested to deserve being in our sources! Rather than add a
; questionable ``automatic'' scheme that restricts admissible loop$ recursive
; functions or confusing the user by sometimes generating an induction scheme
; and sometimes not, we have opted to NEVER produce a scheme and leave it to
; the user (possibly with help from definductor) to introduce effective
; schemes.
; Note 2: The three ignored arguments are artifacts of an earlier experiment in
; which we tried to automatically generate induction schemes for ``inductible''
; loop$ recursive functions. When we conducted that experiment we needed those
; arguments and had to refactor the code to get them. But when we abandoned
; automatically generating induction schemes for loop$ recursive functions we
; decided not to revert to the earlier factoring, just in case we someday
; decide to handle such functions here.
(cond ((null (cdr names))
(if loop$-recursion
nil
(list (induction-machine-for-fn names (car bodies)
(car ruler-extenders-lst)))))
(t nil)))
(defun putprop-induction-machine-lst
(loop$-recursion names arglists measure-alist bodies
ruler-extenders-lst subversive-p wrld)
; Note: If names has more than one element we do nothing. We only
; know how to interpret induction machines for singly recursive fns.
(cond ((cdr names) wrld)
(subversive-p wrld)
(t (putprop (car names)
'induction-machine
(car (induction-machines
loop$-recursion names arglists measure-alist bodies
ruler-extenders-lst wrld))
wrld))))
(defun putprop-quick-block-info-lst (names t-machines wrld)
; We do not know how to compute quick-block-info for non-trivial
; mutually-recursive cliques. We therefore don't do anything for
; those functions. If names is a list of length 1, we do the
; computation. We assume we can find the formals of the name in wrld.
(cond ((null (cdr names))
(putprop (car names)
'quick-block-info
(quick-block-info (car names)
(formals (car names) wrld)
(car t-machines))
wrld))
(t wrld)))
(defmacro big-mutrec (names)
; All mutual recursion nests with more than the indicated number of defuns will
; be processed by installing intermediate worlds, for improved performance. We
; have seen an improvement of roughly two orders of magnitude in such a case.
; The value below is merely heuristic, chosen with very little testing; we
; should feel free to change it.
`(> (length ,names) 20))
(defun get-sig-fns1 (ee-lst)
(cond ((endp ee-lst)
nil)
(t (let ((ee-entry (car ee-lst)))
(cond ((and (eq (car ee-entry) 'encapsulate)
(cddr ee-entry)) ; pass-2
(append (get-sig-fns1 (cdr ee-lst)) ; usually nil
(strip-cars (cadr ee-entry))))
(t
(get-sig-fns1 (cdr ee-lst))))))))
(defun get-sig-fns (wrld)
(get-sig-fns1 (global-val 'embedded-event-lst wrld)))
(defun selected-all-fnnames-lst (formals subset actuals acc)
(cond ((endp formals) acc)
(t (selected-all-fnnames-lst
(cdr formals) subset (cdr actuals)
(if (member-eq (car formals) subset)
(all-fnnames1 nil (car actuals) acc)
acc)))))
(defun subversivep (fns t-machine formals-and-subset-alist wrld)
; See subversive-cliquep for conditions (1) and (2).
(cond ((endp t-machine) nil)
(t (or
; Condition (1):
(intersectp-eq fns
(instantiable-ancestors
(all-fnnames-lst (access tests-and-call
(car t-machine)
:tests))
wrld
nil))
; Condition (2):
(let* ((call (access tests-and-call
(car t-machine)
:call))
(entry
(assoc-eq (ffn-symb call)
formals-and-subset-alist))
(formals (assert$ entry (cadr entry)))
(subset (cddr entry))
(measured-call-args-ancestors
(instantiable-ancestors
(selected-all-fnnames-lst formals subset
(fargs call) nil)
wrld
nil)))
(intersectp-eq fns measured-call-args-ancestors))
; Recur:
(subversivep fns (cdr t-machine) formals-and-subset-alist wrld)))))
(defun subversive-cliquep (fns t-machines formals-and-subset-alist wrld)
; Here, fns is a list of functions introduced in an encapsulate. If we are
; using the [Front] rule (from the Structured Theory paper) to move some
; functions forward, then fns is the list of ones that are NOT moved: they all
; use the signature functions somehow. T-machines is a list of termination
; machines for some clique of functions defined within the encapsulate. The
; clique is subversive if some function defined in the clique has a subversive
; t-machine.
; Intuitively, a t-machine is subversive if its admission depended on
; properties of the witnesses for signature functions. That is, the definition
; uses signature functions in a way that affects the termination argument.
; Technically a t-machine is subversive if some tests-and-call record in it has
; either of the following properties:
; (1) a test mentions a function in fns
; (2) an argument of a call in a measured position mentions a function in fns.
; Observe that if a clique is not subversive then every test and argument to
; every recursive call uses functions defined outside the encapsulate. If we
; are in a top-level encapsulate, then a non-subversive clique is a ``tight''
; clique wrt the set S of functions in the initial world of the encapsulate,
; where ``tight'' is defined by the Structured Theory paper, i.e.: for every
; subterm u of a ruler or recursive call in the clique, all function symbols of
; u belong to S (but now we restrict to measured positions in recursive
; calls).
(cond ((endp t-machines) nil)
(t (or (subversivep fns (car t-machines) formals-and-subset-alist wrld)
(subversive-cliquep fns (cdr t-machines)
formals-and-subset-alist wrld)))))
(defun prove-termination-non-recursive (names bodies mp rel hints otf-flg
big-mutrec ctx ens wrld state)
; This function separates out code from put-induction-info.
(er-progn
(cond
(hints
(let ((bogus-defun-hints-ok
(cdr (assoc-eq :bogus-defun-hints-ok
(table-alist 'acl2-defaults-table
(w state))))))
(cond
((eq bogus-defun-hints-ok :warn)
(pprogn
(warning$ ctx "Non-rec"
"Since ~x0 is non-recursive your supplied :hints will be ~
ignored (as these are used only during termination ~
proofs). Perhaps either you meant to supply ~
:guard-hints or the body of the definition is incorrect."
(car names))
(value nil)))
(bogus-defun-hints-ok ; t
(value nil))
(t ; bogus-defun-hints-ok = nil, the default
(er soft ctx
"Since ~x0 is non-recursive it is odd that you have supplied ~
:hints (which are used only during termination proofs). We ~
suspect something is amiss, e.g., you meant to supply ~
:guard-hints or the body of the definition is incorrect. To ~
avoid this error, see :DOC set-bogus-defun-hints-ok."
(car names))))))
(t (value nil)))
(er-let*
((wrld1 (update-w big-mutrec wrld))
(pair (prove-termination names nil nil mp rel nil otf-flg bodies nil
ctx ens wrld1 state nil)))
; We know that pair is of the form (col . ttree), where col is the column
; the output state is in.
(value (list (car pair)
wrld1
(cdr pair))))))
(defun prove-termination-recursive (names arglists measures t-machines
mp rel hints otf-flg bodies
measure-debug
ctx ens wrld state)
; This function separates out code from put-induction-info.
; First we get the measures for each function. That may cause an error if we
; couldn't guess one for some function.
(er-let* ((measure-alist (guess-measure-alist names arglists
measures
t-machines
ctx wrld state))
(hints (cond
((member-eq (ld-skip-proofsp state)
'(include-book
include-book-with-locals
initialize-acl2))
; Hints are ignored below (by prove-termination) when proofs are skipped. But
; as with defthm-fn1 (for example), we translate default hints when
; ld-skip-proofsp is t because in that case our intention is to do all checks
; except for the actual proof attempts.
(value nil))
(hints ; hints and default-hints already translated
(value hints))
(t (let ((default-hints (default-hints wrld)))
(if default-hints ; not yet translated
(translate-hints
(cons "Measure Lemma for" (car names))
default-hints ctx wrld state)
(value hints))))))
(pair (prove-termination names
t-machines
measure-alist
mp
rel
hints
otf-flg
bodies
measure-debug
ctx
ens
wrld
state
nil)))
; Ok, we have managed to prove termination! Pair is a pair of the form (col .
; ttree), where col tells us what column the printer is in and ttree describes
; the proofs done.
(value (list* (car pair) measure-alist (cdr pair)))))
(defun put-induction-info-recursive
(loop$-recursion names arglists col ttree measure-alist t-machines
ruler-extenders-lst bodies mp rel wrld state)
; This function separates out code from put-induction-info.
; We have proved termination. Col tells us what column the printer is in and
; ttree describes the proofs done. We now store the 'justification of each
; function, the induction machine for each function, and the quick-block-info.
(let* ((subset-lst
; Below, we rely on the fact that this subset-lst corresponds, in order, to
; names. See the warnings comment in guess-measure-alist.
(collect-all-vars (strip-cdrs measure-alist)))
(sig-fns (get-sig-fns wrld))
(subversive-p (and sig-fns
(subversive-cliquep
sig-fns
t-machines
(pairlis$ names
(pairlis$ arglists
subset-lst))
wrld)))
(wrld2
(putprop-induction-machine-lst
loop$-recursion names arglists measure-alist bodies
ruler-extenders-lst subversive-p wrld))
(wrld3
(putprop-justification-lst measure-alist
subset-lst
mp rel
ruler-extenders-lst
subversive-p wrld2))
(wrld4 (putprop-quick-block-info-lst names
t-machines
wrld3)))
; We are done. We will return the final wrld and the ttree describing
; the proofs we did.
(value
(list* col
wrld4
(push-lemma
(cddr (assoc-eq rel
(global-val
'well-founded-relation-alist
wrld4)))
ttree)
subversive-p))))
(defun maybe-warn-or-error-on-non-rec-measure (name ctx wrld state)
(let ((bogus-defun-hints-ok
(cdr (assoc-eq :bogus-defun-hints-ok
(table-alist 'acl2-defaults-table wrld)))))
(cond
((eq bogus-defun-hints-ok :warn)
(pprogn
(warning$ ctx "Non-rec"
"Since ~x0 is non-recursive your supplied :measure will be ~
ignored (as the :measure is used only during termination ~
proofs)."
name)
(value nil)))
(bogus-defun-hints-ok ; t
(value nil))
(t ; bogus-defun-hints-ok = nil, the default
(er soft ctx
"It is illegal to supply a measure for a non-recursive function, as ~
has been done for ~x0. To avoid this error, see :DOC ~
set-bogus-measure-ok."
name)))))
(defun collect-problematic-quoted-fns (names0 fns wrld progs unwars)
; Fns is a list of quoted symbols used as function symbols in :FN or :EXPR
; slots. We collect into progs and unwars the :program mode ones and the
; un-warranted :logic ones that require warrants.
; Warning: Do not call this function during boot-strapping! Warrant
; information is not available.
(cond ((endp fns)
(mv (reverse progs)
(reverse unwars)))
((or
(member-eq (car fns) names0)
; The following hons-get is equivalent to (apply$-primp (car fns)).
(hons-get (car fns) ; *badge-prim-falist* is not yet defined!
(unquote
(getpropc '*badge-prim-falist*
'const nil wrld)))
; We similarly inspect the value of *apply$-boot-fns-badge-alist*
(assoc-eq (car fns)
(unquote
(getpropc '*apply$-boot-fns-badge-alist*
'const nil wrld))))
(collect-problematic-quoted-fns names0 (cdr fns)
wrld progs unwars))
((programp (car fns) wrld)
(collect-problematic-quoted-fns names0 (cdr fns) wrld
(add-to-set-eq (car fns) progs)
unwars))
((not (get-warrantp (car fns) wrld))
(collect-problematic-quoted-fns names0 (cdr fns) wrld progs
(add-to-set-eq (car fns) unwars)))
(t
(collect-problematic-quoted-fns names0 (cdr fns) wrld
progs unwars))))
(defun maybe-warn-on-problematic-quoted-fns (names measures bodies ctx wrld state)
(if (global-val 'boot-strap-flg wrld)
(value nil)
(mv-let (progs unwars)
(collect-problematic-quoted-fns
names
(all-fnnames! t ; list of terms flg
:inside ; collect from inside quotes
nil ; these terms are outside of quotes
(append measures bodies)
nil ; ilks
wrld
nil)
wrld nil nil)
(pprogn
(cond
(progs
(warning$ ctx "Problematic-quoted-fns"
"The definition~#0~[~/s~] of ~&0 ~#0~[is~/are~] in ~
:LOGIC mode but mention~#0~[s~/~] the :PROGRAM mode ~
function~#1~[~/s~] ~&1 in one or more :FN or :EXPR ~
slots. Conjectures about ~#0~[~&0~/the functions ~
defined in the clique~] may not be provable until ~
~#1~[this program is~/these programs are~] converted to ~
:LOGIC mode and warranted! See :DOC verify-termination ~
and defwarrant."
names
progs))
(t state))
(cond
(unwars
(warning$ ctx "Problematic-quoted-fns"
"The definition~#0~[~/s~] of ~&0 ~#0~[is~/are~] in ~
:LOGIC mode but mention~#0~[s~/~] the unwarranted ~
function~#1~[~/s~] ~&1 in one or more :FN or :EXPR ~
slots. Conjectures about ~#0~[~&0~/the functions ~
defined in the clique~] may not be provable until ~
~#1~[this unwarranted function is~/these unwarranted ~
functions are~] warranted! See :DOC defwarrant."
names
unwars))
(t state))
(value nil)))))
(defun put-induction-info ( ; we assume loop$-recursion-checkedp = t
loop$-recursion
names arglists measures ruler-extenders-lst bodies
mp rel hints otf-flg big-mutrec measure-debug
ctx ens wrld state)
; WARNING: This function installs a world. That is safe at the time of this
; writing because this function is only called by defuns-fn0, which is only
; called by defuns-fn, where that call is protected by a revert-world-on-error.
; Reminder: If user books start to call this function (or any of our functions
; that rely on loop$ recursion having been checked), we need to add
; loop$-recursion-checkedp as a new formal and start by calling
; choke-on-loop$-recursion. The call of put-induction-info-recursive, below,
; has the first argument t in our code because we know loop$ recursion has been
; checked down there. So that t stays t even if we add
; loop$-recursion-checkedp as a new formal above.
; We are processing a clique of mutually recursive functions with the names,
; arglists, measures, ruler-extenders-lst, and bodies given. All of the above
; lists are in 1:1 correspondence. The hints is the result of appending
; together all of the hints provided. Mp and rel are the domain predicate and
; well-founded relation to be used. We attempt to prove the admissibility of
; the recursions. We cause an error if any proof fails. We put a lot of
; properties under the function symbols, namely:
; recursivep all fns in names
; justification all recursive fns in names
; induction-machine the singly recursive fn in name*
; quick-block-info the singly recursive fn in name*
; symbol-class :ideal all fns in names
; *If names consists of exactly one recursive fn, we store its
; induction-machine and its quick-block-info, otherwise we do not.
; If no error occurs, we return a triple consisting of the column the printer
; is in, the final value of wrld and a tag-tree documenting the proofs we did.
; Note: The function could be declared to return 5 values, but we would rather
; use the standard state and error primitives and so it returns 3 and lists
; together the three "real" answers.
(let ((wrld1 (putprop-recursivep-lst t loop$-recursion names bodies wrld)))
; The put above stores a note on each function symbol as to whether it is
; recursive or not. An important question arises: have we inadvertently
; assumed something axiomatically about inadmissible functions? We say no.
; None of the functions in question have bodies yet, so the simplifier doesn't
; care about properties such as 'recursivep. However, we make use of this
; property below to decide if we need to prove termination.
(cond ((and (null (cdr names))
(null (getpropc (car names) 'recursivep nil wrld1)))
; If only one function is being defined and it is non-recursive, we can quit.
; But we have to store the symbol-class and we have to print out the
; program/unwarranted warning and the admission messages with prove-termination
; so the rest of our processing is uniform.
(er-progn
(cond ((equal (car measures) *no-measure*)
(value nil))
(t (maybe-warn-or-error-on-non-rec-measure (car names) ctx
wrld1 state)))
(maybe-warn-on-problematic-quoted-fns
names
(if (equal (car measures) *no-measure*)
nil
measures)
bodies
ctx wrld1 state)
(prove-termination-non-recursive names bodies mp rel hints otf-flg
big-mutrec ctx ens wrld1 state)))
(t
; Otherwise we first construct the termination machines for all the
; functions in the clique.
(let ((t-machines
(termination-machines t ; loop$-recursion-checkedp
loop$-recursion
names arglists bodies
ruler-extenders-lst)))
; Next we get the measures for each function. That may cause an error
; if we couldn't guess one for some function.
(er-let*
((wrld1 (update-w
; Sol Swords sent an example in which a clause-processor failed during a
; termination proof. That problem goes away if we install the world, which we
; do by making the following binding. This seems particularly important now
; that raw-ev-fncall calls chk-raw-ev-fncall to ensure that the world is
; (essentially) installed.
t ; formerly big-mutrec
wrld1))
(temp (maybe-warn-on-problematic-quoted-fns
names
(if (equal (car measures) *no-measure*)
nil
measures)
bodies
ctx wrld1 state))
; Temp is intentionally ignored; it is nil and a warning may have been printed.
(triple (prove-termination-recursive
names arglists measures t-machines mp rel hints
otf-flg bodies measure-debug ctx ens wrld1 state)))
(let* ((col (car triple))
(measure-alist (cadr triple))
(ttree (cddr triple)))
(put-induction-info-recursive
loop$-recursion
names arglists col ttree measure-alist t-machines
ruler-extenders-lst bodies mp rel wrld1 state))))))))
; We next worry about storing the normalized bodies.
(defun destructure-definition (term install-body ens wrld ttree)
; Term is a translated term that is the :corollary of a :definition rule. If
; install-body is non-nil then we intend to update the 'def-bodies
; property; and if moreover, install-body is :normalize, then we want to
; normalize the resulting new body. Ens is an enabled structure if
; install-body is :normalize; otherwise ens is ignored.
; We return (mv hyps equiv fn args body new-body ttree) or else nils if we fail
; to recognize the form of term. Hyps results flattening the hypothesis of
; term, when a call of implies, into a list of hypotheses. Failure can be
; detected by checking for (null fn) since nil is not a legal fn symbol.
(mv-let
(hyps equiv fn-args body)
(case-match term
(('implies hyp (equiv fn-args body))
(mv (flatten-ands-in-lit hyp)
equiv
fn-args
body))
((equiv fn-args body)
(mv nil
equiv
fn-args
body))
(& (mv nil nil nil nil)))
(let ((equiv (if (member-eq equiv *equality-aliases*)
'equal
equiv))
(fn (and (consp fn-args) (car fn-args))))
(cond
((and fn
(symbolp fn)
(not (member-eq fn
; Hide is disallowed in chk-acceptable-definition-rule.
'(quote if)))
(equivalence-relationp equiv wrld))
(mv-let (body ttree)
(cond ((eq install-body :NORMALIZE)
(normalize (remove-guard-holders body wrld)
nil ; iff-flg
nil ; type-alist
ens
wrld
ttree
(normalize-ts-backchain-limit-for-defs wrld)))
(t (mv body ttree)))
(mv hyps
equiv
fn
(cdr fn-args)
body
ttree)))
(t (mv nil nil nil nil nil nil))))))
(defun member-rewrite-rule-rune (rune lst)
; Lst is a list of :rewrite rules. We determine whether there is a
; rule in lst with the :rune rune.
(cond ((null lst) nil)
((equal rune (access rewrite-rule (car lst) :rune)) t)
(t (member-rewrite-rule-rune rune (cdr lst)))))
(defun replace-rewrite-rule-rune (rune rule lst)
; Lst is a list of :rewrite rules and one with :rune rune is among them.
; We replace that rule with rule.
(cond ((null lst) nil)
((equal rune (access rewrite-rule (car lst) :rune))
(cons rule (cdr lst)))
(t (cons (car lst) (replace-rewrite-rule-rune rune rule (cdr lst))))))
; We massage the hyps with this function to speed rewrite up.
(defun preprocess-hyp (hyp wrld)
; In Nqthm, this function also replaced (not (zerop x)) by
; ((numberp x) (not (equal x '0))).
; Lemma replace-consts-cp-correct1 in community book
; books/clause-processors/replace-defined-consts.lisp failed after we added
; calls of mv-list to the macroexpansion of mv-let calls in Version_4.0, which
; allowed lemma replace-const-corr-replace-const-alists-list to be applied:
; there was a free variable in the hypothesis that had no longer been matched
; when mv-list was introduced. So we have decided to add the calls of
; remove-guard-holders below to take care of such issues.
(case-match hyp
(('atom x)
(list (mcons-term* 'not
(mcons-term* 'consp
(remove-guard-holders x wrld)))))
(& (list (remove-guard-holders hyp wrld)))))
(defun preprocess-hyps (hyps wrld)
(cond ((null hyps) nil)
(t (append (preprocess-hyp (car hyps) wrld)
(preprocess-hyps (cdr hyps) wrld)))))
(defun add-definition-rule-with-ttree (rune nume clique controller-alist
install-body term ens wrld ttree)
; We make a :rewrite rule of subtype 'definition (or 'abbreviation)
; and add it to the 'lemmas property of the appropriate fn. This
; function is defined the way it is (namely, taking term as an arg and
; destructuring it rather than just taking term in pieces) because it
; is also used as the function for adding a user-supplied :REWRITE
; rule of subclass :DEFINITION.
(mv-let
(hyps equiv fn args body ttree)
(destructure-definition term install-body ens wrld ttree)
(let* ((vars-bag (all-vars-bag-lst args nil))
(abbreviationp (and (null hyps)
(null clique)
; Rockwell Addition: We have changed the notion of when a rule is an
; abbreviation. Our new concern is with stobjs and lambdas.
; If fn returns a stobj, we don't consider it an abbreviation unless
; it contains no lambdas. Thus, the updaters are abbreviations but
; lambda-nests built out of them are not. We once tried the idea of
; letting a lambda in a function body disqualify the function as an
; abbreviation, but that made FLOOR no longer an abbreviation and some
; of the fp proofs failed. So we made the question depend on stobjs
; for compatibility's sake.
(abbreviationp
(not (all-nils-or-dfs
; We call getprop rather than calling stobjs-out, because this code may run
; with fn = return-last, and the function stobjs-out causes an error in that
; case. We don't mind treating return-last as an ordinary function here.
(getpropc fn 'stobjs-out '(nil) wrld)))
vars-bag
body)))
(rule
(make rewrite-rule
:rune rune
:nume nume
:hyps (preprocess-hyps hyps wrld)
:equiv equiv
:lhs (mcons-term fn args)
:var-info (cond (abbreviationp (not (null vars-bag)))
(t (var-counts args body)))
:rhs body
:subclass (cond (abbreviationp 'abbreviation)
(t 'definition))
:heuristic-info
(cond (abbreviationp nil)
(t (cons clique controller-alist)))
; Backchain-limit-lst does not make much sense for definitions.
:backchain-limit-lst nil)))
(let ((wrld0 (if (eq fn 'hide)
wrld
(putprop fn 'lemmas
(cons rule (getpropc fn 'lemmas nil wrld))
wrld))))
(cond (install-body
(mv (putprop fn
'def-bodies
(cons (make def-body
:nume nume
:hyp (and hyps (conjoin hyps))
:concl body
:equiv equiv
:rune rune
:formals args
:recursivep clique
:controller-alist controller-alist)
(getpropc fn 'def-bodies nil wrld))
wrld0)
ttree))
(t (mv wrld0 ttree)))))))
(defun add-definition-rule (rune nume clique controller-alist install-body term
ens wrld)
(mv-let (wrld ttree)
(add-definition-rule-with-ttree rune nume clique controller-alist
install-body term ens wrld nil)
(declare (ignore ttree))
wrld))
#+:non-standard-analysis
(defun listof-standardp-macro (lst)
; If the guard for standardp is changed from t, consider changing
; the corresponding calls of mcons-term* to fcons-term*.
(if (consp lst)
(if (consp (cdr lst))
(mcons-term*
'if
(mcons-term* 'standardp (car lst))
(listof-standardp-macro (cdr lst))
*nil*)
(mcons-term* 'standardp (car lst)))
*t*))
(defun putprop-body-lst (names arglists bodies normalizeps
clique controller-alist
#+:non-standard-analysis std-p
ens wrld installed-wrld ttree)
; Rockwell Addition: A major change is the handling of PROG2$ and THE
; below.
; We store the body property for each name in names. It is set to the
; normalized body. Normalization expands some nonrecursive functions, namely
; those on *expandable-boot-strap-non-rec-fns*, which includes old favorites
; like EQ and ATOM. In addition, we eliminate all the RETURN-LASTs and THEs
; from the body. This can be seen as just an optimization of expanding nonrec
; fns.
; We add a definition rule equating the call of name with its normalized body.
; We store the unnormalized body under the property 'unnormalized-body.
; We return two results: the final wrld and a ttree justifying the
; normalization, which is an extension of the input ttree.
; Essay on the Normalization of Bodies
; We normalize the bodies of functions to speed up type-set and rewriting. But
; there are some subtle issues here. Let term be a term and let term' be its
; normalization. We will ignore iff-flg and type-alist here. First, we claim
; that term and term' are equivalent. Thus, if we are allowed to add the axiom
; (fn x) = term then we may add (fn x) = term' too. But while term and term'
; are equivalent they are not interchangeable from the perspective of defun
; processing. For example, as Nqthm taught us, the measure conjectures
; generated from term' may be inadequate to justify the admission of a function
; whose body is term. A classic example is (fn x) = (if (fn x) t t), where the
; normalized body is just t. The Historical Plaque below contains a proof that
; if (fn x) = term' is admissible then there exists one and only one function
; satisfying (fn x) = term. Thus, while the latter definition may not actually
; be admissible it at least will not get us into trouble and in the end the
; issue vis-a-vis admissibility seems to be the technical one of exactly how we
; wish to define the Principle of Definition.
; Historical Plaque from Nqthm
; The following extensive comment used to guard the definition of
; DEFN0 in Nqthm and is placed here partly as a nostalgic reminder of
; decades of work and partly because it has some good statistics in it
; that we might still want to look at.
; This function is FUNCALLed and therefore may not be made a MACRO.
; The list of comments on this function do not necessarily describe
; the code below. They have been left around in reverse chronology
; order to remind us of the various combinations of preprocessing
; we have tried.
; If we ever get blown out of the water while normalizing IFs in a
; large defn, read the following comment before abandoning
; normalization.
; 18 August 1982. Here we go again! At the time of this writing
; the preprocessing of defns is as follows, we compute the
; induction and type info on the translated body and store under
; sdefn the translated body. This seems to slow down the system a
; lot and we are going to change it so that we store under sdefn
; the result of expanding boot strap nonrec fns and normalizing
; IFs. As nearly as we can tell from the comments below, we have
; not previously tried this. According to the record, we have
; tried expanding all nonrec fns, and we have tried expanding boot
; strap fns and doing a little normalization. The data that
; suggests this will speed things up is as follows. Consider the
; first call of SIMPLIFY-CLAUSE in the proof of PRIME-LIST-TIMES
; -LIST. The first three literals are trivial but the fourth call
; of SIMPLIFY-CLAUSE1 is on (NOT (PRIME1 C (SUB1 C))). With SDEFNs
; not expanded and normalized -- i.e., under the processing as it
; was immediately before the current change -- there are 2478 calls
; of REWRITE and 273 calls of RELIEVE-HYPS for this literal. With
; all defns preprocessed as described here those counts drop to
; 1218 and 174. On a sample of four theorems, PRIME-LIST-TIMES-
; LIST, PRIME-LIST-PRIME-FACTORS, FALSIFY1-FALSIFIES, and ORDERED-
; SORT, the use of normalized and expanded sdefns saves us 16
; percent of the conses over the use of untouched sdefns, reducing
; the cons counts for those theorems from 880K to 745K. It seems
; unlikely that this preprocessing will blow us out of the water on
; large defns. For the EV used in UNSOLV and for the 386L M with
; subroutine call this new preprocessing only marginally increases
; the size of the sdefn. It would be interesting to see a function
; that blows us out of the water. When one is found perhaps the
; right thing to do is to so preprocess small defns and leave big
; ones alone.
; 17 December 1981. Henceforth we will assume that the very body
; the user supplies (modulo translation) is the body that the
; theorem-prover uses to establish that there is one and only one
; function satisfying the definition equation by determining that
; the given body provides a method for computing just that
; function. This prohibits our "improving" the body of definitions
; such as (f x) = (if (f x) a a) to (f x) = a.
; 18 November 1981. We are sick of having to disable nonrec fns in
; order to get large fns processed, e.g., the interpreter for our
; 386L class. Thus, we have decided to adopt the policy of not
; touching the user's typein except to TRANSLATE! it. The
; induction and type analysis as well as the final SDEFN are based
; on the translated typein.
; Before settling with the preprocessing used below we tried
; several different combinations and did provealls. The main issue
; was whether we should normalize sdefns. Unfortunately, the
; incorporation of META0-LEMMAS was also being experimented with,
; and so we do not have a precise breakdown of who is responsible
; for what. However, below we give the total stats for three
; separate provealls. The first, called 1PROVEALL, contained
; exactly the code below -- except that the ADD-DCELL was given the
; SDEFN with all the fn names replaced by *1*Fns instead of a fancy
; TRANSLATE-TO-INTERLISP call. Here are the 1PROVEALL stats.
; Elapsed time = 9532.957, CPU time = 4513.88, GC time = 1423.261,
; IO time = 499.894, CONSes consumed = 6331517.
; We then incorporated META0-LEMMAS. Simultaneously, we tried
; running the RUN fns through DEFN and found that we exploded. The
; expansion of nonrec fns and the normalization of IFs before the
; induction analysis transformed functions of CONS-COUNT 300 to
; functions of CONS-COUNT exceeding 18K. We therefore decided to
; expand only BOOT-STRAP fns -- and not NORMALIZE-IFS for the
; purposes of induction analysis. After the induction and type
; analyses were done, we put down an SDEFN with some trivial IF
; simplification performed -- e.g., IF X Y Y => Y and IF bool T F
; => bool -- but not a NORMALIZE-IFs version. We then ran a
; proveall with CANCEL around as a META0-LEMMA. The result was
; about 20 percent slower than the 1PROVEALL and used 15 percent
; more CONSes. At first this was attributed to CANCEL. However,
; we then ran two simultaneous provealls, one with META0-LEMMAS set
; to NIL and one with it set to ((1CANCEL . CORRECTNESS-OF-CANCEL)).
; The result was that the version with CANCEL available used
; slightly fewer CONSes than the other one -- 7303311 to 7312505
; That was surprising because the implementation of META0-LEMMAS
; uses no CONSes if no META0-LEMMAS are available, so the entire 15
; percent more CONSes had to be attributed to the difference in the
; defn processing. This simultaneous run was interesting for two
; other reasons. The times -- while still 20 percent worse than
; 1PROVEALL -- were one half of one percent different, with CANCEL
; being the slower. That means having CANCEL around does not cost
; much at all -- and the figures are significant despite the slop
; in the operating system's timing due to thrashing because the two
; jobs really were running simultaneously. The second interesting
; fact is that CANCEL can be expected to save us a few CONSes
; rather than cost us.
; We therefore decided to return the DEFN0 processing to its
; original state. Only we did it in two steps. First, we put
; NORMALIZE-IFs into the pre-induction processing and into the
; final SDEFN processing. Here are the stats on the resulting
; proveall, which was called PROVEALL-WITH-NORM-AND-CANCEL but not
; saved. Elapsed time = 14594.01, CPU time = 5024.387, GC time =
; 1519.932, IO time = 593.625, CONSes consumed = 6762620.
; While an improvement, we were still 6 percent worse than
; 1PROVEALL on CONSes. But the only difference between 1PROVEALL
; and PROVEALL-WITH-NORM-AND-CANCEL -- if you discount CANCEL which
; we rightly believed was paying for itself -- was that in the
; former induction analyses and type prescriptions were being
; computed from fully expanded bodies while in the latter they were
; computed from only BOOT-STRAP-expanded bodies. We did not
; believe that would make a difference of over 400,000 CONSes, but
; had nothing else to believe. So we went to the current state,
; where we do the induction and type analyses on the fully expanded
; and normalized bodies -- bodies that blow us out of the water on
; some of the RUN fns. Here are the stats for
; PROVEALL-PROOFS.79101, which was the proveall for that version.
; Elapsed time = 21589.84, CPU time = 4870.231, GC time = 1512.813,
; IO time = 554.292, CONSes consumed= 6356282.
; Note that we are within 25K of the number of CONSes used by
; 1PROVEALL. But to TRANSLATE-TO-INTERLISP all of the defns in
; question costs 45K. So -- as expected -- CANCEL actually saved
; us a few CONSes by shortening proofs. It takes only 18 seconds
; to TRANSLATE-TO-INTERLISP the defns, so a similar argument does
; not explain why the latter proveall is 360 seconds slower than
; 1PROVEALL. But since the elapsed time is over twice as long, we
; believe it is fair to chalk that time up to the usual slop
; involved in measuring cpu time on a time sharing system.
; We now explain the formal justification of the processing we do
; on the body before testing it for admissibility.
; We do not work with the body that is typed in by the user but
; with an equivalent body' produced by normalization and the
; expansion of nonrecursive function calls in body. We now prove
; that if (under no assumptions about NAME except that it is a
; function symbol of the correct arity) (a) body is equivalent to
; body' and (b) (name . args) = body' is accepted under our
; principle of definition, then there exists exactly one function
; satisfying the original equation (name . args) = body.
; First observe that since the definition (name . args) = body' is
; accepted by our principle of definition, there exists a function
; satisfying that equation. But the accepted equation is
; equivalent to the equation (name . args) = body by the
; hypothesis that body is equivalent to body'.
; We prove that there is only one such function by induction.
; Assume that the definition (name . args) = body has been accepted
; under the principle of definition. Suppose that f is a new name
; and that (f . args) = bodyf, where bodyf results from replacing
; every use of name as a function symbol in body with f. It
; follows that (f . args) = bodyf', where bodyf' results from
; replacing every use of name as a function symbol in body' with f.
; We can now easily prove that (f . args) = (name . args) by
; induction according to the definition of name. Q.E.D.
; One might be tempted to think that if the defn with body' is
; accepted under the principle of definition then so would be the
; defn with body and that the use of body' was merely to make the
; implementation of the defn principle more powerful. This is not
; the case. For example
; (R X) = (IF (R X) T T)
; is not accepted by the definitional principle, but we would
; accept the body'-version (R X) = T, and by our proof, that
; function uniquely satisfies the equation the user typed in.
; One might be further tempted to think that if we changed
; normalize so that (IF X Y Y) = Y was not applied, then the two
; versions were inter-acceptable under the defn principle. This is
; not the case either. The function
; (F X) = (IF (IF (X.ne.0) (F X-1) F) (F X-1) T)
; is not accepted under the principle of defn. Consider its
; normalized body.
(cond ((null names) (mv wrld ttree))
(t (let* ((fn (car names))
(args (car arglists))
(body (car bodies))
(normalizep (car normalizeps))
(rune (fn-rune-nume fn nil nil installed-wrld))
(nume (fn-rune-nume fn t nil installed-wrld)))
(let* ((eqterm (fcons-term* 'equal
(fcons-term fn args)
body))
(term #+:non-standard-analysis
(if (and std-p (consp args))
(fcons-term*
'implies
(listof-standardp-macro args)
eqterm)
eqterm)
#-:non-standard-analysis
eqterm)
#+:non-standard-analysis
(wrld (if std-p
(putprop fn 'constrainedp t
(putprop fn 'constraint-lst (list term) wrld))
wrld)))
(mv-let
(wrld ttree)
(add-definition-rule-with-ttree
rune nume clique controller-alist
(if normalizep :NORMALIZE t) ; install-body
term ens
(putprop fn
'unnormalized-body
body
wrld)
ttree)
(putprop-body-lst (cdr names)
(cdr arglists)
(cdr bodies)
(cdr normalizeps)
clique controller-alist
#+:non-standard-analysis std-p
ens
wrld installed-wrld ttree)))))))
; We now develop the facility for guessing the type-prescription of a defuned
; function. When guards were part of the logic, the first step was to guess
; the types implied by the guard. We no longer have to do that, but the
; utility written for it is used elsewhere and so we keep it here.
; Suppose you are trying to determine the type implied by term for some
; variable x. The key trick is to normalize the term and replace every true
; output by x and every nil output by a term with an empty type-set. Then take
; the type of that term. For example, if term is (if (if p q) r nil) then it
; normalizes to (if p (if q (if r t nil) nil) nil) and so produces the
; intermediate term (if p (if q (if r x e ) e ) e ), where x is the formal in
; whose type we are interested and e is a new variable assumed to be of empty
; type.
(defun type-set-implied-by-term1 (term tvar fvar)
; Term is a normalized propositional term. Tvar and fvar are two variable
; symbols. We return a normalized term equivalent to (if term tvar fvar)
; except we drive tvar and fvar as deeply into term as possible.
(cond ((variablep term)
(fcons-term* 'if term tvar fvar))
((fquotep term)
(if (equal term *nil*) fvar tvar))
((eq (ffn-symb term) 'if)
(fcons-term* 'if
(fargn term 1)
(type-set-implied-by-term1 (fargn term 2) tvar fvar)
(type-set-implied-by-term1 (fargn term 3) tvar fvar)))
(t
; We handle all non-IF applications here, even lambda applications.
; Once upon a time we considered driving into the body of a lambda.
; But that introduces a free var in the body, namely fvar (or whatever
; the new variable symbol is) and there are no guarantees that type-set
; works on such a non-term.
(fcons-term* 'if term tvar fvar))))
(defun type-set-implied-by-term (var not-flg term ens wrld ttree)
; Given a variable and a term, we determine a type set for the
; variable under the assumption that the term is non-nil. If not-flg
; is t, we negate term before using it. This function is not used in
; the guard processing but is needed in the compound-recognizer work.
; The ttree returned is 'assumption-free (provided the initial ttree
; is also).
(let* ((new-var (genvar 'genvar "EMPTY" nil (all-vars term)))
(type-alist (list (list* new-var *ts-empty* nil))))
(mv-let (normal-term ttree)
(normalize term t nil ens wrld ttree
(backchain-limit wrld :ts))
(type-set
(type-set-implied-by-term1 normal-term
(if not-flg new-var var)
(if not-flg var new-var))
nil nil type-alist ens wrld ttree nil nil))))
(defun putprop-initial-type-prescriptions (names wrld)
; Suppose we have a clique of mutually recursive fns, names. Suppose
; that we can recover from wrld both the formals and body of each
; name in names.
; This function adds to the front of each 'type-prescriptions property
; of the names in names an initial, empty guess at its
; type-prescription. These initial rules are unsound and are only the
; starting point of our iterative guessing mechanism. Oddly, the
; :rune and :nume of each rule is the same! We use the
; *fake-rune-for-anonymous-enabled-rule* for the rune and the nume
; nil. We could create the proper runes and numes (indeed, we did at
; one time) but those runes then find their way into the ttrees of the
; various guesses (and not just the rune of the function being typed
; but also the runes of its clique-mates). By adopting this fake
; rune, we prevent that.
; The :term and :hyps we create for each rule are appropriate and survive into
; the final, accurate guess. But the :basic-ts and :vars fields are initially
; empty here and are filled out by the iteration.
(cond
((null names) wrld)
(t (let ((fn (car names)))
(putprop-initial-type-prescriptions
(cdr names)
(putprop fn
'type-prescriptions
(cons (make type-prescription
:rune *fake-rune-for-anonymous-enabled-rule*
:nume nil
:term (mcons-term fn (formals fn wrld))
:hyps nil
:backchain-limit-lst nil
:basic-ts *ts-empty*
:vars nil
:corollary *t*)
(getpropc fn 'type-prescriptions nil wrld))
wrld))))))
; We now turn to the problem of iteratively guessing new
; type-prescriptions. The root of this guessing process is the
; computation of the type-set and formals returned by a term.
(defun map-returned-formals-via-formals (formals pockets returned-formals)
; Formals is the formals list of a lambda expression, (lambda formals
; body). Pockets is a list in 1:1 correspondence with formals. Each
; pocket in pockets is a set of vars. Finally, returned-formals is a
; subset of formals. We return the set of vars obtained by unioning
; together the vars in those pockets corresponding to those in
; returned-formals.
; This odd little function is used to help determine the returned
; formals of a function defined in terms of a lambda-expression.
; Suppose foo is defined in terms of ((lambda formals body) arg1 ...
; argn) and we wish to determine the returned formals of that
; expression. We first determine the returned formals in each of the
; argi. That produces our pockets. Then we determine the returned
; formals of body -- note however that the formals returned by body
; are not the formals of foo but the formals of the lambda. The
; returned formals of body are our returned-formals. This function
; can then be used to convert the returned formals of body into
; returned formals of foo.
(cond ((null formals) nil)
((member-eq (car formals) returned-formals)
(union-eq (car pockets)
(map-returned-formals-via-formals (cdr formals)
(cdr pockets)
returned-formals)))
(t (map-returned-formals-via-formals (cdr formals)
(cdr pockets)
returned-formals))))
(defun map-type-sets-via-formals (formals ts-lst returned-formals)
; This is just like the function above except instead of dealing with
; a list of lists which are unioned together we deal with a list of
; type-sets which are ts-unioned.
(cond ((null formals) *ts-empty*)
((member-eq (car formals) returned-formals)
(ts-union (car ts-lst)
(map-type-sets-via-formals (cdr formals)
(cdr ts-lst)
returned-formals)))
(t (map-type-sets-via-formals (cdr formals)
(cdr ts-lst)
returned-formals))))
(defun vector-ts-union (ts-lst1 ts-lst2)
; Given two lists of type-sets of equal lengths we ts-union
; corresponding elements and return the resulting list.
(cond ((null ts-lst1) nil)
(t (cons (ts-union (car ts-lst1) (car ts-lst2))
(vector-ts-union (cdr ts-lst1) (cdr ts-lst2))))))
(defun map-cons-tag-trees (lst ttree)
; Cons-tag-tree every element of lst into ttree.
(cond ((null lst) ttree)
(t (map-cons-tag-trees
(cdr lst)
(cons-tag-trees (car lst) ttree)))))
(defun type-set-and-returned-formals-with-rule1
(alist rule-vars type-alist ens wrld basic-ts vars-ts vars ttree)
; See type-set-with-rule1 for a slightly simpler version of this.
; Note: This function is really just a loop that finishes off the
; computation done by type-set-and-returned-formals-with-rule, below.
; It would be best not to try to understand this function until you
; have read that function and type-set-and-returned-formals.
; Alist maps variables in a type-prescription to terms. The context in which
; those terms occur is described by type-alist. Rule-vars is the list of :vars
; of the rule.
; The last four arguments are accumulators that will become four of the
; answers delivered by type-set-and-returned-formals-with-rule, i.e.,
; a basic-ts, the type-set of a set of vars, the set of vars, and the
; justifying ttree. We assemble these four answers by sweeping over
; alist, considering each var and its image term. If the var is not
; in the rule-vars, we go on. If the var is in the rule-vars, then
; its image is a possible value of the term for which we are computing
; a type-set. If its image is a variable, we accumulate it and its
; type-set into vars and vars-ts. If its image is not a variable, we
; accumulate its type-set into basic-ts.
; The ttree returned is 'assumption-free (provided the initial ttree
; is also).
(cond
((null alist) (mv basic-ts vars-ts vars type-alist ttree))
((member-eq (caar alist) rule-vars)
(mv-let (ts ttree)
(type-set (cdar alist) nil nil type-alist ens wrld ttree nil nil)
(let ((variablep-image (variablep (cdar alist))))
(type-set-and-returned-formals-with-rule1
(cdr alist) rule-vars
type-alist ens wrld
(if variablep-image
basic-ts
(ts-union ts basic-ts))
(if variablep-image
(ts-union ts vars-ts)
vars-ts)
(if variablep-image
(add-to-set-eq (cdar alist) vars)
vars)
ttree))))
(t
(type-set-and-returned-formals-with-rule1
(cdr alist) rule-vars
type-alist ens wrld
basic-ts
vars-ts
vars
ttree))))
(defun type-set-and-returned-formals-with-rule (tp term type-alist ens wrld
ttree)
; This function is patterned after type-set-with-rule, which the
; reader might understand first.
; The ttree returned is 'assumption-free (provided the initial ttree
; and type-alist are also).
(cond
((enabled-numep (access type-prescription tp :nume) ens)
(mv-let
(unify-ans unify-subst)
(one-way-unify (access type-prescription tp :term)
term)
(cond
(unify-ans
(with-accumulated-persistence
(access type-prescription tp :rune)
(basic-ts vars-ts vars type-alist ttree)
(not (and (ts= basic-ts *ts-unknown*)
(ts= vars-ts *ts-empty*)
(null vars)))
(let* ((backchain-limit (backchain-limit wrld :ts))
(type-alist (extend-type-alist-with-bindings
unify-subst nil nil type-alist nil ens wrld nil nil
nil backchain-limit)))
(mv-let
(relieve-hyps-ans type-alist ttree)
(type-set-relieve-hyps (access type-prescription tp :rune)
term
(access type-prescription tp :hyps)
(access type-prescription tp
:backchain-limit-lst)
nil
nil
unify-subst
type-alist
nil ens wrld nil ttree
nil nil backchain-limit 1)
(cond
(relieve-hyps-ans
; Subject to the conditions in ttree, we now know that the type set of term is
; either in :basic-ts or else that term is equal to the image under unify-subst
; of some var in the :vars of the rule. Our charter is to return five results:
; basic-ts, vars-ts, vars, type-alist and ttree. We do that with the
; subroutine below. It sweeps over the unify-subst, considering each vi and
; its image, ai. If ai is a variable, then it accumulates ai into the returned
; vars (which is initially nil below) and the type-set of ai into vars-ts
; (which is initially *ts-empty* below). If ai is not a variable, it
; accumulates the type-set of ai into basic-ts (which is initially :basic-ts
; below).
(type-set-and-returned-formals-with-rule1
unify-subst
(access type-prescription tp :vars)
type-alist ens wrld
(access type-prescription tp :basic-ts)
*ts-empty*
nil
(push-lemma
(access type-prescription tp :rune)
ttree)))
(t
; We could not establish the hyps of the rule. Thus, the rule tells us
; nothing about term.
(mv *ts-unknown* *ts-empty* nil type-alist ttree)))))))
(t
; The :term of the rule does not unify with our term.
(mv *ts-unknown* *ts-empty* nil type-alist ttree)))))
(t
; The rule is disabled.
(mv *ts-unknown* *ts-empty* nil type-alist ttree))))
(defun type-set-and-returned-formals-with-rules
(tp-lst term type-alist ens w ts vars-ts vars ttree)
; See type-set-with-rules for a simpler model of this function. We
; try to apply each type-prescription in tp-lst, "conjoining" the
; results into an accumulating type-set, ts, and vars (and its
; associated type-set, vars-ts). However, if a rule fails to change
; the accumulating answers, we ignore it.
; However, we cannot really conjoin two type-prescriptions and get a
; third. We do, however, deduce a valid conclusion. A rule
; essentially gives us a conclusion of the form (or basic-ts
; var-equations), where basic-ts is the proposition that the term is
; of one of several given types and var-equations is the proposition
; that the term is one of several given vars. Two rules therefore
; tell us (or basic-ts1 var-equations1) and (or basic-ts2
; var-equations2). Both of these propositions are true. From them we
; deduce the truth
; (or (and basic-ts1 basic-ts2)
; (or var-equations1 var-equations2)).
; Note that we conjoin the basic type-sets but we disjoin the vars. The
; validity of this conclusion follows from the tautology
; (implies (and (or basic-ts1 var-equations1)
; (or basic-ts2 var-equations2))
; (or (and basic-ts1 basic-ts2)
; (or var-equations1 var-equations2))).
; It would be nice if we could conjoin both sides, but that's not valid.
; Recall that we actually must also return the union of the type-sets
; of the returned vars. Since the "conjunction" of two rules leads us
; to union the vars together we just union their types together too.
; The ttree returned is 'assumption free provided the initial ttree and
; type-alist are also.
(cond
((null tp-lst)
(mv-let
(ts1 ttree1)
(type-set term nil nil type-alist ens w ttree nil nil)
(let ((ts2 (ts-intersection ts1 ts)))
(mv ts2 vars-ts vars (if (ts= ts2 ts) ttree ttree1)))))
(t (mv-let
(ts1 vars-ts1 vars1 type-alist1 ttree1)
(type-set-and-returned-formals-with-rule (car tp-lst) term
type-alist ens w ttree)
(let* ((ts2 (ts-intersection ts1 ts))
(unchangedp (and (ts= ts2 ts)
(equal type-alist type-alist1))))
; If the type-set established by the new rule doesn't change (i.e.,
; narrow) what we already know, we simply choose to ignore the new
; rule. If it does change, then ts2 is smaller and we have to union
; together what we know about the vars and report the bigger ttree.
(type-set-and-returned-formals-with-rules
(cdr tp-lst)
term type-alist1 ens w
ts2
(if unchangedp
vars-ts
(ts-union vars-ts1 vars-ts))
(if unchangedp
vars
(union-eq vars1 vars))
(if unchangedp
ttree
ttree1)))))))
(mutual-recursion
(defun type-set-and-returned-formals (term type-alist ens wrld ttree)
; Term is the if-normalized body of a defined function. The
; 'type-prescriptions property of that fn (and all of its peers in its mutually
; recursive clique) may or may not be nil. If non-nil, it may contain many
; enabled rules. (When guards were part of the logic, we computed the type-set
; of a newly defined function twice, once before and once after verifying its
; guards. So during the second pass, a valid rule was present.) Among the
; rules is one that is possibly unsound and represents our current guess at the
; type. We compute, from that guess, a "basic type-set" for term and a list of
; formals that might be returned by term. An odd aspect of this ttree is that
; it will probably include the rune of the very rule we are trying to create,
; since its use in this process is essentially as an induction hypothesis.
; This function returns four results. The first is the basic type-set
; computed, the second is another type-set which we call the variables
; type-set, and the third is the set of returned formals. Those three results
; satisfy the claim made below. Informally, we view the second result as the
; union of the type-sets of the returned formals. The fourth result is a ttree
; justifying all the type-set reasoning done so far, accumulated onto the
; initial ttree. The ttree returned is 'assumption-free provided the initial
; ttree and type-alist are also.
; The function works by walking through the if structure of the body, using the
; normal assume-true-false to construct the governing type-alist for each
; branch. Upon arriving at a leaf term we compute the result. If the term is
; a quote or a call to an ACL2 primitive, we just use type-set. If the term is
; a call of a defun'd function, we interpret its type-prescription.
; CLAIM. Consider an input term and type-alist, as well as the return values
; (mv bts vts vars ttree). Assume that the variables of term satisfy the
; requirements of the type-alist. Then either term satisfies the type-set bts,
; or else term is equal to some v in vars, in which case term satisfies vts.
; (Moreover, the returned ttree is an extension of the input ttree that
; justifies these conclusions, but we do not remark further on that ttree.)
; (Remark. If we care to be a bit pedantic, then we may formulate the Claim in
; terms of first-order logic as follows. A type-alist ta and a type-set ts
; naturally give rise to first-order formulas phi_ta(x1,...xn) and phi_ts(x),
; respectively. The Claim says that the following is a theorem of the theory
; corresponding to the given world, wrld, where the given term has variables
; x1, ..., xn, vars is {v1,...,vk}, and ta is the given type-alist:
; phi_ta(x1,...,xn)
; ->
; phi_bts(term) \/ (phi_vts(term) & (term = v1 \/ ... \/ term = vk)).
; End of Remark.)
; Historical Plaque from Nqthm.
; In Nqthm, the root of the guessing process was DEFN-TYPE-SET, which was
; mutually recursive with DEFN-ASSUME-TRUE-FALSE. The following comment could
; be found at the entrance to the guessing process:
; *************************************************************
; THIS FUNCTION WILL BE COMPLETELY UNSOUND IF TYPE-SET IS EVER
; REACHABLE FROM WITHIN IT. IN PARTICULAR, BOTH THE TYPE-ALIST AND
; THE TYPE-PRESCRIPTION FOR THE FN BEING PROCESSED ARE SET TO ONLY
; PARTIALLY ACCURATE VALUES AS THIS FN COMPUTES THE REAL TYPE-SET.
; *************************************************************
; We now believe that this dreadful warning is an overstatement of the case.
; It is true that in Nqthm the type-alist used in DEFN-TYPE-SET would cause
; trouble if it found its way into TYPE-SET, because it bound vars to "defn
; type-sets" (pairs of type-sets and variables) instead of to type-sets. But
; the fear of the inaccurate TYPE-PRESCRIPTIONs above is misplaced we think.
; We believe that if one guesses a type-prescription and then confirms that it
; accurately describes the function body, then the type-prescription is
; correct. Therefore, in ACL2, far from fencing type-set away from
; "defun-type-set" we use it explicitly. This has the wonderful advantage that
; we do not duplicate the type-set code (which is even worse in ACL2 than it
; was in Nqthm).
(cond
((variablep term)
; Consider the following historical comment.
; Term is a formal variable. We compute its type-set under
; type-alist. If it is completely unrestricted, then we will say that
; formal is sometimes returned. Otherwise, we will say that it is not
; returned. Once upon a time we always said it was returned. But the
; term (if (integerp x) (if (< x 0) (- x) x) 0) as occurs in
; integer-abs, then got the type-set "nonnegative integer or x" which
; meant that it effectively had the type-set unknown.
; Thus, in Version_8.0 and many preceding versions, for the variable, term, we
; heuristically returned (mv *ts-empty* ts (list term) ttree) when the type of
; term was completely unrestricted (i.e., equal to *ts-unknown*), and otherwise
; we returned (mv ts *ts-empty* nil ttree). We then realized that we could
; defer this choice until popping up to the top level (by introducing function
; type-set-and-returned-formals-top), which strengthened the resulting type
; prescription in some cases. Here is an example, where previously the
; generated type-prescription was trivial -- :args foo reported that the Type
; is "built-in (or unrestricted)" -- but now it's as expected -- the reported
; Type is (EQUAL (F1 X Y) X).
; (defund f1 (x y)
; (if (integerp x)
; (if (consp y)
; (f1 x (cdr y))
; x)
; x))
; Observe that the code below satisfies our Claim, since if term satisfies the
; requirements of type-alist, then it satisfies the computed type-set, by
; correctness of the function, type-set.
(mv-let (ts ttree)
(type-set term nil nil type-alist ens wrld ttree nil nil)
(mv *ts-empty* ts (list term) ttree)))
((fquotep term)
; Term is a constant. We return a basic type-set consisting of the type-set of
; term. Our Claim holds, again by correctness of type-set.
(mv-let (ts ttree)
(type-set term nil nil type-alist ens wrld ttree nil nil)
(mv ts *ts-empty* nil ttree)))
((flambda-applicationp term)
; Let term be ((lambda (...u...) body) ...arg...). Let the formals in term be
; x1, ..., xn. We compute a basic type-set, bts, some returned vars, vars, and
; a variable type-set, vts, for a lambda application as follows.
; (1) For each argument, arg, obtain bts-arg, vts-arg, and vars-arg, which are
; the basic type-set, the variable type-set, and the returned variables with
; respect to the given type-alist.
; (2) Build a new type-alist, type-alist-body, by binding the formals of the
; lambda, (...u...), to the types of its arguments (...arg...) computed with
; respect to the given type-alist.
; (3) Obtain bts-body, vts-body, and vars-body, by recursively processing body
; under type-alist-body.
; (4) Create a preliminary bts by unioning bts-body and those of the bts-args
; in positions that are sometimes returned, as specified by vars-body.
; (5) Create the final vars by unioning together those of the vars-args in
; positions that are sometimes returned, as specified by vars-body.
; (6) Union together the variable type-sets computed for those final vars with
; respect to type-alist, to create a preliminary vts.
; (7) Create the final bts and vts by intersecting each of the preliminary bts
; and vts (from (4) and (6)) with the union of bts-body and vts-body.
; We prove the Claim by induction on the term. We may assume:
; (a) The hypothesis of the Claim holds for the given term and type-alist: that
; is, the variables of term satisfy the requirements of type-alist.
; For convenience we assume that the formals of the lambda are disjoint from
; the free variables of the given term; otherwise, just rename them and observe
; that the results are unchanged by that renaming. Now consider the following
; assertion.
; (*) Each free variable u of body is equal to the corresponding actual, arg.
; By beta-reduction, term is provably equal to body/s where s is the
; substitution mapping each formal u of the lambda to the corresponding actual,
; arg. With that motivation, we will feel free below to view term as equal to
; body under the assumption, (*), since of course the following is valid for
; any property P: (forall (u) (u=arg -> P(body))) <-> P(body/s).
; By (a), each free variable xi of term satisfies its type-set computed from
; the given type-alist. Therefore the type-sets are valid that are computed
; for each argument, arg, with respect to that type-alist. Thus:
; (b) The hypothesis of the Claim holds for body and body-type-alist under the
; assumption (*).
; We apply the inductive hypothesis to (b) and obtain:
; (c) The Claim holds under assumption (*) for the triple calculated for body
; with respect to type-alist-body: bts-body, vts-body, and vars-body.
; It follows from (c) that term satisfies bts-body or vts-body; so the
; intersection in (7) with the union of these two types is harmless.
; Therefore, it suffices to prove the Claim for the bts, vts, and vars computed
; in (4) through (6) above. That is, it suffices to show, assuming validity of
; the given type-alist, that either term satisfies bts, or else term satisfies
; vts and is equal to some variable in vars. It therefore suffices to assume
; (*) and (by (b)) the validity of body-type-alist, and show that either body
; satisfies bts or else body satisfies vts and is equal to some variable in
; vars.
; So assume (*); thus by (c), either body satisfies bts-body, or else body is
; equal to some variable in vars-body. In the former case we are done, so
; assume that body is equal to some variable, u, in vars-body. Then by (*),
; body is equal to the corresponding arg (that is: u is a formal of the lambda
; being applied, and arg is the corresponding actual). By the inductive
; hypothesis together with (4) and (5), either arg has type bts or else arg is
; equal to some variable in vars. Since body is equal to arg, then body has
; type bts or else body is equal to some variable in vars. By (6) and (a), if
; body is equal to some variable in vars then body has type vts, which
; concludes the proof of the Claim.
(mv-let (bts-args vts-args vars-args ttree-args)
(type-set-and-returned-formals-lst (fargs term)
type-alist
ens wrld)
(mv-let (bts-body vts-body vars-body ttree)
(type-set-and-returned-formals
(lambda-body (ffn-symb term))
(zip-variable-type-alist
(lambda-formals (ffn-symb term))
(pairlis$ (vector-ts-union bts-args vts-args)
ttree-args))
ens wrld ttree)
(let* ((bts (ts-union bts-body
(map-type-sets-via-formals
(lambda-formals (ffn-symb term))
bts-args
vars-body)))
(vars (map-returned-formals-via-formals
(lambda-formals (ffn-symb term))
vars-args
vars-body))
(ts-and-ttree-lst
(type-set-lst vars nil nil type-alist nil ens wrld
nil nil (backchain-limit wrld :ts)))
(vts0
; Below we make unconventional use of map-type-sets-via-formals. Its first and
; third arguments are equal and thus every element of its second argument will
; be ts-unioned into the answer. This is just a hackish way to union together
; the type-sets of all the returned formals.
(map-type-sets-via-formals
vars
(strip-cars ts-and-ttree-lst)
vars))
(ts1 (ts-union bts-body vts-body)))
(mv (ts-intersection bts ts1)
(ts-intersection vts0 ts1)
vars
(map-cons-tag-trees (strip-cdrs ts-and-ttree-lst)
ttree))))))
((eq (ffn-symb term) 'if)
; If by type-set reasoning we can see which way the test goes, we can clearly
; focus on that branch. So now we consider (if t1 t2 t3) where we don't know
; which way t1 will go. We compute the union of the respective components of
; the answers for t2 and t3. In general, the type-set of any instance of this
; if will be at most the union of the type-sets of the instances of t2 and t3.
; (In the instance, t1' might be decidable and a smaller type-set could be
; produced.)
(mv-let
(must-be-true
must-be-false
true-type-alist
false-type-alist
ts-ttree)
(assume-true-false (fargn term 1)
nil nil nil type-alist ens wrld
nil nil nil)
; Observe that ts-ttree does not include ttree. If must-be-true and
; must-be-false are both nil, ts-ttree is nil and can thus be ignored.
(cond
(must-be-true
; Probably it would be sound to return (mv *ts-empty* *ts-empty* nil
; (cons-tag-trees ts-ttree ttree)). Since the context is contradictory. But
; this hasn't been an issue as far as we know, so we'll avoid making an
; airtight soundness argument until the need arises.
(type-set-and-returned-formals (fargn term 2)
true-type-alist ens wrld
(cons-tag-trees ts-ttree ttree)))
(must-be-false
(type-set-and-returned-formals (fargn term 3)
false-type-alist ens wrld
(cons-tag-trees ts-ttree ttree)))
(t (mv-let
(basic-ts2 formals-ts2 formals2 ttree)
(type-set-and-returned-formals (fargn term 2)
true-type-alist
ens wrld ttree)
(mv-let
(basic-ts3 formals-ts3 formals3 ttree)
(type-set-and-returned-formals (fargn term 3)
false-type-alist
ens wrld ttree)
(mv (ts-union basic-ts2 basic-ts3)
(ts-union formals-ts2 formals-ts3)
(union-eq formals2 formals3)
ttree)))))))
(t
(let* ((fn (ffn-symb term))
(recog-tuple (most-recent-enabled-recog-tuple fn wrld ens)))
(cond
(recog-tuple
(mv-let (ts ttree1)
(type-set (fargn term 1) nil nil type-alist ens wrld ttree nil
nil)
(mv-let (ts ttree)
(type-set-recognizer recog-tuple ts ttree1 ttree)
(mv ts *ts-empty* nil ttree))))
(t
(type-set-and-returned-formals-with-rules
(getpropc (ffn-symb term) 'type-prescriptions nil wrld)
term type-alist ens wrld
*ts-unknown* *ts-empty* nil ttree)))))))
(defun type-set-and-returned-formals-lst
(lst type-alist ens wrld)
(cond
((null lst) (mv nil nil nil nil))
(t (mv-let (basic-ts returned-formals-ts returned-formals ttree)
(type-set-and-returned-formals (car lst)
type-alist ens wrld nil)
(mv-let (ans1 ans2 ans3 ans4)
(type-set-and-returned-formals-lst (cdr lst)
type-alist
ens wrld)
(mv (cons basic-ts ans1)
(cons returned-formals-ts ans2)
(cons returned-formals ans3)
(cons ttree ans4)))))))
)
(defun type-set-and-returned-formals-top (term ens wrld ttree)
(mv-let (basic-type-set returned-vars-type-set returned-vars ttree)
(type-set-and-returned-formals term nil ens wrld ttree)
(cond ((ts= returned-vars-type-set -1)
(mv basic-type-set returned-vars ttree))
(t
(mv (ts-union basic-type-set returned-vars-type-set)
nil
ttree)))))
(defun guess-type-prescription-for-fn-step (name body ens wrld ttree)
; This function takes one incremental step towards the type- prescription of
; name in wrld. Body is the normalized body of name. We assume that the
; current guess for a type-prescription for name is the car of the
; 'type-prescriptions property. That is, initialization has occurred and every
; iteration keeps the current guess at the front of the list.
; We get the type-set of and formals returned by body. We convert the two
; answers into a new type-prescription and replace the current car of the
; 'type-prescriptions property.
; We return the new world and an 'assumption-free ttree extending ttree.
(let* ((ttree0 ttree)
(old-type-prescriptions
(getpropc name 'type-prescriptions nil wrld))
(tp (car old-type-prescriptions)))
(mv-let (new-basic-type-set new-returned-vars ttree)
(type-set-and-returned-formals-top body ens wrld ttree)
(cond ((ts= new-basic-type-set *ts-unknown*)
; Ultimately we will delete this rule. But at the moment we wish merely to
; avoid contaminating the ttree of the ongoing process by whatever we've
; done to derive this.
(mv (putprop name
'type-prescriptions
(cons (change type-prescription tp
:basic-ts *ts-unknown*
:vars nil)
(cdr old-type-prescriptions))
wrld)
ttree0))
(t
(mv (putprop name
'type-prescriptions
(cons (change type-prescription tp
:basic-ts new-basic-type-set
:vars new-returned-vars)
(cdr old-type-prescriptions))
wrld)
ttree))))))
(defconst *clique-step-install-interval*
; This interval represents how many type prescriptions are computed before
; installing the resulting intermediate world. The value below is merely
; heuristic, chosen with very little testing; we should feel free to change it.
30)
(defun guess-and-putprop-type-prescription-lst-for-clique-step
(names bodies ens wrld ttree interval state)
; Given a list of function names and their normalized bodies
; we take one incremental step toward the final type-prescription of
; each fn in the list. We return a world containing the new
; type-prescription for each fn and a ttree extending ttree.
; Note: During the initial coding of ACL2 the iteration to guess
; type-prescriptions was slightly different from what it is now. Back
; then we used wrld as the world in which we computed all the new
; type-prescriptions. We returned those new type-prescriptions to our
; caller who determined whether the iteration had repeated. If not,
; it installed the new type-prescriptions to generate a new wrld' and
; called us on that wrld'.
; It turns out that that iteration can loop indefinitely. Consider the
; mutually recursive nest of foo and bar where
; (defun foo (x) (if (consp x) (not (bar (cdr x))) t))
; (defun bar (x) (if (consp x) (not (foo (cdr x))) nil))
; Below are the successive type-prescriptions under the old scheme:
; iteration foo type bar type
; 0 {} {}
; 1 {T NIL} {NIL}
; 2 {T} {T NIL}
; 3 {T NIL} {NIL}
; ... ... ...
; Observe that the type of bar in round 1 is incomplete because it is
; based on the incomplete type of foo from round 0. This kind of
; incompleteness is supposed to be closed off by the iteration.
; Indeed, in round 2 bar has got its complete type-set. But the
; incompleteness has now been transferred to foo: the round 2
; type-prescription for foo is based on the incomplete round 1
; type-prescription of bar. Isn't this an elegant example?
; The new iteration computes the type-prescriptions in a strict linear
; order. So that the round 1 type-prescription of bar is based on the
; round 1 type-prescription of foo.
(cond ((null names) (mv wrld ttree state))
(t (mv-let
(erp val state)
(update-w (int= interval 0) wrld)
(declare (ignore erp val))
(mv-let
(wrld ttree)
(guess-type-prescription-for-fn-step
(car names)
(car bodies)
ens wrld ttree)
(guess-and-putprop-type-prescription-lst-for-clique-step
(cdr names)
(cdr bodies)
ens
wrld
ttree
(if (int= interval 0)
*clique-step-install-interval*
(1- interval))
state))))))
(mutual-recursion
(defun guarded-termp (x w)
; We assume that x is a termp in some world whose theory extends that of the
; world, w. Here we check that in addition, x is a termp in w by checking that
; all function symbols are defined in w.
(declare (xargs :guard (and (pseudo-termp x)
(plist-worldp w))))
(cond ((atom x) t)
((eq (car x) 'quote)
t)
((not (mbt (true-listp x))) nil)
((not (mbt (pseudo-term-listp (cdr x)))) nil)
(t (if (symbolp (car x))
(not (eq (getpropc (car x) 'formals t w) t))
(and (guarded-termp (caddr (car x)) w)
(guarded-term-listp (cdr x) w))))))
(defun guarded-term-listp (lst w)
(declare (xargs :guard (and (pseudo-term-listp lst)
(plist-worldp w))))
(cond ((endp lst) (equal lst nil))
(t (and (guarded-termp (car lst) w)
(guarded-term-listp (cdr lst) w)))))
)
(defun conjoin-type-prescriptions (tp1 tp2 ens wrld)
; Tp1 and tp2 are each a runic type-prescription record or an atom,
; representing the unknown type-prescription. However, tp2 need not have a
; valid :corollary field.
; If tp1 and tp2 are both records, then they both have the same rune. If tp1
; is an atom, it is nil; if tp2 is an atom, it is a nume. We return either nil
; or a type-prescription record implied by the conjunction of tp1 and tp2. If
; tp1 is a record then it was supplied by some cert-data. If tp2 is a record
; then it is a valid type prescription in wrld (other than perhaps the
; :corollary field, as noted above); thus, we prefer to return a modification
; of tp2 rather than of tp1, so that we don't have to change the :nume field.
(cond
((null tp1)
(cond
((consp tp2)
(mv-let (corollary ttree)
(convert-type-prescription-to-term tp2 ens wrld)
(mv (change type-prescription tp2
:corollary corollary)
ttree)))
(t (mv nil nil))))
(t ; tp1 is a runic type-prescription record
(assert$
(and (null (access type-prescription tp1 :hyps))
(null (access type-prescription tp1 :backchain-limit-lst)))
(cond
((atom tp2) ; tp2 is a nume
(cond
((guarded-termp (access type-prescription tp1 :corollary)
wrld)
(mv (change type-prescription tp1
:nume tp2)
(push-lemma *fake-rune-for-cert-data* nil)))
(t
(mv-let (corollary ttree)
(convert-type-prescription-to-term tp1 ens wrld)
(mv (change type-prescription tp1
:nume tp2
:corollary corollary)
(push-lemma *fake-rune-for-cert-data* ttree))))))
(t ; tp1 and tp2 are both runic type-prescription records
(assert$
; Sanity check that both tp1 and tp2 are runic type-prescriptions for the same
; function symbol, differing at most in their :basic-ts, :vars, and :corollary.
(and (null (access type-prescription tp2 :hyps))
(null (access type-prescription tp2 :backchain-limit-lst))
(equal (access type-prescription tp1 :term)
(access type-prescription tp2 :term))
(equal (access type-prescription tp1 :rune)
(access type-prescription tp2 :rune)))
(let ((basic-ts1 (access type-prescription tp1 :basic-ts))
(basic-ts2 (access type-prescription tp2 :basic-ts)))
(cond
((and (ts-subsetp basic-ts1 basic-ts2)
(guarded-termp (access type-prescription tp1 :corollary)
wrld)) ; common case
(mv (change type-prescription tp1
:nume (access type-prescription tp2 :nume))
; Even though we are only using tp1 to give us the corollary, that corollary
; could have originally been computed using some rules (in a call of
; convert-type-prescription-to-term), so we credit the cert-data in which that
; computed type-prescription, tp1, was stored.
(push-lemma *fake-rune-for-cert-data* nil)))
((ts-subsetp basic-ts2 basic-ts1) ; tp2 is stronger
(mv-let (corollary ttree)
(convert-type-prescription-to-term tp2 ens wrld)
(mv (change type-prescription tp2
:corollary corollary)
ttree)))
(t ; need to intersect the two :basic-ts fields
(let* ((vars1 (access type-prescription tp1 :vars))
(vars2 (access type-prescription tp2 :vars))
(tp (cond
((equal vars1 vars2)
(change type-prescription tp2
:basic-ts
(ts-intersection basic-ts1 basic-ts2)))
(t
; If the :term is none of the :vars of either tp1 or tp2, then its type must be
; the :basic-ts of tp1 and also the :basic-ts of tp2.
(change type-prescription tp2
:basic-ts
(ts-intersection basic-ts1 basic-ts2)
:vars (union-eq vars1 vars2))))))
(mv-let (corollary ttree)
(convert-type-prescription-to-term tp ens wrld)
(mv (change type-prescription tp
:corollary corollary)
ttree)))))))))))))
(defun cleanse-type-prescriptions (names type-prescriptions-lst def-nume
rmp-cnt ens wrld installed-wrld
cert-data-tp-entry ttree)
; Names is a clique of function symbols. Type-prescriptions-lst is in
; 1:1 correspondence with names and gives the value in wrld of the
; 'type-prescriptions property for each name. (We provide this just
; because our caller happens to be holding it.) This function should
; be called when we have completed the guessing process for the
; type-prescriptions for names. This function does two sanitary
; things: (a) it deletes the guessed rule if its :basic-ts is
; *ts-unknown*, and (b) in the case that the guessed
; rule is kept, it is given the rune and nume described by the Essay
; on the Assignment of Runes and Numes by DEFUNS. It is assumed that
; def-nume is the nume of (:DEFINITION fn), where fn is the car of
; names. We delete *ts-unknown* rules just to save type-set the
; trouble of relieving their hyps or skipping them.
; Rmp-cnt (which stands for "runic-mapping-pairs count") is the length of the
; 'runic-mapping-pairs entry for the functions in names (all of which have the
; same number of mapping pairs). We increment our def-nume by rmp-cnt on each
; iteration.
; cert-data-tp-entry is a cert-data-entry for the key, :type-prescription.
; This function knows that the defun runes for each name are laid out
; as follows, where i is def-nume:
; i (:definition name) ^
; i+1 (:executable-counterpart name)
; i+2 (:type-prescription name) ; rmp-cnt=3 or 4
; i+3 (:induction name) ; optional v
; Furthermore, we know that the nume of the :definition rune for the kth
; (0-based) name in names is def-nume+(k*rmp-cnt); that is, we assigned
; numes to the names in the same order as the names appear in names.
(cond
((null names) (mv wrld ttree))
(t
(let* ((fn (car names))
(lst (car type-prescriptions-lst))
(tp1 (cert-data-val fn cert-data-tp-entry))
(tp2 ; still can have unset :corollary field
(cond
((ts= *ts-unknown* (access type-prescription (car lst)
:basic-ts))
; We bind tp2 to a nume in this case; see conjoin-type-prescriptions.
(+ 2 def-nume))
(t (change type-prescription
(car lst)
:rune (list :type-prescription fn)
:nume (+ 2 def-nume))))))
(mv-let (new-tp ttree1)
(conjoin-type-prescriptions tp1 tp2 ens installed-wrld)
(let ((ttree2 (cons-tag-trees ttree1 ttree)))
(mv-let
(wrld ttree3)
(cond
((null new-tp)
(mv (putprop fn 'type-prescriptions (cdr lst) wrld)
ttree))
(t (mv (putprop fn 'type-prescriptions
(cons new-tp (cdr lst))
wrld)
ttree2)))
(cleanse-type-prescriptions (cdr names)
(cdr type-prescriptions-lst)
(+ rmp-cnt def-nume)
rmp-cnt ens wrld installed-wrld
cert-data-tp-entry ttree3))))))))
(defun guess-and-putprop-type-prescription-lst-for-clique
(names bodies def-nume ens wrld ttree big-mutrec cert-data-tp-entry state)
; We assume that in wrld we find 'type-prescriptions for every fn in
; names. We compute new guesses at the type-prescriptions for each fn
; in names. If they are all the same as the currently stored ones we
; quit. Otherwise, we store the new guesses and iterate. Actually,
; when we quit, we cleanse the 'type-prescriptions as described above.
; We return the final wrld and a ttree extending ttree. Def-nume is
; the nume of (:DEFINITION fn), where fn is the first element of names
; and is used in the cleaning up to install the proper numes in the
; generated rules.
(let ((old-type-prescriptions-lst
(getprop-x-lst names 'type-prescriptions wrld)))
(mv-let (wrld1 ttree state)
(guess-and-putprop-type-prescription-lst-for-clique-step
names bodies ens wrld ttree *clique-step-install-interval* state)
(er-progn
(update-w big-mutrec wrld1)
(cond ((equal old-type-prescriptions-lst
(getprop-x-lst names 'type-prescriptions wrld1))
(mv-let
(wrld2 ttree)
(cleanse-type-prescriptions
names
old-type-prescriptions-lst
def-nume
(length (getpropc (car names) 'runic-mapping-pairs nil
wrld))
ens
wrld
wrld1
cert-data-tp-entry
ttree)
(er-progn
; Warning: Do not use set-w! here, because if we are in the middle of a
; top-level include-book, that will roll the world back to the start of that
; include-book. We have found that re-installing the world omits inclusion of
; the compiled files for subsidiary include-books (see description of bug fix
; in :doc note-2-9 (bug fixes)).
(update-w big-mutrec wrld t)
(update-w big-mutrec wrld2)
(mv wrld2 ttree state))))
(t
(guess-and-putprop-type-prescription-lst-for-clique
names
bodies
def-nume ens wrld1 ttree big-mutrec cert-data-tp-entry
state)))))))
(defun get-normalized-bodies (names wrld)
; Return the normalized bodies for names in wrld.
; WARNING: We ignore the runes and hyps for the normalized bodies returned. So
; this function is probably only of interest when names are being introduced,
; where the 'def-bodies properties have been placed into wrld but no new
; :definition rules with non-nil :install-body fields have been proved for
; names.
(cond ((endp names) nil)
(t (cons (access def-body
(def-body (car names) wrld)
:concl)
(get-normalized-bodies (cdr names) wrld)))))
(defun cert-data-putprop-type-prescription-lst-for-clique
(cert-data-tp-entry names def-nume rmp-cnt ttree ens wrld installed-wrld
changedp)
; Rmp-cnt (which stands for "runic-mapping-pairs count") is the length of the
; 'runic-mapping-pairs entry for the functions in names (all of which have the
; same number of mapping pairs). We increment our def-nume by rmp-cnt on each
; iteration, as is done in cleanse-type-prescriptions. Changedp is initially
; nil, but as we recur it becomes t if we ever extend wrld (with a putprop
; call).
(cond
((endp names)
(mv wrld
(if changedp
(push-lemma *fake-rune-for-cert-data* ttree)
ttree)))
(t (let* ((fn (car names))
(cert-data-pair (cert-data-pair fn cert-data-tp-entry)))
(cond
((null cert-data-pair)
(cert-data-putprop-type-prescription-lst-for-clique
cert-data-tp-entry
(cdr names)
(+ rmp-cnt def-nume)
rmp-cnt
ttree
ens
wrld ; no change; also ok, (putprop fn 'type-prescriptions nil wrld)
installed-wrld
changedp))
(t
(let ((cert-data-tp (cdr cert-data-pair)))
(mv-let (corollary ttree1)
(if (or (null cert-data-tp)
(guarded-termp (access type-prescription cert-data-tp
:corollary)
installed-wrld))
(mv nil nil) ; nothing to do
(convert-type-prescription-to-term cert-data-tp ens
; We use the installed world (the one before cleansing started) for efficient
; handling of large mutual recursion nests.
installed-wrld))
(cert-data-putprop-type-prescription-lst-for-clique
cert-data-tp-entry
(cdr names)
(+ rmp-cnt def-nume)
rmp-cnt
(cons-tag-trees ttree1 ttree)
ens
(putprop fn 'type-prescriptions
(list (if corollary
(change type-prescription cert-data-tp
:nume (+ 2 def-nume)
:corollary corollary)
(change type-prescription cert-data-tp
:nume (+ 2 def-nume))))
wrld)
installed-wrld
t)))))))))
(defun putprop-type-prescription-lst (names subversive-p def-nume ens wrld
ttree state)
; Names is a list of mutually recursive fns being introduced. We assume that
; for each fn in names we can obtain from wrld the 'formals and the normalized
; body (from 'def-bodies). Def-nume must be the nume assigned (:DEFINITION
; fn), where fn is the first element of names. See the Essay on the Assignment
; of Runes and Numes by DEFUNS. We compute type-prescriptions for each fn in
; names and store them. We return the new wrld and a ttree extending ttree
; justifying what we've done.
; This function knows that HIDE should not be given a
; 'type-prescriptions property.
; Historical Plaque for Versions Before 1.8
; In 1.8 we "eliminated guards from the ACL2 logic." Prior to that guards were
; essentially hypotheses on the definitional equations. This complicated many
; things, including the guessing of type-prescriptions. After a function was
; known to be Common Lisp compliant we could recompute its type-prescription
; based on the fact that we knew that every subfunction in it would return its
; "expected" type. Here is a comment from that era, preserved for posterity.
; On Guards: In what way is the computed type-prescription influenced
; by the changing of the 'guards-checked property from nil to t?
; The key is illustrated by the following fact: type-set returns
; *ts-unknown* if called on (+ x y) with gc-flg nil but returns a
; subset of *ts-acl2-number* if called with gc-flg t. To put this into
; context, suppose that the guard for (fn x y) is (g x y) and that it
; is not known by type-set that (g x y) implies that both x and y are
; acl2-numberps. Suppose the body of fn is (+ x y). Then the initial
; type-prescription for fn, computed when the 'guards-checked property
; is nil, will have the basic-type-set *ts-unknown*. After the guards
; have been checked the basic type-set will be *ts-acl2-number*.
(cond
((and (consp names)
(eq (car names) 'hide)
(null (cdr names)))
(mv wrld ttree state))
(subversive-p
; We avoid storing a runic type-prescription rule for any subversive function,
; because our fixed-point algorithm assumes that the definition provably
; terminates, which may not be the case for subversive functions.
; Below is a series of two examples. The first is the simpler of the two, and
; shows the basic problem. It succeeds in Version_3.4.
; (encapsulate
; ()
;
; (defun h (x) (declare (ignore x)) t)
;
; (in-theory (disable (:type-prescription h)))
;
; (local (in-theory (enable (:type-prescription h))))
;
; (encapsulate (((f *) => *))
; (local (defun f (x) (cdr x)))
; (defun g (x)
; (if (consp x) (g (f x)) (h x))))
;
; (defthm atom-g
; (atom (g x))
; :rule-classes nil)
; )
;
; (defthm contradiction nil
; :hints (("Goal" :use ((:instance
; (:functional-instance
; atom-g
; (f identity)
; (g (lambda (x)
; (if (consp x) x t))))
; (x '(a b))))))
; :rule-classes nil)
; Our first solution was to erase type-prescription rules for subversive
; functions after the second pass through the encapsulate. While that dealt
; with the example above -- atom-g was no longer provable -- the problem was
; that the type-prescription rule can be used to normalize a non-subversive
; (indeed, non-recursive) definition later in the same encapsulate, before the
; type-prescription rule has been erased. The second example shows how that
; works:
; (encapsulate
; ()
;
; (defun h (x) (declare (ignore x)) t)
;
; (in-theory (disable (:type-prescription h)))
;
; (local (in-theory (enable (:type-prescription h))))
;
; (encapsulate (((f *) => *))
; (local (defun f (x) (cdr x)))
; (defun g (x)
; (if (consp x) (g (f x)) (h x)))
; (defun k (x)
; (g x)))
;
; ; The following in-theory event is optional; it emphasizes that the problem is
; ; with the use of the bogus type-prescription for g in normalizing the body of
; ; k, not with the direct use of a type-prescription rule in subsequent
; ; proofs.
; (in-theory (disable (:type-prescription k) (:type-prescription g)))
;
; (defthm atom-k
; (atom (k x))
; :rule-classes nil)
; )
;
; (defthm contradiction nil
; :hints (("Goal" :use ((:instance
; (:functional-instance
; atom-k
; (f identity)
; (g (lambda (x)
; (if (consp x) x t)))
; (k (lambda (x)
; (if (consp x) x t))))
; (x '(a b))))))
; :rule-classes nil)
(mv wrld ttree state))
(t
(let* ((cert-data-tp-entry-pair
(cert-data-entry-pair :type-prescription state))
(cert-data-tp-entry
(cdr cert-data-tp-entry-pair))
(cert-data-pass1-saved
(cert-data-entry-pair :pass1-saved state)))
; If cert-data-tp-entry-pair is non-nil then its cdr is a cert-data-entry for
; :type-prescription, i.e., a fast-alist that associates each (function symbol)
; key with a type-prescription record. If moreover cert-data-pass1-saved is
; true, then we are in pass 2 of either encapsulate or certify-book; otherwise
; (still assuming that cert-data-tp-entry-pair is non-nil) we are including a
; certified book. Since redefinition is possible, we avoid the temptation to
; check that no name in names has a non-nil 'type-prescriptions property.
(cond
((and cert-data-tp-entry-pair
; As noted above, the next conjunct says that the cert-data-tp-entry-pair did
; not come from the first pass of either a non-trivial encapsulate or
; certify-book. Thus, we must be including an already-certified book.
(not cert-data-pass1-saved))
(mv-let
(wrld ttree)
(cert-data-putprop-type-prescription-lst-for-clique
cert-data-tp-entry
names
def-nume
(length (getpropc (car names) 'runic-mapping-pairs nil wrld))
ttree ens wrld wrld nil)
(mv wrld ttree state)))
(t (let ((bodies (get-normalized-bodies names wrld))
(big-mutrec (big-mutrec names)))
(er-let* ((wrld1 (update-w big-mutrec
(putprop-initial-type-prescriptions
names wrld))))
(guess-and-putprop-type-prescription-lst-for-clique
names
bodies
def-nume
ens
wrld1
ttree
big-mutrec
cert-data-tp-entry
state)))))))))
; So that finishes the type-prescription business. Now to level-no...
(defun putprop-level-no-lst (names wrld)
; We compute the level-no properties for all the fns in names, assuming they
; have no such properties in wrld (i.e., we take advantage of the fact that
; when max-level-no sees a nil 'level-no it acts like it saw 0). Note that
; induction and rewriting do not use heuristics for 'level-no, so it seems
; reasonable not to recompute the 'level-no property when adding a :definition
; rule with non-nil :install-body value. We assume that we can get the
; 'recursivep and the 'def-bodies property of each fn in names from wrld.
(cond ((null names) wrld)
(t (let ((maximum (max-level-no (body (car names) t wrld) wrld)))
(putprop-level-no-lst (cdr names)
(putprop (car names)
'level-no
(if (getpropc (car names)
'recursivep nil wrld)
(1+ maximum)
maximum)
wrld))))))
; Next we put the primitive-recursive-defun property
(defun primitive-recursive-argp (var term wrld)
; Var is some formal of a recursively defined function. Term is the actual in
; the var position in a recursive call in the definition of the function.
; I.e., we are recursively replacing var by term in the definition. Is this
; recursion in the p.r. schema? Well, that is impossible to tell by just
; looking at the recursion, because we need to know that the tests governing
; the recursion are also in the scheme. In fact, we don't even check that; we
; just rely on the fact that the recursion was justified and so some governing
; test does the job. So, ignoring tests, what is a p.r. function? It is one
; in which every formal is replaced either by itself or by an application of a
; (nest of) primitive recursive destructors to itself. The primitive recursive
; destructors recognized here are all unary function symbols with level-no 0
; (e.g., car, cdr, nqthm::sub1, etc) as well as terms of the form (+ & -n) and
; (+ -n &), where -n is negative.
; A consequence of this definition (before we turned 1+ into a macro) is that
; 1+ is a primitive recursive destructor! Thus, the classic example of a
; terminating function not in the classic p.r. scheme,
; (fn x y) = (if (< x y) (fn (1+ x) y) 0)
; is now in the "p.r." scheme. This is a crock!
; Where is this notion used? The detection that a function is "p.r." is made
; after its admittance during the defun principle. The appropriate flag is
; stored under the property 'primitive-recursive-defunp. This property is only
; used (as of this writing) by induction-complexity1, where we favor induction
; candidates suggested by non-"p.r." functions. Thus, the notion of "p.r." is
; entirely heuristic and only affects which inductions we choose.
; Why don't we define it correctly? That is, why don't we only recognize
; functions that recurse via car, cdr, etc.? The problem is the
; introduction of the "NQTHM" package, where we want NQTHM::SUB1 to be a p.r.
; destructor -- even in the defn of NQTHM::LESSP which must happen before we
; prove that NQTHM::SUB1 decreases according to NQTHM::LESSP. The only way to
; fix this, it seems, would be to provide a world global variable -- perhaps a
; new field in the acl2-defaults-table -- to specify which function symbols are
; to be considered p.r. destructors. We see nothing wrong with this solution,
; but it seems cumbersome at the moment. Thus, we adopted this hackish notion
; of "p.r." and will revisit the problem if and when we see counterexamples to
; the induction choices caused by this notion.
(cond ((variablep term) (eq var term))
((fquotep term) nil)
(t (let ((fn (ffn-symb term)))
(case
fn
(binary-+
(or (and (nvariablep (fargn term 1))
(fquotep (fargn term 1))
(rationalp (cadr (fargn term 1)))
(< (cadr (fargn term 1)) 0)
(primitive-recursive-argp var (fargn term 2) wrld))
(and (nvariablep (fargn term 2))
(fquotep (fargn term 2))
(rationalp (cadr (fargn term 2)))
(< (cadr (fargn term 2)) 0)
(primitive-recursive-argp var (fargn term 1) wrld))))
(otherwise
(and (symbolp fn)
(fargs term)
(null (cdr (fargs term)))
(= (get-level-no fn wrld) 0)
(primitive-recursive-argp var (fargn term 1) wrld))))))))
(defun primitive-recursive-callp (formals args wrld)
(cond ((null formals) t)
(t (and (primitive-recursive-argp (car formals) (car args) wrld)
(primitive-recursive-callp (cdr formals) (cdr args) wrld)))))
(defun primitive-recursive-callsp (formals calls wrld)
(cond ((null calls) t)
(t (and (primitive-recursive-callp formals (fargs (car calls)) wrld)
(primitive-recursive-callsp formals (cdr calls) wrld)))))
(defun primitive-recursive-machinep (formals machine wrld)
; Machine is an induction machine for a singly recursive function with
; the given formals. We return t iff every recursive call in the
; machine has the property that every argument is either equal to the
; corresponding formal or else is a primitive recursive destructor
; nest around that formal.
(cond ((null machine) t)
(t (and
(primitive-recursive-callsp formals
(access tests-and-calls
(car machine)
:calls)
wrld)
(primitive-recursive-machinep formals (cdr machine) wrld)))))
(defun putprop-primitive-recursive-defunp-lst (names wrld)
; The primitive-recursive-defun property of a function name indicates
; whether the function is defined in the primitive recursive schema --
; or, to be precise, in a manner suggestive of the p.r. schema. We do
; not actually check for syntactic adherence to the rules and this
; property is of heuristic use only. See the comment in
; primitive-recursive-argp.
; We say a defun'd function is p.r. iff it is not recursive, or else it
; is singly recursive and every argument position of every recursive call
; is occupied by the corresponding formal or else a nest of primitive
; recursive destructors around the corresponding formal.
; Observe that our notion doesn't include any inspection of the tests
; governing the recursions and it doesn't include any check of the
; subfunctions used. E.g., the function that collects all the values of
; Ackermann's functions is p.r. if it recurses on cdr's.
(cond ((null names) wrld)
((cdr names) wrld)
((primitive-recursive-machinep (formals (car names) wrld)
(getpropc (car names)
'induction-machine nil wrld)
wrld)
(putprop (car names)
'primitive-recursive-defunp
t
wrld))
(t wrld)))
; Onward toward defuns... Now we generate the controller-alists.
(defun make-controller-pocket (formals vars)
; Given the formals of a fn and a measured subset, vars, of formals,
; we generate a controller-pocket for it. A controller pocket is a
; list of t's and nil's in 1:1 correspondence with the formals, with
; t in the measured slots.
(cond ((null formals) nil)
(t (cons (if (member (car formals) vars)
t
nil)
(make-controller-pocket (cdr formals) vars)))))
(defun make-controller-alist1 (names wrld)
; Given a clique of recursive functions, we return the controller alist built
; for the 'justification. A controller alist is an alist that maps fns in the
; clique to controller pockets. The controller pockets describe the measured
; arguments in a justification. We assume that all the fns in the clique have
; been justified (else none would be justified).
; This function should not be called on a clique consisting of a single,
; non-recursive fn (because it has no justification).
(cond ((null names) nil)
(t (cons (cons (car names)
(make-controller-pocket
(formals (car names) wrld)
(access justification
(getpropc (car names)
'justification
'(:error
"See MAKE-CONTROLLER-ALIST1.")
wrld)
:subset)))
(make-controller-alist1 (cdr names) wrld)))))
(defun make-controller-alist (names wrld)
; We store a controller-alists property for every recursive fn in names. We
; assume we can get the 'formals and the 'justification for each fn from wrld.
; If there is a fn with no 'justification, it means the clique consists of a
; single non-recursive fn and we store no controller-alists. We generate one
; controller pocket for each fn in names.
; The controller-alist associates a fn in the clique to a controller pocket. A
; controller pocket is a list in 1:1 correspondence with the formals of the fn
; with a t in slots that are controllers. The controllers assigned for the fns
; in the clique by a given controller-alist were used jointly in the
; justification of the clique.
(and (getpropc (car names) 'justification nil wrld)
(make-controller-alist1 names wrld)))
(defun max-nume-exceeded-error (ctx)
(er hard ctx
"ACL2 assumes that no nume exceeds ~x0. It is very surprising that ~
this bound is about to be exceeded. We are causing an error because ~
for efficiency, ACL2 assumes this bound is never exceeded. Please ~
contact the ACL2 implementors with a request that this assumption be ~
removed from enabled-numep."
(fixnum-bound)))
(defun putprop-defun-runic-mapping-pairs1 (names def-nume tp-flg ind-flg wrld)
; Names is a list of function symbols. For each fn in names we store some
; runic mapping pairs. We always create (:DEFINITION fn) and (:EXECUTABLE-
; COUNTERPART fn). If tp-flg is t, we create (:TYPE-PRESCRIPTION fn). If
; ind-flg is t we create (:INDUCTION fn). However, ind-flg is t only if tp-flg
; is t (that is, tp-flg = nil and ind-flg = t never arises). Thus, we may
; store 2 (tp-flg = nil; ind-flg = nil), 3 (tp-flg = t; ind-flg = nil), or 4
; (tp-flg = t; ind-flg = t) runes. As of this writing, we never call this
; function with tp-flg nil but ind-flg t and the function is not prepared for
; that possibility.
; WARNING: Don't change the layout of the runic-mapping-pairs without
; considering all the places that talk about the Essay on the Assignment of
; Runes and Numes by DEFUNS.
(cond ((null names) wrld)
(t (putprop-defun-runic-mapping-pairs1
(cdr names)
(+ 2 (if tp-flg 1 0) (if ind-flg 1 0) def-nume)
tp-flg
ind-flg
(putprop
(car names) 'runic-mapping-pairs
(list* (cons def-nume (list :DEFINITION (car names)))
(cons (+ 1 def-nume)
(list :EXECUTABLE-COUNTERPART (car names)))
(if tp-flg
(list* (cons (+ 2 def-nume)
(list :TYPE-PRESCRIPTION (car names)))
(if ind-flg
(list (cons (+ 3 def-nume)
(list :INDUCTION (car names))))
nil))
nil))
wrld)))))
(defun putprop-defun-runic-mapping-pairs (names tp-flg wrld)
; Essay on the Assignment of Runes and Numes by DEFUNS
; Names is a clique of mutually recursive function names. For each
; name in names we store a 'runic-mapping-pairs property. Each name
; gets either four (tp-flg = t) or two (tp-flg = nil) mapping pairs:
; ((n . (:definition name))
; (n+1 . (:executable-counterpart name))
; (n+2 . (:type-prescription name)) ; only if tp-flg
; (n+3 . (:induction name))) ; only if tp-flg and name is
; ; recursively defined
; where n is the next available nume. Important aspects to this
; include:
; * Fn-rune-nume knows where the :definition and :executable-counterpart
; runes are positioned.
; * Several functions (e.g. augment-runic-theory) exploit the fact
; that the mapping pairs are ordered ascending.
; * function-theory-fn1 knows that if the token of the first rune in
; the 'runic-mapping-pairs is not :DEFINITION then the base symbol
; is not a function symbol.
; * Get-next-nume implicitly exploits the fact that the numes are
; consecutive integers -- it adds the length of the list to
; the first nume to get the next available nume.
; * Cleanse-type-prescriptions knows that the same number of numes are
; consumed by each function in a DEFUNS. We have consistently used
; the formal parameter def-nume when we were enumerating numes for
; definitions.
; * Convert-theory-to-unordered-mapping-pairs1 knows that if the first rune in
; the list is a :definition rune, then the length of this list is 4 if and
; only if the list contains an :induction rune, in which case that rune is
; last in the list.
; In short, don't change the layout of this property unless you
; inspect every occurrence of 'runic-mapping-pairs in the system!
; (Even that won't find the def-nume uses.) Of special note is the
; fact that all non-constrained function symbols are presumed to have
; the same layout of 'runic-mapping-pairs as shown here. Constrained
; symbols have a nil 'runic-mapping-pairs property.
; We do not allocate the :type-prescription or :induction runes or their numes
; unless tp-flg is non-nil. This way we can use this same function to
; initialize the 'runic-mapping-pairs for primitives like car and cdr, without
; wasting runes and numes. We like reusing this function for that purpose
; because it isolates the place we create the 'runic-mapping-pairs for
; functions.
(let ((next-nume (get-next-nume wrld)))
(prog2$ (or (<= (the-fixnat next-nume)
(- (the-fixnat (fixnum-bound))
(the-fixnat (* (the-fixnat 4)
(the-fixnat (length names))))))
(max-nume-exceeded-error 'putprop-defun-runic-mapping-pairs))
(putprop-defun-runic-mapping-pairs1
names
next-nume
tp-flg
(and tp-flg
(getpropc (car names) 'recursivep nil wrld))
wrld))))
; NOTE: Several functions formerly defined here in support of guard
; verification have been moved to history-management.lisp, to support the
; definition of guard-theorem.
; We have to collect every subroutine mentioned by any member of the clique and
; check that its guards have been checked. We cause an error if not. Once we
; have checked that all the subroutines have had their guards checked, we
; generate the guard clauses for the new functions.
(defun print-verify-guards-msg (names col state)
; Note that names is either a singleton list containing a theorem name
; or is a mutually recursive clique of function names.
; This function increments timers. Upon entry, the accumulated time
; is charged to 'other-time. The time spent in this function is
; charged to 'print-time.
(cond
((ld-skip-proofsp state) state)
(t
(pprogn
(increment-timer 'other-time state)
(mv-let (col state)
(io? event nil (mv col state)
(col names)
(fmt1 "~#0~[This lambda expression~/~&1~] ~#1~[is~/are~] ~
compliant with Common Lisp.~|"
(list (cons #\0 (if (and (consp names)
(consp (car names))
(eq (car (car names)) 'lambda))
0 1))
(cons #\1 names))
col
(proofs-co state)
state nil)
:default-bindings ((col 0)))
(declare (ignore col))
(increment-timer 'print-time state))))))
(defun collect-ideals (names wrld acc)
(cond ((null names) acc)
((eq (symbol-class (car names) wrld) :ideal)
(collect-ideals (cdr names) wrld (cons (car names) acc)))
(t (collect-ideals (cdr names) wrld acc))))
(defun collect-non-ideals (names wrld)
(cond ((null names) nil)
((eq (symbol-class (car names) wrld) :ideal)
(collect-non-ideals (cdr names) wrld))
(t (cons (car names) (collect-non-ideals (cdr names) wrld)))))
(defun collect-non-common-lisp-compliants (names wrld)
(cond ((null names) nil)
((eq (symbol-class (car names) wrld) :common-lisp-compliant)
(collect-non-common-lisp-compliants (cdr names) wrld))
(t (cons (car names)
(collect-non-common-lisp-compliants (cdr names) wrld)))))
(defun all-fnnames1-exec (flg x acc)
; Keep this in sync with all-fnnames1. Also see the comment about
; all-fnnames1-exec in put-invariant-risk before modifying this function.
(cond (flg ; x is a list of terms
(cond ((null x) acc)
(t (all-fnnames1-exec nil (car x)
(all-fnnames1-exec t (cdr x) acc)))))
((variablep x) acc)
((fquotep x) acc)
((flambda-applicationp x)
(all-fnnames1-exec nil (lambda-body (ffn-symb x))
(all-fnnames1-exec t (fargs x) acc)))
((eq (ffn-symb x) 'return-last)
(cond ((equal (fargn x 1) '(quote mbe1-raw))
(all-fnnames1-exec nil (fargn x 2) acc))
((and (equal (fargn x 1) '(quote ec-call1-raw))
(nvariablep (fargn x 3))
(not (fquotep (fargn x 3)))
(not (flambdap (ffn-symb (fargn x 3)))))
(all-fnnames1-exec t (fargs (fargn x 3)) acc))
(t (all-fnnames1-exec t
(fargs x)
(add-to-set-eq (ffn-symb x) acc)))))
(t
(all-fnnames1-exec t (fargs x)
(add-to-set-eq (ffn-symb x) acc)))))
(defmacro all-fnnames-exec (term)
`(all-fnnames1-exec nil ,term nil))
(defun collect-guards-and-bodies (lst)
; Lst is a list of well-formed lambda objects. We collect the set of all
; guards and bodies.
(cond
((endp lst) nil)
(t (add-to-set-equal
(lambda-object-guard (car lst))
(add-to-set-equal
(lambda-object-body (car lst))
(collect-guards-and-bodies (cdr lst)))))))
(defun chk-common-lisp-compliant-subfunctions-cmp (names0 names terms wrld str
ctx)
; See chk-common-lisp-compliant-subfunctions and note especially its warning
; about how not all names have been defined in wrld.
(cond ((null names) (value-cmp nil))
(t (let* ((fns (set-difference-eq
(all-fnnames1-exec
t ; list of terms (all-fnnames-exec (car terms))
(cons (car terms)
(if (global-val 'boot-strap-flg wrld)
nil
(collect-guards-and-bodies
(collect-certain-lambda-objects
:well-formed
(car terms)
wrld
nil))))
(if (global-val 'boot-strap-flg wrld)
nil
(all-fnnames! nil :inside nil
(car terms)
nil wrld nil)))
names0))
(bad (collect-non-common-lisp-compliants fns wrld)))
(cond
(bad
(er-cmp ctx "The ~@0 for ~x1 calls the function~#2~[ ~&2~/s ~
~&2~], the guards of which have not yet been ~
verified. See :DOC verify-guards."
str (car names) bad))
(t (mv-let (warrants unwarranteds)
(if (global-val 'boot-strap-flg wrld)
(mv nil nil)
(warrants-for-tamep-lambdap-lst
(collect-certain-lambda-objects :well-formed
(car terms)
wrld nil)
wrld nil nil))
(declare (ignore warrants))
(cond
(unwarranteds
(er-cmp ctx "The ~@0 for ~x1 applies the function~#2~[ ~
~&2~/s ~&2~] which ~#2~[is~/are~] not yet ~
warranted. Lambda objects containing ~
unwarranted function symbols are not ~
provably tame and can't be applied. See ~
:DOC verify-guards."
str (car names) unwarranteds))
(t (chk-common-lisp-compliant-subfunctions-cmp
names0 (cdr names) (cdr terms)
wrld str ctx))))))))))
(defun chk-common-lisp-compliant-subfunctions (names0 names terms wrld str ctx
state)
; Names0 is a list of function symbols being defined (or that have been
; defined). Names is a terminal sublist of names0 and terms is a list of the
; guards or the bodies of the function symbols listed in names. Names and
; terms are in 1:1 correspondence. We check that every function symbol (other
; than those listed in names0) used in terms -- including symbols used in
; quoted well-formed lambda objects! -- is Common Lisp compliant. If not, we
; cause an error. (During boot-strapping, this function does not look for
; well-formed lambda objects because we can't identify them prior to setting up
; badges for all primitives.)
; Also, contrary to its name, this function also checks that all user-defined
; functions occurring in well-formed lambda objects in the bodies are
; warranted. If a lambda object contains a badged but unwarranted function
; symbol then it can be well-formed but not provably tame and thus guard
; verification will fail.
; Str is a string used in our error message and is "guard", "split-types
; expression", "body" or "auxiliary function". Note that this function is used
; by chk-acceptable-defuns and by chk-acceptable-verify-guards and
; chk-stobj-field-descriptor. In the first usage, names have not been defined
; yet; in the other two they have. So be careful about using wrld to get
; properties of names.
(cmp-to-error-triple (chk-common-lisp-compliant-subfunctions-cmp
names0 names terms wrld str ctx)))
(defun chk-acceptable-verify-guards-formula-cmp (name x ctx wrld state-vars)
(mv-let (erp term bindings)
(translate1-cmp x
:stobjs-out
'((:stobjs-out . :stobjs-out))
t ; known-stobjs
ctx wrld state-vars)
(declare (ignore bindings))
(cond
((and erp (null name)) ; erp is a ctx and val is a msg
(mv-let (erp2 term2 bindings)
(translate1-cmp x t nil t ctx wrld state-vars)
(declare (ignore bindings term2))
(cond
(erp2 ; translation for formulas fails, so rely on previous error
(mv erp term))
(t (er-cmp ctx
"The guards for the given formula cannot be verified ~
because it has the wrong syntactic form for evaluation, ~
perhaps due to restrictions about multiple-values, ~
stobjs or dfs. See :DOC verify-guards.")))))
(erp
(er-cmp ctx
"The guards for ~x0 cannot be verified because its formula has ~
the wrong syntactic form for evaluation, perhaps due to ~
multiple-value or stobj restrictions. See :DOC verify-guards."
(or name x)))
((collect-non-common-lisp-compliants (all-fnnames-exec term)
wrld)
(er-cmp ctx
"The formula ~#0~[named ~x1~/~x1~] contains a call of the ~
function~#2~[ ~&2~/s ~&2~], the guards of which have not yet ~
been verified. See :DOC verify-guards."
(if name 0 1)
(or name x)
(collect-non-common-lisp-compliants (all-fnnames-exec term)
wrld)))
(t
(er-progn-cmp
(chk-common-lisp-compliant-subfunctions-cmp
(list name) (list name)
(list term)
wrld "formula" ctx)
(value-cmp (cons :term term)))))))
(defun guard-simplify-msg (x)
(msg "The only legal values for :GUARD-SIMPIFY are ~x0 and ~x1. The value ~
~x2 is thus illegal.~@3"
t :limited x
(if (eq x nil)
(msg " (Consider using :LIMITED in place of ~x0.)"
nil)
"")))
(defun guard-simplify-p (x ctx)
; In the ACL2 source code, a guard-simplify parameter has value t or :limited.
; This function checks that (though we should already have checked for that
; before calls of this function), and it returns nil when x is :limited.
; Keep this in sync with the second (final) binding of guard-simplify in
; chk-acceptable-defuns1 and a similar binding in
; chk-acceptable-verify-guards-cmp.
(declare (xargs :guard (member-eq x '(t :limited))))
(cond ((eq x t) t)
((eq x :limited) nil)
(t
(er hard ctx "~@0" (guard-simplify-msg x)))))
(defun chk-acceptable-verify-guards-cmp (name rrp guard-simplify ctx wrld
state-vars)
; We check that name is acceptable input for verify-guards and either cause an
; error or return the list of objects from which guard clauses should be
; generated or (when rrp = t, we might return 'redundant). We're more precise
; below.
; Below we describe a case analysis on name, a Test to perform, and the
; Non-Erroneous Value to return if the test succeeds. If a Test fails or the
; case analysis on name is exhausted without specifying an answer, an error is
; caused. When name is a function symbol we'll use names to be the set of
; function symbols in name's clique.
; * if name is a :common-lisp-compliant function symbol or lambda expression
; and rrp = t:
; Test: T
; Non-Erroneous Value: 'redundant.
; * if name is a function symbol:
; Test: is every subfunction in the definitions of names -- including symbols
; in quoted well-formed lambda objects -- except possibly names themselves
; :common-lisp-compliant?
; Non-erroneous Value: names
; * if name is a theorem name:
; Test: is every function used in the formula :common-lisp-compliant?
; Note: This test leaves out quoted well-formed lambda objects from consideration
; because we're not really interested in fast execution of instances of thms.
; Non-erroneous Value: (list name)
; * if name is a lambda object:
; Test: is name a well-formed lambda object and every function symbol in it
; (including in the :guard and body) -- including symbols in quoted
; well-formed lambda objects :common-lisp-compliant?
; Non-erroneous Value: (list name)
; * if name is a lambda$ expression
; Test: can name be translated non-erroneously to name', where name' is a
; well-formed lambda object, and is every function in name' (including in the
; :guard and body) -- including symbols in well-formed lambda objects
; :common-lisp-compliant?
; Non-erroneous Value: (list name')
; Otherwise, an error is caused.
; One might wonder when two peers in a clique can have different symbol-classes,
; e.g., how is it possible (as implied above) for name to be :ideal but for one
; of its peers to be :common-lisp-compliant or :program? Redefinition. For
; example, the clique could have been admitted as :logic and then later one
; function in it redefined as :program. Because redefinition invalidates the
; system, we could do anything in this case. What we choose to do is to cause
; an error and say you can't verify the guards of any of the functions in the
; nest.
; Motivation: When rrp is t, we get one of three answers: the redundancy signal
; (if name is compliant), a list of objects (either names or well-formed lambda
; expressions) from which to generate guard obligations (if such obligations
; can be generated), or an error. If rrp is nil, we get either the objects
; from which to generate guard obligations or an error. The ``objects'' are
; all either names or well-formed lambda expressions. We use rrp = nil when we
; are trying to (re-)generate the guard obligations as by
; verify-guards-formula. Note that rrp = nil is more strict than rrp = t in
; the sense that with rrp=t we might be 'redundant but with rrp=nil the same
; name might generate an error because it's in a clique that, due to
; redefinition, now has a :program mode function in it.
(er-let*-cmp
((ignore
(cond
; Keep this in sync with guard-simplify-p and a similar check in
; chk-acceptable-defuns1.
((member-eq guard-simplify '(t :limited))
(value-cmp nil))
(t (er-cmp ctx "~@0" (guard-simplify-msg guard-simplify)))))
(name
(cond
((symbolp name)
(value-cmp name))
((and (consp name)
(or (eq (car name) 'lambda)
(eq (car name) 'lambda$)))
(cond
((eq (car name) 'lambda)
(cond
((well-formed-lambda-objectp name wrld)
; We call hons-copy-lambda-object? here for the same reason that is given in
; translate11-lambda-object. Our convention is to call
; hons-copy-lambda-object? on QUOTEd lambda objects and name isn't quoted. So
; we quote it just so the error message is nice. And then we unquote the
; returned value when there's no error.
(mv-let (erp val)
(hons-copy-lambda-object? (kwote name))
(cond
(erp (er-cmp ctx "~@0" val))
(t (value-cmp (unquote val))))))
(t (er-cmp ctx
"~x0 is not a well-formed LAMBDA expression. See :DOC ~
verify-guards."
name))))
(t
(mv-let (erp val bindings)
(translate11-lambda-object
name
'(nil) ; stobjs-out
nil ; bindings
nil ; known-stobjs
nil ; flet-alist
name
'verify-guards
wrld
state-vars
nil)
(declare (ignore bindings))
(mv erp (if erp val (unquote val)))))))
(t (er-cmp ctx
"~x0 is not a symbol, a lambda object, or a lambda$ ~
expression. See :DOC verify-guards."
name)))))
; Name is now either a symbol or a consp, and if it is a consp it is a
; well-formed lambda object.
(let ((symbol-class
(cond ((symbolp name)
(symbol-class name wrld))
((member-equal name
(global-val 'common-lisp-compliant-lambdas wrld))
:common-lisp-compliant)
(t
; Name is known to be a well-formed lambda, but it may contain all classes of
; badged symbols, including :program mode ones. We don't know how to classify
; the lambda and so we'll choose the worst possibility. But as of this writing
; symbol-class is not used in the lambda case below!
:program))))
(cond
((and rrp
(eq symbol-class :common-lisp-compliant))
(value-cmp 'redundant))
((consp name)
; Name is a well-formed lambda object. If every fn in the guard and body is
; compliant (so guard obligations can be computed) we return the list
; containing the well-formed lambda expression derived from name which is now
; the value of the variable of that name.
(let* ((names (list name))
(guards (list (lambda-object-guard name)))
(bodies (list (lambda-object-body name))))
(er-progn-cmp
(chk-common-lisp-compliant-subfunctions-cmp
names names guards wrld "guard" ctx)
(chk-common-lisp-compliant-subfunctions-cmp
names names bodies wrld "body" ctx)
(value-cmp names))))
; Old stuff:
; (bad-guard-fns
; (collect-non-common-lisp-compliants (all-fnnames guard) wrld))
; (bad-body-fns
; (collect-non-common-lisp-compliants (all-fnnames body) wrld)))
;
; ; Any non-compliant fns in the guard or body are known to be :ideal because
; ; :program mode fns are not allowed in well-formed lambda objects.
;
; (cond
; (bad-guard-fns
; (er-cmp ctx
; "This lambda expression cannot be guard verified because ~
; the guard mentions ~&0 which ~#0~[is~/are~] not guard ~
; verified: ~x1."
; bad-guard-fns
; name))
; (bad-body-fns
; (er-cmp ctx
; "This lambda expression cannot be guard verified because ~
; the body mentions ~&0 which ~#0~[is~/are~] not guard ~
; verified: ~x1."
; bad-body-fns
; name))
; (t (value-cmp (list name))))
; ))
((getpropc name 'theorem nil wrld)
; Theorems are of either symbol-class :ideal or :common-lisp-compliant.
(er-progn-cmp
(chk-acceptable-verify-guards-formula-cmp
name
(getpropc name 'untranslated-theorem nil wrld)
ctx wrld state-vars)
(value-cmp (list name))))
((function-symbolp name wrld)
(case symbol-class
(:program
(er-cmp ctx
"~x0 is in :program mode. Only :logic mode functions can ~
have their guards verified. See :DOC verify-guards."
name))
((:ideal :common-lisp-compliant)
(let* ((recp (getpropc name 'recursivep nil wrld))
(names (cond
((null recp)
(list name))
(t recp)))
(bad-names (if (eq symbol-class :ideal)
(collect-non-ideals names wrld)
(collect-programs names wrld))))
(cond (bad-names
(er-cmp ctx
"One or more of the mutually-recursive peers of ~
~x0 ~#1~[was not defined in :logic mode~/either ~
was not defined in :logic mode or has already had ~
its guards verified~]. The offending ~
function~#2~[ is~/s are~] ~&2. We thus cannot ~
verify the guards of ~x0. This situation can ~
arise only through redefinition."
name
(if (eq symbol-class :ideal) 1 0)
bad-names))
(t
(er-progn-cmp
(chk-common-lisp-compliant-subfunctions-cmp
names names
(guard-lst names nil wrld)
wrld "guard" ctx)
(chk-common-lisp-compliant-subfunctions-cmp
names names
(getprop-x-lst names 'unnormalized-body wrld)
wrld "body" ctx)
(value-cmp names))))))
(otherwise ; the symbol-class :common-lisp-compliant is handled above
(er-cmp ctx
"Implementation error: Unexpected symbol-class, ~x0, for ~
the function symbol ~x1."
symbol-class name))))
(t (let ((fn (deref-macro-name name (macro-aliases wrld))))
(er-cmp ctx
"~x0 is not a function symbol or a theorem name in the ~
current ACL2 world. ~@1"
name
(cond ((eq fn name) "See :DOC verify-guards.")
(t (msg "Note that ~x0 is a macro-alias for ~x1. ~
Consider calling verify-guards with ~
argument ~x1 instead, or use ~
verify-guards+. See :DOC verify-guards, ~
see :DOC verify-guards+, and see :DOC ~
macro-aliases-table."
name fn))))))))))
(defun chk-acceptable-verify-guards (name rrp guard-simplify ctx wrld state)
(cmp-to-error-triple
(chk-acceptable-verify-guards-cmp name rrp guard-simplify ctx wrld
(default-state-vars t))))
(defun guard-obligation-clauses (x guard-debug ens wrld state)
; X is of one of three forms: (i) a list of function names and/or well-formed
; lambda object, (ii) a singleton list containing a theorem name, or (iii)
; (:term . y) where y must be a translated term. Returns two results. The
; first is a set of clauses justifying the guards x, i.e., in case (i) the
; guards of all the functions in x, (ii) the guards of the theorem's formula,
; or (iii) the guards of term y. The second result is an assumption-free
; tag-tree justifying that set of clauses.
; Ens may be an actual ens or :do-not-simplify, in which case no simplification
; that depends on the current set of enabled rules will take place in producing
; the guard clauses.
(mv-let (cl-set cl-set-ttree)
(cond ((and (consp x)
(eq (car x) :term))
(mv-let (cl-set cl-set-ttree)
(guard-clauses+
(cdr x)
(and guard-debug :top-level)
nil ;stobj-optp = nil
nil ens wrld
(f-get-global 'safe-mode state)
(gc-off state)
nil
nil)
(mv cl-set cl-set-ttree)))
((and (consp x)
(null (cdr x))
(symbolp (car x))
(getpropc (car x) 'theorem nil wrld))
(mv-let (cl-set cl-set-ttree)
(guard-clauses+
(getpropc (car x) 'theorem nil wrld)
(and guard-debug (car x))
nil ;stobj-optp = nil
nil ens wrld
(f-get-global 'safe-mode state)
(gc-off state)
nil
nil)
(mv cl-set cl-set-ttree)))
(t (guard-clauses-for-clique
x
(cond ((null guard-debug) nil)
((cdr x) 'mut-rec)
(t t))
ens
wrld
(f-get-global 'safe-mode state)
; It is important to turn on guard-checking here. If we avoid this binding,
; then we can get a hard Lisp error as follows, because a call of
; eval-ground-subexpressions from guard-clauses-for-fn should have failed (due
; to a guard violation) but didn't. (Since guard-clauses-for-fn isn't called
; in the two cases above for :term and 'theorem, we aren't aware of needing to
; take this extra care in those cases.)
; (set-guard-checking nil)
; (defun foo (x)
; (declare (xargs :guard (consp x)))
; (cons x (car 3)))
; (set-guard-checking t)
; (foo '(a b))
; Note that we do not need to bind to :all, since for calls of a guard-verified
; function such as foo, above, t and :all behave the same: if the guard holds
; at the top, then it holds through all evaluation, including recursive calls.
nil ; gc-off
nil)))
; Cl-set-ttree is 'assumption-free.
(mv-let (cl-set cl-set-ttree)
(clean-up-clause-set cl-set
(if (eq ens :do-not-simplify) nil ens)
wrld cl-set-ttree state)
; Cl-set-ttree is still 'assumption-free.
(mv cl-set cl-set-ttree))))
(defun guard-obligation (x rrp guard-debug guard-simplify ctx state)
(let* ((wrld (w state))
(namep (and (symbolp x)
(not (keywordp x))
(not (defined-constant x wrld)))))
(er-let*-cmp
((y
(cond (namep (chk-acceptable-verify-guards-cmp
x rrp guard-simplify ctx wrld (default-state-vars t)))
(t (chk-acceptable-verify-guards-formula-cmp
nil x ctx wrld (default-state-vars t))))))
(cond
((and namep (eq y 'redundant))
(value-cmp :redundant))
(t (mv-let (cl-set cl-set-ttree)
(guard-obligation-clauses
y guard-debug
(if (guard-simplify-p guard-simplify ctx)
(ens state)
:do-not-simplify)
wrld state)
(value-cmp (list* y cl-set cl-set-ttree))))))))
(defun prove-guard-clauses-msg (names cl-set cl-set-ttree displayed-goal
verify-guards-formula-p
guard-simplify ctx state)
(let ((simp-phrase (tilde-*-simp-phrase cl-set-ttree)))
(cond
((null cl-set)
(fmt "The guard conjecture for ~#0~[this lambda expression~/~&1~/the ~
given term~] is trivial to prove~#2~[~/, given ~*3~].~@4"
(list (cons #\0 (if names
(if (consp (car names))
0 1)
2))
(cons #\1 names)
(cons #\2 (if (nth 4 simp-phrase) 1 0))
(cons #\3 simp-phrase)
(cons #\4 (if verify-guards-formula-p "~|" " ")))
(proofs-co state)
state
nil))
(t
(pprogn
(fms "The ~@0 for ~#1~[this lambda expression~/~&2~/the given ~
term~]~#3~[~/, given ~*4,~] is~%~%Goal~%~Q56."
(list (cons #\0
(if (guard-simplify-p guard-simplify ctx)
"non-trivial part of the guard conjecture"
"guard conjecture (with only :limited ~
simplification)"))
(cons #\1 (if names
(if (consp (car names))
0 1)
2))
(cons #\2 names)
(cons #\3 (if (nth 4 simp-phrase) 1 0))
(cons #\4 simp-phrase)
(cons #\5 displayed-goal)
(cons #\6 (or (term-evisc-tuple nil state)
(and (gag-mode)
(let ((tuple
(gag-mode-evisc-tuple state)))
(cond ((eq tuple t)
(term-evisc-tuple t state))
(t tuple)))))))
(proofs-co state)
state
nil)
(mv 0 ; don't care
state))))))
(defun verify-guards-formula-fn (x rrp guard-debug guard-simplify ctx state)
(er-let* ((tuple (cmp-to-error-triple
(guard-obligation x rrp guard-debug guard-simplify
'verify-guards-formula
state))))
(cond ((eq tuple :redundant)
(value :redundant))
(t
(let ((names (car tuple))
(displayed-goal (prettyify-clause-set (cadr tuple)
(let*-abstractionp
state)
(w state)))
(cl-set-ttree (cddr tuple)))
(mv-let (col state)
(prove-guard-clauses-msg (if (and (consp names)
(eq (car names) :term))
nil
names)
(cadr tuple) cl-set-ttree
displayed-goal t guard-simplify ctx
state)
(declare (ignore col))
(value :invisible)))))))
(defmacro verify-guards-formula (x &key rrp guard-debug (guard-simplify 't)
&allow-other-keys)
`(verify-guards-formula-fn ',x ',rrp ',guard-debug ',guard-simplify
'verify-guards-formula state))
(defun prove-guard-clauses (names hints otf-flg guard-debug guard-simplify
ctx ens wrld state)
; Names is either a clique of mutually recursive functions or else a singleton
; list containing either a theorem name or a well-formed lambda object. We
; generate and attempt to prove the guard conjectures for the formulas in
; names. We generate suitable output explaining what we are doing. This is an
; error/value/state producing function that returns a pair of the form (col
; . ttree) when non-erroneous. Col is the column in which the printer is left.
; We always output something and we always leave the printer ready to start a
; new sentence. Ttree is a tag-tree describing the proof.
; This function increments timers. Upon entry, any accumulated time
; is charged to 'other-time. The printing done herein is charged
; to 'print-time and the proving is charged to 'prove-time.
(cond
((ld-skip-proofsp state) (value '(0 . nil)))
(t
(mv-let
(cl-set cl-set-ttree state)
(pprogn (io? event nil state
(names)
(fms "Computing the guard conjecture for ~&0....~|"
(list (cons #\0 names))
(proofs-co state)
state
nil))
(mv-let (cl-set cl-set-ttree)
(guard-obligation-clauses
names guard-debug
(if (guard-simplify-p guard-simplify ctx)
ens
:do-not-simplify)
wrld state)
(mv cl-set cl-set-ttree state)))
; Cl-set-ttree is 'assumption-free.
(pprogn
(increment-timer 'other-time state)
(let ((displayed-goal (prettyify-clause-set cl-set
(let*-abstractionp state)
wrld)))
(mv-let
(col state)
(io? event nil (mv col state)
(names cl-set cl-set-ttree displayed-goal guard-simplify ctx)
(prove-guard-clauses-msg names cl-set cl-set-ttree
displayed-goal nil guard-simplify ctx
state)
:default-bindings ((col 0)))
(pprogn
(increment-timer 'print-time state)
(cond
((null cl-set)
(value (cons col cl-set-ttree)))
(t
(mv-let (erp ttree state)
(prove (termify-clause-set cl-set)
(make-pspv ens wrld state
:displayed-goal displayed-goal
:otf-flg otf-flg)
hints
ens wrld ctx state)
(cond
(erp
(mv-let
(erp1 val state)
(er soft ctx
"The proof of the guard conjecture for ~&0 has ~
failed. You may wish to avoid specifying a guard, ~
or to supply option :VERIFY-GUARDS ~x1 with the ~
:GUARD.~@2~|"
names
nil
(if guard-debug
""
" Otherwise, you may wish to specify :GUARD-DEBUG ~
T; see :DOC verify-guards."))
(declare (ignore erp1))
(mv (msg
"The proof of the guard conjecture for ~&0 has ~
failed; see the discussion above about ~&1. "
names
(if guard-debug
'(:VERIFY-GUARDS)
'(:VERIFY-GUARDS :GUARD-DEBUG)))
val
state)))
(t
(mv-let (col state)
(io? event nil (mv col state)
(names)
(fmt "That completes the proof of the guard theorem ~
for ~&0. "
(list (cons #\0 names))
(proofs-co state)
state
nil)
:default-bindings ((col 0)))
(pprogn
(increment-timer 'print-time state)
(value
(cons (or col 0)
(cons-tag-trees
cl-set-ttree
ttree))))))))))))))))))
(defun maybe-remove-invariant-risk (names wrld new-wrld)
; Names is either a list of function names or a singleton containing a
; well-formed lambda object. All the elements of names have been guard
; verified. We ignore the lambdas and just focus on the function names and
; remove (set to nil) the invariant-risk property if it has been set. Note
; that invariant-risk concerns :program mode functions (that might perform
; unchecked modifications to stobjs or arrays). But all well-formed lambda
; objects are composed entirely of :logic mode functions.
(cond ((endp names) new-wrld)
(t (let ((new-wrld
(if (and (symbolp (car names))
(getpropc (car names) 'invariant-risk nil wrld)
(equal (guard (car names) t wrld) *t*))
(putprop (car names) 'invariant-risk nil new-wrld)
new-wrld)))
(maybe-remove-invariant-risk (cdr names) wrld new-wrld)))))
(defun verify-guards-fn1 (names hints otf-flg guard-debug
guard-simplify ctx state)
; This function is called on a either a singleton list containing a theorem
; name or a well-formed lambda expression or a list of one or more recursively
; defined fns.
; In any case, we know the theorem/functions are composed entirely of compliant
; subfunctions. Hints is a properly translated hints list. This is an
; error/value/state producing function. We cause an error if some subroutine
; of names has not yet had its guards checked or if we cannot prove the guards.
; Otherwise, the "value" is a pair of the form (wrld . ttree), where wrld
; results from storing symbol-class :common-lisp-compliant for each name and
; ttree is the ttree proving the guards.
; Note: In a series of conversations started around 13 Jun 94, with Bishop
; Brock, we came up with a new proposal for the form of guard conjectures.
; However, we have decided to delay the experimentation with this proposal
; until we evaluate the new logic of Version 1.8. But, the basic idea is this.
; Consider two functions, f and g, with guards a and b, respectively. Suppose
; (f (g x)) occurs in a context governed by q. Then the current guard
; conjectures are
; (1) q -> (b x) ; guard for g holds on x
; (2) q -> (a (g x)) ; guard for f holds on (g x)
; Note that (2) contains (g x) and we might need to know that x satisfies the
; guard for g here. Another way of putting it is that if we have to prove both
; (1) and (2) we might imagine forward chaining through (1) and reformulate (2)
; as (2') q & (b x) -> (a (g x)).
; Now in the days when guards were part of the logic, this was a pretty
; compelling idea because we couldn't get at the definition of (g x) in (2)
; without establishing (b x) and thus formulation (2) forced us to prove
; (1) all over again during the proof of (2). But it is not clear whether
; we care now, because the smart user will define (g x) to "do the right thing"
; for any x and thus f will approve of (g x). So it is our expectation that
; this whole issue will fall by the wayside. It is our utter conviction of
; this that leads us to write this note. Just in case...
; ++++++++++++++++++++++++++++++
;
; Date: Sun, 2 Oct 94 17:31:10 CDT
; From: kaufmann (Matt Kaufmann)
; To: moore
; Subject: proposal for handling generalized booleans
;
; Here's a pretty simple idea, I think, for handling generalized Booleans. For
; the rest of this message I'll assume that we are going to implement the
; about-to-be-proposed handling of guards. This proposal doesn't address
; functions like member, which could be thought of as returning generalized
; booleans but in fact are completely specified (when their guards are met).
; Rather, the problem we need to solve is that certain functions, including EQUAL
; and RATIONALP, only specify the propositional equivalence class of the value
; returned, and no more. I'll call these "problematic functions" for the rest of
; this note.
;
; The fundamental ideas of this proposal are as follows.
;
; ====================
;
; (A) Problematic functions are completely a non-issue except for guard
; verification. The ACL2 logic specifies Boolean values for functions that are
; specified in dpANS to return generalized Booleans.
;
; (B) Guard verification will generate not only the current proof obligations,
; but also appropriate proof obligations to show that for all values returned by
; relevant problematic functions, only their propositional equivalence class
; matters. More on this later.
;
; (C) If a function is problematic, it had better only be used in propositional
; contexts when used in functions or theorems that are intended to be
; :common-lisp-compliant. For example, consider the following.
;
; (defun foo (x y z)
; (if x
; (equal y z)
; (cons y z)))
;
; This is problematic, and we will never be able to use it in a
; :common-lisp-compliant function or formula for other than its propositional
; value (unfortunately).
;
; ====================
;
; Elaborating on (B) above:
;
; So for example, if we're verifying guards on
;
; (... (foo (rationalp x) ...) ...)
;
; then there will be a proof obligation to show that under the appropriate
; hypotheses (from governing IF tests),
;
; (implies (and a b)
; (equal (foo a ...) (foo b ...)))
;
; Notice that I've assumed that a and b are non-NIL. The other case, where a and
; b are both NIL, is trivial since in that case a and b are equal.
;
; Finally, most of the time no such proof obligation will be generated, because
; the context will make it clear that only the propositional equivalence class
; matters. In fact, for each function we'll store information that gives
; ``propositional arguments'' of the function: arguments for which we can be
; sure that only their propositional value matters. More on this below.
;
; ====================
;
; Here are details.
;
; ====================
;
; 1. Every function will have a ``propositional signature,'' which is a list of
; T's and NIL's. The CAR of this list is T when the function is problematic.
; The CDR of the list is in 1-1 correspondence with the function's formals (in
; the same order, of course), and indicates whether the formal's value only
; matters propositionally for the value of the function.
;
; For example, the function
;
; (defun bar (x y z)
; (if x
; (equal y z)
; (equal y nil)))
;
; has a propositional signature of (T T NIL NIL). The first T represents the
; fact that this function is problematic. The second T represents the fact that
; only the propositional equivalence class of X is used to compute the value of
; this function. The two NILs say that Y and Z may have their values used other
; than propositionally.
;
; An argument that corresponds to a value of T will be called a ``propositional
; argument'' henceforth. An OBSERVATION will be made any time a function is
; given a propositional signature that isn't composed entirely of NILs.
;
; (2) Propositional signatures will be assigned as follows, presumably hung on
; the 'propositional-signature property of the function. We intend to ensure
; that if a function is problematic, then the CAR of its propositional signature
; is T. The converse could fail, but it won't in practice.
;
; a. The primitives will have their values set using a fixed alist kept in sync
; with *primitive-formals-and-guards*, e.g.:
;
; (defconst *primitive-propositional-signatures*
; '((equal (t nil nil))
; (cons (nil nil nil))
; (rationalp (t nil))
; ...))
;
; In particular, IF has propositional signature (NIL T NIL NIL): although IF is
; not problematic, it is interesting to note that its first argument is a
; propositional argument.
;
; b. Defined functions will have their propositional signatures computed as
; follows.
;
; b1. The CAR is T if and only if some leaf of the IF-tree of the body is the
; call of a problematic function. For recursive functions, the function itself
; is considered not to be problematic for the purposes of this algorithm.
;
; b2. An argument, arg, corresponds to T (i.e., is a propositional argument in
; the sense defined above) if and only if for every subterm for which arg is an
; argument of a function call, arg is a propositional argument of that function.
;
; Actually, for recursive functions this algorithm is iterative, like the type
; prescription algorithm, in the sense that we start by assuming that every
; argument is propositional and iterate, continuing to cut down the set of
; propositional arguments until it stabilizes.
;
; Consider for example:
;
; (defun atom-listp (lst)
; (cond ((atom lst) (eq lst nil))
; (t (and (atom (car lst))
; (atom-listp (cdr lst))))))
;
; Since EQ returns a generalized Boolean, ATOM-LISTP is problematic. Since
; the first argument of EQ is not propositional, ATOM-LISTP has propositional
; signature (T NIL).
;
; Note however that we may want to replace many such calls of EQ as follows,
; since dpANS says that NULL really does return a Boolean [I guess because it's
; sort of synonymous with NOT]:
;
; (defun atom-listp (lst)
; (cond ((atom lst) (null lst))
; (t (and (atom (car lst))
; (atom-listp (cdr lst))))))
;
; Now this function is not problematic, even though one might be nervous because
; ATOM is, in fact, problematic. However, ATOM is in the test of an IF (because
; of how AND is defined). Nevertheless, the use of ATOM here is of issue, and
; this leads us to the next item.
;
; (3) Certain functions are worse than merely problematic, in that their value
; may not even be determined up to propositional equivalence class. Consider for
; example our old favorite:
;
; (defun bad (x)
; (equal (equal x x) (equal x x)))
;
; In this case, we can't really say anything at all about the value of BAD, ever.
;
; So, every function is checked that calls of problematic functions in its body
; only occur either at the top-level of its IF structure or in propositional
; argument positions. This check is done after the computation described in (2)b
; above.
;
; So, the second version of the definition of ATOM-LISTP above,
;
; (defun atom-listp (lst)
; (cond ((atom lst) (null lst))
; (t (and (atom (car lst))
; (atom-listp (cdr lst))))))
;
; is OK in this sense, because both calls of ATOM occur in the first argument of
; an IF call, and the first argument of IF is propositional.
;
; Functions that fail this check are perfectly OK as :ideal functions; they just
; can't be :common-lisp-compliant. So perhaps they should generate a warning
; when submitted as :ideal, pointing out that they can never be
; :common-lisp-compliant.
;
; -- Matt
#-acl2-loop-only
(declare (ftype (function (t t t) (values t))
add-good-lambda-objects-to-cl-cache))
(let ((wrld (w state))
(ens (ens state)))
(er-let*
((pair (prove-guard-clauses names hints otf-flg guard-debug guard-simplify
ctx ens wrld state)))
; Pair is of the form (col . ttree)
(let* ((col (car pair))
(ttree1 (cdr pair))
(wrld1 (maybe-remove-invariant-risk names wrld wrld))
; The next line finds all the well-formed lambda objects in the fns whose guard
; obligations have just been verified. We put them all on the compliant
; lambdas list. But we also use the lambda-objects in the raw Lisp code below
; to extend the cache. If a defun has ill-formed lambdas and we verify guards
; on the function the ill-formed lambdas are not verified. And we don't add
; them to the cache. We could add :UGLY cache lines for them because they may
; well reach apply$. If and when they reach apply$ they'll be added to the
; cache on an as-needed basis. This may slow down evaluation, but they're
; interpreted by *1* apply$ anyway so the user couldn't care much!
(lambda-objects
(and (not (global-val 'boot-strap-flg wrld1))
(collect-well-formed-lambda-objects-lst names wrld1)))
(wrld2 (global-set 'common-lisp-compliant-lambdas
(union-equal
lambda-objects
(global-val 'common-lisp-compliant-lambdas
wrld1))
wrld1))
; Now upgrade the symbol-class (except for the case where names is a
; single lambda).
(wrld3
(if (and (consp names)
(consp (car names)))
wrld2
(putprop-x-lst1 names 'symbol-class
:common-lisp-compliant wrld2))))
; Add a :GOOD cl-cache-line for each lambda-object just verified. Ill-formed
; lambda objects are ignored here but will be added to the cache (as :UGLY) if
; and when they are apply$'d.
#-acl2-loop-only
(add-good-lambda-objects-to-cl-cache lambda-objects wrld3 state)
(pprogn
(print-verify-guards-msg names col state)
(value (cons wrld3 ttree1)))))))
(defun convert-runes-to-unordered-mapping-pairs (lst wrld ans)
; This function is based on convert-theory-to-unordered-mapping-pairs1.
; However, the present function expects each member R of lst to be in the shape
; of a rune, without runic abbreviations (see :DOC theories); moreover, here we
; do not require R to be a legal rune, as we simply ignore it in that case.
(cond
((endp lst) ans)
(t (convert-runes-to-unordered-mapping-pairs
(cdr lst) wrld
(let ((pair (frunic-mapping-pair (car lst) wrld)))
(cond (pair (cons pair ans))
(t ans)))))))
(defun augment-theory-weak (lst wrld)
; This function is like augment-theory, except that here we require each member
; R of lst to be a rune syntactically (not merely a runic abbreviation), but
; not necessarily to be a rune. If R is not a legal rune, then we ignore it.
(declare (xargs :guard (true-listp lst)))
(duplicitous-sort-car
nil
(convert-runes-to-unordered-mapping-pairs lst wrld nil)))
(defun with-useless-runes-aux (name state)
; See also with-useless-runes.
; There are two interesting cases: reading and writing a @useless-runes.lsp
; file. (Neither applies in some cases, in particular when skipping proofs and
; when name is nil.)
; First consider the case of reading a useless-runes file, where state global
; 'useless-runes is non-nil with :tag FAST-ALIST and :data fal, a fast alist,
; and name is a non-nil symbol that is associated in fal with a non-nil list,
; lst. So, lst is a list of lists of syntactic runes, that is, objects that
; are runes syntactically but might not all be legal runes. Then this function
; returns (mv 'read useless-runes-1 useless-runes-2), where useless-runes-1 is
; to be used when processing the event named name and useless-runes-2 is to be
; used after that completes. Thus if thy is the augmented runic theory
; corresponding to (car lst), then useless-runes-1 is the result of updating
; the current useless-runes by setting the :tag to THEORY and the :data to thy,
; and useless-runes-2 is the result of updating the :data field of
; useless-runes, by modifying fal to associate name with (cdr lst).
; Next consider the case of writing a useless-runes file, where state global
; 'useless-runes has a non-nil value whose :tag is CHANNEL and :data is data
; (which provides a channel and a list of package names). Then we return (mv
; 'write data useless-runes).
; Otherwise we return (mv nil nil useless-runes).
(let ((useless-runes (f-get-global 'useless-runes state)))
(cond
((or (null useless-runes)
(null name)
(ld-skip-proofsp state)
; The following should generally be false when not skipping proofs. But it is
; conceivable that a make-event form with :check-expansion t causes an event to
; be evaluated during include-book when not skipping proofs. That event might
; involve packages not known at the time a @useless-runes.lsp is read, so we do
; not want to write useless-runes to that file in this case. It thus makes
; sense also not to read a non-existent entry in the "read" mode of using an
; existing @useless-runes.lsp.
(global-val 'include-book-path (w state)))
(mv nil nil useless-runes))
((eq (access useless-runes useless-runes :tag) 'FAST-ALIST)
(let* ((fal (access useless-runes useless-runes :data))
(lst (cdr (hons-get name fal))))
(cond ((consp lst)
(let* ((runes (car lst))
(fal (hons-acons name (cdr lst) fal))
(useless-runes-2 (change useless-runes useless-runes
:data fal)))
(cond
; We avoid useless-runes in ACL2(p), since proofs can go in a different
; direction than ACL2(p). We use a different mechanism for avoiding
; useless-runes in ACL2(p) than in ACL2(r); see useless-runes-value and
; useless-runes-filename. It's not clear which is better, but it's also not
; clear that there's much reason to change either one at this point.
#+acl2-par
((f-get-global 'waterfall-parallelism state)
(mv 'read-but-skip nil useless-runes-2))
(t (mv 'read
(change useless-runes useless-runes
:tag 'THEORY
:data (augment-theory-weak runes (w state)))
useless-runes-2)))))
(t (mv nil nil useless-runes)))))
((and (eq (access useless-runes useless-runes :tag) 'CHANNEL)
#+acl2-par
(not (f-get-global 'waterfall-parallelism state)))
(mv 'write
(access useless-runes useless-runes :data)
useless-runes))
(t (mv nil nil useless-runes)))))
(defun accp-info (state)
#-acl2-loop-only
(mv nil
(wormhole-data
(cdr (assoc-eq 'ACCUMULATED-PERSISTENCE *wormhole-status-alist*)))
state)
#+acl2-loop-only
(read-acl2-oracle state))
(defun useless-runes (accp-info)
; This function is based on show-accumulated-persistence-phrase.
(let* ((totals (access accp-info accp-info :totals))
(alist (merge-sort-lexorder
(show-accumulated-persistence-phrase2
:USELESS
(car (last totals))
nil))))
(show-accumulated-persistence-list
(show-accumulated-persistence-remove-useless alist nil)
nil)))
(defun bad-useless-runes (useless-runes known-pkgs acc)
(cond ((endp useless-runes) (reverse acc))
(t (bad-useless-runes
(cdr useless-runes)
known-pkgs
(let ((rune (caddr (car useless-runes))))
(if (member-equal (symbol-package-name (base-symbol rune))
known-pkgs)
acc
(cons (car useless-runes) acc)))))))
(defun set-difference-equal-sorted (lst1 lst2)
; We return (set-difference-equal lst1 lst2), but take advantage of the
; following assumption: lst2 is a subsequence of lst1.
(cond ((null lst2) lst1)
((endp lst1)
(er hard? 'set-difference-equal-sorted
"Implementation error: Reached end of lst1 before end of lst2."))
((equal (car lst1) (car lst2))
(set-difference-equal-sorted (cdr lst1) (cdr lst2)))
(t (cons (car lst1)
(set-difference-equal-sorted (cdr lst1) lst2)))))
(defun remove-executable-counterpart-useless-runes1 (useless-runes)
(cond
((endp useless-runes) nil)
(t (let ((rune (caddr (car useless-runes))))
(cond
((eq (car rune) :executable-counterpart)
(remove-executable-counterpart-useless-runes1 (cdr useless-runes)))
(t (cons rune
(remove-executable-counterpart-useless-runes1
(cdr useless-runes)))))))))
(defun executable-counterpart-useless-runes-p (useless-runes)
(cond
((endp useless-runes) nil)
(t (let ((rune (caddr (car useless-runes))))
(cond
((eq (car rune) :executable-counterpart)
t)
(t (executable-counterpart-useless-runes-p (cdr useless-runes))))))))
(defun remove-executable-counterpart-useless-runes (useless-runes)
(cond ((executable-counterpart-useless-runes-p useless-runes) ; optimization
(remove-executable-counterpart-useless-runes1 useless-runes))
(t useless-runes)))
(defun print-rune (rune channel state)
(pprogn (princ$ #\( channel state)
(prin1$ (car rune) channel state)
(princ$ #\Space channel state)
(prin1$ (cadr rune) channel state)
(cond ((cddr rune)
(pprogn (princ$ " . " channel state)
(prin1$ (cddr rune) channel state)))
(t state))
(princ$ #\) channel state)))
(defun useless-runes-report-p (lst)
(cond ((atom lst) (null lst))
((let ((x (car lst)))
(and (true-listp x)
(natp (car x))
(natp (cadr x))
(caddr x) ; very weak check; should syntactically be a rune
(null (cdddr x))))
(useless-runes-report-p (cdr lst)))
(t nil)))
(defun print-useless-runes-tuples-rec (lst channel state)
; See print-useless-runes-tuples. Here, lst is initially non-nil.
(declare (xargs :guard (useless-runes-report-p lst)))
(cond
((endp lst)
(pprogn (newline channel state)
(princ$ " )" channel state)
(newline channel state)))
(t (let* ((triple (car lst))
(n1 (car triple))
(n2 (cadr triple))
(rune (caddr triple)))
(pprogn (newline channel state)
(princ$ " (" channel state)
(prin1$ n1 channel state)
(princ$ #\Space channel state)
(prin1$ n2 channel state)
(princ$ #\Space channel state)
(print-rune rune channel state)
(princ$ #\) channel state)
(print-useless-runes-tuples-rec (cdr lst) channel state))))))
(defun print-useless-runes-tuples (lst channel state)
; This function prints the given list of triples (n1 n2 rune), each preceded by
; a newline and a space, and finally concluding with a space, right paren, and
; newline. Except, if lst is empty then we print the concluding right paren
; immediately, without a space.
(declare (xargs :guard (useless-runes-report-p lst)))
(cond ((null lst)
(pprogn (princ$ #\) channel state)
(newline channel state)))
(t (print-useless-runes-tuples-rec lst channel state))))
(defun print-useless-runes (name channel known-pkgs state)
; We are given an output channel for a @useless-runes.lsp file. We print an
; entry into the indicated channel of the form (name . (pairlis$ frames thy)),
; where thy is the :list output for useless-runes that is associated with the
; event named name that is now completing. Note that each useless-rune is of
; the form (frames tries rune). These are sorted by accumulated-persistence,
; highest first. We omit any rune whose base symbol is not among the known
; packages, with a comment explaining which are omitted.
; We call remove-executable-counterpart-useless-runes to remove useless-runes
; that are :executable-counterpart runes. We do so because there are many
; places that we do not track such runes for accumulated-persistence, including
; (as of this writing) sublis-var!, eval-ground-subexpressions1, scons-term,
; ev-respecting-ens, and even rewrite and expand-abbreviations.
; WARNING: Avoid fmt functions below! They use the world's evisc-table, which
; the user may have modified so that certain numbers are printed as constants
; -- but we expect the numbers to be printed as numbers!
(error-free-triple-to-state
'print-useless-runes
(er-let* ((accp-info (accp-info state)))
(state-global-let*
((current-package "ACL2" set-current-package-state)
(fmt-hard-right-margin 10000 set-fmt-hard-right-margin)
(fmt-soft-right-margin 10000 set-fmt-soft-right-margin))
(let* ((bad-pkg
(not (member-equal (symbol-package-name name) known-pkgs)))
(useless-runes0 (remove-executable-counterpart-useless-runes
(useless-runes accp-info)))
(bad-tuples (and (not bad-pkg) ; else print all useless-runes
(bad-useless-runes useless-runes0 known-pkgs
nil)))
(useless-runes (if (null bad-tuples) ; optimization
useless-runes0
(set-difference-equal-sorted useless-runes0
bad-tuples))))
(pprogn
(cond
(bad-tuples
(pprogn
; WARNING: Before changing any of this printing, see the WARNING above about
; fmt functions.
(princ$ "; Just below, skipping the following list where the rune"
channel state)
(newline channel state)
(princ$ "; has an unknown package in the certification world:"
channel state)
(newline channel state)
(princ$ "#|" channel state)
(newline channel state)
(princ$ "(" channel state)
(print-useless-runes-tuples bad-tuples channel state)
(princ$ "|#" channel state)
(newline channel state)))
(bad-pkg
(pprogn
(princ$ "; Omitting the following entry because the package of"
channel state)
(newline channel state)
(princ$ "; the event name is unknown in the certification world."
channel state)
(newline channel state)
(princ$ "#|" channel state)
(newline channel state)))
(t state))
(princ$ #\( channel state)
(prin1$ name channel state)
(print-useless-runes-tuples useless-runes channel state)
(cond (bad-pkg
(pprogn (princ$ "|#" channel state)
(newline channel state)))
(t state))
(prog2$
; We are done printing, so turn off accumulated-persistence.
(accumulated-persistence nil)
(value nil))))))))
(defun augmented-runes-after-1 (nume pairs)
; We compute (loop$ for x in pairs when (> (car x) nume) collect x), but
; without using loop$ because of a boot-strapping problem.
(cond ((endp pairs) nil)
((> (caar pairs) nume)
(cons (car pairs)
(augmented-runes-after-1 nume (cdr pairs))))
(t nil)))
(defun augmented-runes-after (nume wrld ans)
; We collect into ans all (nume . rune) pairs for which the nume exceeds the
; given nume. See also extend-with-augmented-runes-after. A related function
; is universal-theory-fn1, but in the present function we do not reverse ans.
; Also, for efficiency, we do not check for redefinition, as we expect to use
; this function only during book certification, where redefinition should not
; take place without a trust tag.
; Here we elaborate on the point above about redefinition not taking place. We
; will call this function only during certification, when nume is at least the
; :index-of-last-enabling of the current enabled structure. Because of the
; update-current-theory call in end-prehistoric-world, we know that nume is
; therefore not inside the ground-zero theory. So even if redefinition takes
; place during the boot-strap, that will not affect validity of runes we
; collect here, since redefinition does not happen (without a trust tag) after
; the boot-strap while certifying a book.
(cond ((endp wrld) (reverse ans))
((and (eq (cadr (car wrld)) 'runic-mapping-pairs)
; We expect the following condition to be false, but we might as well check for
; it.
(not (eq (cddr (car wrld)) *acl2-property-unbound*)))
(let ((pairs (cddr (car wrld))))
(cond ((null pairs)
(augmented-runes-after nume (cdr wrld) ans))
((> (caar pairs) nume)
(augmented-runes-after nume
(cdr wrld)
(append pairs ans)))
((<= (car (car (last pairs))) nume)
(reverse ans))
(t (revappend ans
(augmented-runes-after-1 nume pairs))))))
(t
(augmented-runes-after nume (cdr wrld) ans))))
(defun extend-set-difference-theories (thy1 thy2 nm1 wrld)
; Thy1 is an augmented runic theory whose numes do not exceed the given nume,
; nm1. Thy2 is an augmented runic theory whose first nume nm2 exceeds the
; integer nm1.
; Let thy1+ be the augmented theory that extends thy1 with all (nume . rune)
; pairs in wrld for which nm1 < nume. (In our intended application, there is
; an enabled-structure whose theory-array is a header consed onto thy1 and
; whose index-of-last-enabling is nm1.) We return the set-difference of thy1+
; and thy2.
; Note that thy1+ = thy1++ U thy1, where thy1++ is the list of (nume . rune)
; pairs defined above. So thy1+ \ thy2 = (thy1++ \ thy2) U (thy1 \ thy2),
; which we can compute as an augmented runic theory as clearly shown in the
; code below.
; We aren't careful with respect to redefinition (due to our use of
; augmented-runes-after). However, we intend to use this function only when
; certifying books, where redefinition is presumably nonexistent (at least,
; without a trust tag).
(let ((thy1++ (augmented-runes-after nm1 wrld nil)))
(append (set-difference-augmented-theories thy1++ thy2 nil)
(set-difference-augmented-theories thy1 thy2 nil))))
(defun useless-runes-ens (ens wrld state)
; This function returns a modified ens derived for the current event, if any,
; from the @useless-runes.lsp file if there is one. If there is no such
; current event then we return ens.
; Notice that we use load-theory-into-enabled-structure-1 below rather than
; load-theory-into-enabled-structure. That is because for efficiency, we don't
; check theory-invariants for the event-level modification of ens when using
; @useless-runes.lsp. After all, there is already no guarantee that proofs
; will go through when disabling useless-runes; also, most theory-invariants
; probably insist that certain runes can't be simultaneously enabled, and that
; would be maintained by disabling some runes. We considered checking
; theory-invariants before writing out an entry to the @useless-runes.lsp file.
; That wouldn't be perfect, since it's conceivable that the theory-invariant
; would fail at that point but would succeed in a hint given to "Goal", which
; might the enabled-structure actually used by that event.
(let ((active-useless-runes (active-useless-runes state)))
(cond
(active-useless-runes
(let* ((ens-theory (cdr (access enabled-structure ens :theory-array)))
(index-of-last-enabling (access enabled-structure ens
:index-of-last-enabling))
(ext-p (< index-of-last-enabling
(caar active-useless-runes))))
(mv-let (index-of-last-enabling thy)
(cond (ext-p
(mv (caar active-useless-runes)
(extend-set-difference-theories ens-theory
active-useless-runes
index-of-last-enabling
wrld)))
(t
(mv index-of-last-enabling
(set-difference-augmented-theories ens-theory
active-useless-runes
nil))))
(load-theory-into-enabled-structure-1
thy
(if ext-p 'ext-p t) ; augmented-p
ens
t ; incrmt-array-name-info
index-of-last-enabling wrld))))
(t ens))))
(defmacro with-useless-runes (name wrld form)
; Name is the name of an event currently being processed. This macro is
; employed to wrap the given form in code that manages the reading or writing
; of a @useless-runes.lsp file, when appropriate. See with-useless-runes-aux
; for more details; in particular, (with-useless-runes name wrld form) is
; essentially just form when skipping proofs.
; When reading such a file (during certification or execution of an ld call),
; then state global useless-runes is a useless-runes record whose :tag and
; :data are 'FAST-ALIST and fal, a fast-alist. If name is associated with a
; non-empty list of augmented runic theories in fal, the first such theory,
; thy, is popped from that list, but during the processing of form the
; useless-runes record is changed to have :tag and data 'THEORY and thy.
; When writing such a file we get a suitable channel from the useless-runes
; record.
`(let ((wur-name ,name)
(wur-wrld ,wrld))
(mv-let (r/w wur-1 wur-2)
(with-useless-runes-aux wur-name state)
(pprogn
(case r/w
((read #+acl2-par read-but-skip)
(f-put-global 'useless-runes wur-2 state))
(write (prog2$ (accumulated-persistence t) state))
(otherwise state))
(state-global-let*
((useless-runes
(case r/w
(read wur-1)
(otherwise wur-2)))
(global-enabled-structure
(case r/w
(read (useless-runes-ens (ens state) wur-wrld state))
(otherwise (ens state)))))
(mv-let (erp val state)
(check-vars-not-free (wur-name wur-wrld wur-1 wur-2)
,form)
(case r/w
(write (pprogn (print-useless-runes wur-name
(car wur-1)
(cdr wur-1)
state)
(prog2$ (accumulated-persistence nil)
(mv erp val state))))
(otherwise (mv erp val state)))))))))
(defun verify-guards-old-dcl-alist (fn wrld)
; Return an alist mapping each of :guard-hints, :guard-debug, and
; :guard-simplify to their values in the existing (latest) definition of fn,
; if any; else return nil.
(let ((ev (get-defun-event fn wrld)))
(and ev
(assert$
(and (consp ev)
(member-eq (car ev) '(defun defund)))
(let* ((dcls (butlast (cdddr ev) 1))
(h-lst (fetch-dcl-field :guard-hints dcls))
(d-lst (fetch-dcl-field :guard-debug dcls))
(s-lst (fetch-dcl-field :guard-simplify dcls)))
(append (and h-lst (list (cons :guard-hints (car h-lst))))
(and d-lst (list (cons :guard-debug (car d-lst))))
(and s-lst (list (cons :guard-simplify (car s-lst))))))))))
(defun verify-guards-fn (name state
hints hints-p
otf-flg
guard-debug guard-debug-p
guard-simplify guard-simplify-p
event-form)
; Important Note: Don't change the formals of this function without
; reading the *initial-event-defmacros* discussion in axioms.lisp.
(when-logic
"VERIFY-GUARDS"
(with-ctx-summarized
(cond ((and (null hints)
(null otf-flg))
(msg "( VERIFY-GUARDS ~x0)"
name))
(t (cons 'verify-guards name)))
(let* ((wrld (w state))
(event-form (or event-form
(list* 'verify-guards
name
(append
(if hints
(list :hints hints)
nil)
(if otf-flg
(list :otf-flg otf-flg)
nil)))))
(assumep (or (eq (ld-skip-proofsp state) 'include-book)
(eq (ld-skip-proofsp state) 'include-book-with-locals)
(eq (ld-skip-proofsp state) 'initialize-acl2)))
(old-alist (and (symbolp name)
(verify-guards-old-dcl-alist name wrld)))
(guard-simplify (if (and (not guard-simplify-p)
(assoc-eq :guard-simplify old-alist))
(cdr (assoc-eq :guard-simplify old-alist))
guard-simplify)))
(er-let* ((names (chk-acceptable-verify-guards name t guard-simplify ctx
wrld state)))
(cond
((eq names 'redundant)
(stop-redundant-event ctx state))
(t (enforce-redundancy
event-form ctx wrld
(with-useless-runes
name
wrld
(let ((hints
(if (and (not hints-p)
(assoc-eq :guard-hints old-alist))
(cdr (assoc-eq :guard-hints old-alist))
hints))
(guard-debug
(if (and (not guard-debug-p)
(assoc-eq :guard-debug old-alist))
(cdr (assoc-eq :guard-debug old-alist))
guard-debug)))
(er-let*
((hints (if assumep
(value nil)
(translate-hints+
(cons "Guard Lemma for" name)
hints
(default-hints wrld)
ctx wrld state)))
(pair (verify-guards-fn1 names hints otf-flg guard-debug
guard-simplify ctx state)))
; pair is of the form (wrld1 . ttree)
(er-progn
(chk-assumption-free-ttree (cdr pair) ctx state)
(install-event name
event-form
'verify-guards
0
(cdr pair)
nil
nil
nil
(car pair)
state))))))))))
:event-type 'verify-guards
:event event-form)))
; That completes the implementation of verify-guards. We now return
; to the development of defun itself.
; Here is the short-cut used when we are introducing :program functions.
; The super-defun-wart operations are not so much concerned with the
; :program defun-mode as with system functions that need special treatment.
; The wonderful super-defun-wart operations should not, in general, mess with
; the primitive state accessors and updaters. They have to do with a
; boot-strapping problem that is described in more detail in STATE-STATE in
; axioms.lisp.
; The following table has gives the proper STOBJS-IN and STOBJS-OUT
; settings for the indicated functions.
; Warning: If you ever change this table so that it talks about stobjs other
; than STATE, then reconsider oneify-cltl-code. These functions assume that if
; stobjs-in from this table is non-nil then special handling of STATE is
; required; or, at least, they did before Version_2.6.
(defconst *super-defun-wart-table*
; fn stobjs-in stobjs-out
'((COERCE-STATE-TO-OBJECT (STATE) (NIL))
(COERCE-OBJECT-TO-STATE (NIL) (STATE))
(USER-STOBJ-ALIST (STATE) (NIL))
(UPDATE-USER-STOBJ-ALIST (NIL STATE) (STATE))
; Do not add entry for READ-USER-STOBJ-ALIST, which is in *stobjs-out-invalid*.
(WRITE-USER-STOBJ-ALIST (NIL NIL STATE) (STATE))
(STATE-P (STATE) (NIL))
(OPEN-INPUT-CHANNEL-P (NIL NIL STATE) (NIL))
(OPEN-OUTPUT-CHANNEL-P (NIL NIL STATE) (NIL))
(OPEN-INPUT-CHANNEL-ANY-P (NIL STATE) (NIL))
(OPEN-OUTPUT-CHANNEL-ANY-P (NIL STATE) (NIL))
(READ-CHAR$ (NIL STATE) (NIL STATE))
(PEEK-CHAR$ (NIL STATE) (NIL))
(READ-BYTE$ (NIL STATE) (NIL STATE))
(READ-OBJECT (NIL STATE) (NIL NIL STATE))
(READ-ACL2-ORACLE (STATE) (NIL NIL STATE))
(READ-ACL2-ORACLE@PAR (STATE) (NIL NIL))
(READ-RUN-TIME (STATE) (NIL STATE))
(READ-IDATE (STATE) (NIL STATE))
(PRINC$ (NIL NIL STATE) (STATE))
(WRITE-BYTE$ (NIL NIL STATE) (STATE))
(PRINT-OBJECT$-FN (NIL NIL NIL STATE) (STATE))
(GET-GLOBAL (NIL STATE) (NIL))
(BOUNDP-GLOBAL (NIL STATE) (NIL))
(MAKUNBOUND-GLOBAL (NIL STATE) (STATE))
(PUT-GLOBAL (NIL NIL STATE) (STATE))
(GLOBAL-TABLE-CARS (STATE) (NIL))
(OPEN-INPUT-CHANNEL (NIL NIL STATE) (NIL STATE))
(OPEN-OUTPUT-CHANNEL (NIL NIL STATE) (NIL STATE))
(GET-OUTPUT-STREAM-STRING$-FN (NIL STATE) (NIL NIL STATE))
(CLOSE-INPUT-CHANNEL (NIL STATE) (STATE))
(CLOSE-OUTPUT-CHANNEL (NIL STATE) (STATE))
(SYS-CALL-STATUS (STATE) (NIL STATE))))
(defun project-out-columns-i-and-j (i j table)
(cond
((null table) nil)
(t (cons (cons (nth i (car table)) (nth j (car table)))
(project-out-columns-i-and-j i j (cdr table))))))
(defconst *super-defun-wart-binding-alist*
(project-out-columns-i-and-j 0 2 *super-defun-wart-table*))
(defconst *super-defun-wart-stobjs-in-alist*
(project-out-columns-i-and-j 0 1 *super-defun-wart-table*))
(defun super-defun-wart-bindings (bindings)
(cond ((null bindings) nil)
(t (cons (or (assoc-eq (caar bindings)
*super-defun-wart-binding-alist*)
(car bindings))
(super-defun-wart-bindings (cdr bindings))))))
(defun store-stobjs-ins (names stobjs-ins w)
(cond ((null names) w)
(t (store-stobjs-ins (cdr names) (cdr stobjs-ins)
(putprop (car names) 'stobjs-in
(car stobjs-ins) w)))))
(defun store-super-defun-warts-stobjs-in (names wrld)
; Store the built-in stobjs-in values of the super defuns among names, if any.
(cond
((null names) wrld)
((assoc-eq (car names) *super-defun-wart-stobjs-in-alist*)
(store-super-defun-warts-stobjs-in
(cdr names)
(putprop (car names) 'stobjs-in
(cdr (assoc-eq (car names) *super-defun-wart-stobjs-in-alist*))
wrld)))
(t (store-super-defun-warts-stobjs-in (cdr names) wrld))))
(defun collect-old-nameps (names wrld)
(cond ((null names) nil)
((new-namep (car names) wrld)
(collect-old-nameps (cdr names) wrld))
(t (cons (car names) (collect-old-nameps (cdr names) wrld)))))
(defun put-invariant-risk1 (new-fns body-fns wrld)
(cond
((endp body-fns) wrld)
(t (let ((risk-fn
; Risk-fn can be :built-in or a function symbol; see put-invariant-risk.
(getpropc (car body-fns) 'invariant-risk nil wrld)))
(cond (risk-fn (putprop-x-lst1 new-fns 'invariant-risk risk-fn wrld))
(t (put-invariant-risk1 new-fns (cdr body-fns) wrld)))))))
(defun stobjs-guard-only-lst (lst wrld)
; See stobjs-guard-only. Here we do an unnecessary check that the arglist
; consists of a single variable, simply as an optimization that can avoid the
; world lookup done by stobj-recognizer-p.
(cond ((endp lst) t)
(t (and (let ((term (car lst)))
(and (nvariablep term)
(symbolp (ffn-symb term))
(fargs term) ; not nil
(null (cdr (fargs term)))
(variablep (fargn term 1))
(stobj-recognizer-p (ffn-symb term) wrld)))
(stobjs-guard-only-lst (cdr lst) wrld)))))
(defun stobjs-guard-only (guard wrld)
; This function recognizes when guard is a conjunction of stobj recognizer
; calls. There are an implicit function and its stobjs-out that we could pass
; in explicitly, but we only call this for executable functions, so there is no
; need to consider the stobjs-out; we already check elsewhere that the guard is
; well-formed, which guarantees that if a term is a call of a stobj recognizer,
; then it must be called on a declared stobj name.
(stobjs-guard-only-lst (flatten-ands-in-lit guard) wrld))
(defun remove-guard-t (names guards wrld acc)
(cond ((endp names) acc)
(t (remove-guard-t (cdr names)
(cdr guards)
wrld
(if (or (equal (car guards) *t*)
(stobjs-guard-only (car guards)
wrld))
acc
(cons (car names) acc))))))
(defconst *boot-strap-invariant-risk-alist*
; We put a non-nil 'invariant-risk property on every function that might
; violate some ACL2 invariant, if called on arguments that fail to satisfy that
; function's guard. Also see put-invariant-risk.
; This constant should map to t all function symbols that might violate an ACL2
; invariant. See check-invariant-risk. We don't include compress1 or
; compress2 because we believe they don't write out of bounds. We explicitly
; map some function symbols to nil to indicate that our trust that they don't
; have invariant-risk (in spite of calling a function that does).
; In the case of mutual-recursion, only the first name in the nest should be a
; key of this alist; see put-invariant-risk.
; At one point we thought we should mark with invariant-risk all functions that
; have raw code and have state as a formal. But we see almost no way to
; violate an invariant by misguided updates of the (fictional) live state. For
; example, state-p1 specifies that the global-table is an
; ordered-symbol-alistp, but there is no way to get one's hands directly on the
; global-table; and state-p1 also specifies that plist-worldp holds of the
; logical world, and we ensure that by making set-w and related functions
; untouchable. The only exceptions are those mapped to t below, as is checked
; by the function check-invariant-risk. If new exceptions arise, then we
; should map them to t below.
'((aset1 . t) ; could write past the end of the real array
(aset2 . t) ; could write past the end of the real array
(aset1-lst . t)
(check-sum-inc . nil)
(update-iprint-ar-fal . nil)
; Aset1-trusted is an untouchable version of aset1 that we use in our source
; code when we believe that there is no invariant-risk. We could create
; similar trusted analogues of other built-ins mapped to t above.
(aset1-trusted . nil)))
(defun put-invariant-risk (names bodies non-executablep symbol-class guards
wrld)
; We want to avoid the following situation: the raw Lisp version of some
; function occurring in bodies leads to an ill-guarded function call that
; causes an ACL2 invariant to become false.
; Each updater f with non-t type or array type that is introduced by defstobj
; or defabsstobj gets an 'invariant-risk property of f. A built-in function
; may get an 'invariant-risk property; see initialize-invariant-risk. The
; present function, put-invariant-risk, propagates these 'invariant-risk
; properties up through callers.
; When we call all-fnnames1-exec below, we are ignoring :logic code from mbe
; calls. To see that this is sound, first note that we are determining when
; there is a risk of bypassing guard checks that would avoid invariant
; violations. If we are executing :logic code from an mbe call, then we must
; be in the *1* code for a :logic mode function, since :program mode functions
; always execute the :exec code of an mbe call (see oneify), as does raw Lisp
; code. But invariants are checked (in particular, by checking guards for live
; stobj manipulation) when making *1* calls of :logic mode functions. There is
; actually one other case that all-fnnames1-exec ignores function symbols in
; the call tree: it does not collect function symbol F from (ec-call (F ...)).
; But in this case, *1*F or *1*F$INLINE is called, and if there is a non-nil
; 'invariant-risk property for F or F$INLINE (respectively), then we trust that
; oneify has laid down suitable *1* code for F (or F$INLINE) to preserve
; invariants, so there is no risk to bypassing guards in the evaluation of
; bodies.
(cond ((or non-executablep
(null (get-register-invariant-risk-world wrld)))
wrld)
(t (let ((new-fns (if (eq symbol-class :common-lisp-compliant)
(remove-guard-t names guards wrld nil)
names)))
(cond
((null new-fns) ; optimization
wrld)
(t (let ((pair
; The following assoc-eq will be false except during the boot-strap unless the
; function is being redefined. It was tempting to rule out redefinition, but
; the user who redefines a system function is essentially taking on the role of
; implementor. That responsibility, together with a likely desire to preserve
; the special status of the function symbol with respect to invariant-risk,
; seems to make it appropriate not to exempt redefinition here.
; There's potentially a hole created by testing only (car new-fns), since if
; new-fns has more than one element (presumably because of mutual-recursion)
; then maybe it's one of those other members of new-fns besides the first that
; is a key of *boot-strap-invariant-risk-alist*. We'll just have to be aware
; of that when forming *boot-strap-invariant-risk-alist*.
(assoc-eq (car new-fns)
*boot-strap-invariant-risk-alist*)))
(cond
(pair (if (cdr pair)
(putprop-x-lst1 new-fns
'invariant-risk
(car new-fns)
wrld)
wrld))
(t (put-invariant-risk1 new-fns
(all-fnnames1-exec t bodies nil)
wrld))))))))))
(defun defuns-fn-short-cut (loop$-recursion-checkedp
loop$-recursion
names docs pairs guards measures split-types-terms
bodies non-executablep ctx wrld state)
; This function is called by defuns-fn when the functions to be defined are
; :program. It short cuts the normal put-induction-info and other such
; analysis of defuns. The function essentially makes the named functions look
; like primitives in the sense that they can be used in terms and they can be
; evaluated on explicit constants but no axioms or rules are available about
; them. In particular, we do not store 'def-bodies, type-prescriptions, or any
; of the recursion/induction properties normally associated with defuns.
; The prover will reject a formula that contains a call of a :program mode
; function.
; We do take care of the documentation database.
; Like defuns-fn0, this function returns a pair consisting of the new world and
; a tag-tree recording the proofs that were done.
; Quick Refresher on Badged :Program Mode Functions
; :Program mode functions can be badged and can be used, even in formulas
; submitted to the prover, in apply$ contexts, i.e., in quoted lambda objects,
; lambda$ and loop$ and other scions. Thus, such functions can find their way
; into the prover -- but the prover should NEVER encounter a first-order
; application of :program mode function. That is, while the prover might see
; (apply$ 'progmode-fn (list x)) or (collect$ (lambda$ (e) (progmode-fn e))
; lst) it should never see (progmode-fn x) or (progmode-fn (car lst)). In
; particular, when considering rewriting the body of a well-formed lambda
; object the prover must check that there are no :program mode functions in it.
; This can get tricky.
; E.g., one might be tempted to expand (apply$ (lambda$ (e)(progmode-fn e))
; '(5)) to (progmode-fn '5) by beta reduction, but that would be wrong! It
; actually expands to (ev$ '(progmode-fn e) '((e . 5))) which stops because in
; the proof theory (badge 'progmode-fn) is undefined. It is not defined unless
; a warrant for progmode-fn is available and that can't happen until
; progmode-fn is promoted to :logic mode. Note also that while (badge
; 'progmode-fn) is undefined in the proof theory, (badge 'progmode-fn) will
; evaluate to a badge in the evaluation theory if progmode-fn has been badged.
; It's the warrant that moves this knowledge into the proof theory.
; Another issue is how we handle loop$-recursive :program mode functions. At
; the moment defuns-fn0, the only caller of defuns-fn-short-cut, causes an
; error if a :program mode function has a :loop$-recursion t xarg. So we never
; get to this function if loop$-recursion is set.
(declare (ignore docs pairs))
(prog2$
(choke-on-loop$-recursion loop$-recursion-checkedp
names
bodies
'defuns-fn-short-cut)
(er-progn
(cond
((and (null (cdr names)) ; single function
(not (equal (car measures) *no-measure*)) ; explicit measure
(not loop$-recursion)
(not (ffnnamep-mod-mbe (car names) (car bodies)))) ; not recursive
; Warning: Keep the test just above in sync with putprop-recursivep-lst, in the
; sense that a measure is legal only for a singly-recursive function or a list
; of at least two functions.
(maybe-warn-or-error-on-non-rec-measure (car names) ctx wrld state))
(t (value nil)))
(let* (#-acl2-save-unnormalized-bodies
(boot-strap-flg (global-val 'boot-strap-flg wrld))
(wrld0 (cond (non-executablep (putprop-x-lst1 names 'non-executablep
non-executablep
wrld))
(t wrld)))
(wrld1 (cond
#-acl2-save-unnormalized-bodies
(boot-strap-flg wrld0)
(t (putprop-x-lst2 names 'unnormalized-body bodies wrld0))))
(wrld2 (put-invariant-risk
names
bodies
non-executablep
:program ; symbol-class
guards
(putprop-x-lst2-unless
names 'guard guards *t*
(putprop-x-lst2-unless
names 'split-types-term split-types-terms *t*
wrld1)))))
(value (cons wrld2 nil))))))
; Now we develop the output for the defun event.
(defun print-defun-msg/collect-type-prescriptions (names wrld)
; This function returns two lists, a list of names in names with
; trivial type-prescriptions (i.e., NIL 'type-prescriptions property)
; and an alist that pairs names in names with the term representing
; their (non-trivial) type prescriptions.
(cond
((null names) (mv nil nil))
(t (mv-let (fns alist)
(print-defun-msg/collect-type-prescriptions (cdr names) wrld)
(let ((lst (getpropc (car names) 'type-prescriptions nil wrld)))
(cond
((null lst)
(mv (cons (car names) fns) alist))
(t (mv fns
(cons
(cons (car names)
(untranslate
(access type-prescription (car lst) :corollary)
t wrld))
alist)))))))))
(defun print-defun-msg/type-prescriptions1 (alist simp-phrase col state)
; See print-defun-msg/type-prescriptions. We print out a string of
; phrases explaining the alist produced above. We return the final
; col and state. This function used to be a tilde-* phrase, but
; you cannot get the punctuation after the ~xt commands.
(cond ((null alist) (mv col state))
((null (cdr alist))
(fmt1 "the type of ~xn is described by the theorem ~Pt0. ~#p~[~/We ~
used ~*s.~]~|"
(list (cons #\n (caar alist))
(cons #\t (cdar alist))
(cons #\0 (term-evisc-tuple nil state))
(cons #\p (if (nth 4 simp-phrase) 1 0))
(cons #\s simp-phrase))
col
(proofs-co state)
state nil))
((null (cddr alist))
(fmt1 "the type of ~xn is described by the theorem ~Pt0 ~
and the type of ~xm is described by the theorem ~Ps0.~|"
(list (cons #\n (caar alist))
(cons #\t (cdar alist))
(cons #\0 (term-evisc-tuple nil state))
(cons #\m (caadr alist))
(cons #\s (cdadr alist)))
col
(proofs-co state)
state nil))
(t
(mv-let (col state)
(fmt1 "the type of ~xn is described by the theorem ~Pt0, "
(list (cons #\n (caar alist))
(cons #\t (cdar alist))
(cons #\0 (term-evisc-tuple nil state)))
col
(proofs-co state)
state nil)
(print-defun-msg/type-prescriptions1 (cdr alist) simp-phrase
col state)))))
(defun print-defun-msg/type-prescriptions (names ttree wrld col state)
; This function prints a description of each non-trivial
; type-prescription for the functions names. It assumes that at the
; time it is called, it is printing in col. It returns the final col,
; and the final state.
(let ((simp-phrase (tilde-*-simp-phrase ttree)))
(mv-let
(fns alist)
(print-defun-msg/collect-type-prescriptions names wrld)
(cond
((null alist)
(fmt1
" We could deduce no constraints on the type of ~#0~[~&0.~/any of ~
the functions in the clique.~]~#1~[~/ However, in normalizing the ~
definition~#0~[~/s~] we used ~*2.~]~%"
(list (cons #\0 names)
(cons #\1 (if (nth 4 simp-phrase) 1 0))
(cons #\2 simp-phrase))
col
(proofs-co state)
state nil))
(fns
(mv-let
(col state)
(fmt1
" We could deduce no constraints on the type of ~#f~[~vf,~/any of ~
~vf,~] but we do observe that "
(list (cons #\f fns))
col
(proofs-co state)
state nil)
(print-defun-msg/type-prescriptions1 alist simp-phrase col state)))
(t
(mv-let
(col state)
(fmt1
" We observe that " nil col (proofs-co state)
state nil)
(print-defun-msg/type-prescriptions1 alist simp-phrase
col state)))))))
(defun simple-signaturep (fn wrld)
; A simple signature is one in which no stobjs or dfs are involved and the
; output is a single value.
(and (all-nils (stobjs-in fn wrld))
(let ((stobjs-out
; We call getprop rather than calling stobjs-out, because this code may run
; with fn = return-last, and the function stobjs-out causes an error in that
; case. We don't mind treating return-last as an ordinary function here.
(getpropc fn 'stobjs-out '(nil) wrld)))
(and (null (cdr stobjs-out))
(not (eq (car stobjs-out) :df))))))
(defun all-simple-signaturesp (names wrld)
(cond ((endp names) t)
(t (and (simple-signaturep (car names) wrld)
(all-simple-signaturesp (cdr names) wrld)))))
(defun print-defun-msg/signatures1 (names wrld state)
(cond
((endp names) state)
((not (simple-signaturep (car names) wrld))
(pprogn
(fms "~x0 => ~x1."
(list
(cons #\0
(cons (car names)
(prettyify-stobj-flags (stobjs-in (car names) wrld))))
(cons #\1 (prettyify-stobjs-out
; We call getprop rather than calling stobjs-out, because this code may run
; with fn = return-last, and the function stobjs-out causes an error in that
; case. We don't mind treating return-last as an ordinary function here.
(getpropc (car names) 'stobjs-out '(nil) wrld))))
(proofs-co state)
state
nil)
(print-defun-msg/signatures1 (cdr names) wrld state)))
(t (print-defun-msg/signatures1 (cdr names) wrld state))))
(defun print-defun-msg/signatures (names wrld state)
(cond ((all-simple-signaturesp names wrld)
state)
((cdr names)
(pprogn
(fms "The Non-simple Signatures:" nil (proofs-co state) state nil)
(print-defun-msg/signatures1 names wrld state)
(newline (proofs-co state) state)))
(t (pprogn
(print-defun-msg/signatures1 names wrld state)
(newline (proofs-co state) state)))))
(defun print-defun-msg (names ttree wrld col state)
; Once upon a time this function printed more than just the type
; prescription message. We've left the function here to handle that
; possibility in the future. This function returns the final state.
; This function increments timers. Upon entry, the accumulated time
; is charged to 'other-time. The time spent in this function is
; charged to 'print-time.
(cond ((ld-skip-proofsp state)
state)
(t
(io? event nil state
(names ttree wrld col)
(pprogn
(increment-timer 'other-time state)
(mv-let (erp ttree state)
(accumulate-ttree-and-step-limit-into-state ttree :skip state)
(declare (ignore erp))
(mv-let (col state)
(print-defun-msg/type-prescriptions names ttree
wrld col state)
(declare (ignore col))
(pprogn
(print-defun-msg/signatures names wrld state)
(increment-timer 'print-time state)))))))))
(defun get-ignores (lst)
(cond ((null lst) nil)
(t (cons (ignore-vars
(fourth (car lst)))
(get-ignores (cdr lst))))))
(defun get-ignorables (lst)
(cond ((null lst) nil)
(t (cons (ignorable-vars
(fourth (car lst)))
(get-ignorables (cdr lst))))))
(defun irrelevant-vars (dcls)
(cond ((null dcls) nil)
((eq (caar dcls) 'irrelevant)
(append (cdar dcls) (irrelevant-vars (cdr dcls))))
(t (irrelevant-vars (cdr dcls)))))
(defun get-irrelevants (lst)
(cond ((null lst) nil)
(t (cons (irrelevant-vars
(fourth (car lst)))
(get-irrelevants (cdr lst))))))
(defun chk-all-stobj-names (lst src msg ctx wrld state)
; Cause an error if any element of lst is not a legal stobj name in wrld. Src
; is :stobjs, 'signature-stobjs, or :global-stobjs, indicating whether this
; check is done in a context where we are checking the :stobjs keyword value,
; stobjs in a signature, or the :global-stobjs keyword value.
(cond ((endp lst) (value nil))
((not (stobjp (car lst) t wrld))
(er soft ctx
"Every name used as a stobj ~@0 must have been previously ~
defined as a single-threaded object with defstobj or ~
defabsstobj. ~x1 is used as stobj name ~#2~[~/in ~@2 ~]but has ~
not been defined as a stobj."
(cond ((eq src :stobjs)
"with a :STOBJS keyword")
((eq src :stobjs?)
"(whether declared explicitly via the :STOBJS keyword ~
argument or implicitly via *-notation)")
((eq src :global-stobjs)
"with a :GLOBAL-STOBJS keyword")
(t (er hard ctx
"Implementation error: unexpected case of src, ~x0. ~
Please report this error to the ACL2 implementors."
src)))
(car lst)
msg))
(t (chk-all-stobj-names (cdr lst) src msg ctx wrld state))))
(defun get-declared-kwd-names (kwd edcls ctx wrld state)
; Kwd is (as of this writing) :STOBJS or :DFS. Each element of edcls is the
; cdr of a DECLARE form. We look for the ones of the form (XARGS ...) and find
; the first value of keyword kwd in each such xargs. We know there is at most
; one kwd occurrence in each xargs by chk-dcl-lst. We union together all the
; values of that keyword, after checking that each value is legal. We return
; the list of declared kwd names or cause an error.
; Keep this in sync with get-declared-stobjs (which does not do any checking
; and returns a single value).
(cond ((endp edcls) (value nil))
((eq (caar edcls) 'xargs)
(let* ((temp (assoc-keyword kwd (cdar edcls)))
(lst (cond ((null temp) nil)
((null (cadr temp)) nil)
((atom (cadr temp))
(list (cadr temp)))
(t (cadr temp)))))
(cond
(lst
(cond
((not (symbol-listp lst))
(er soft ctx
"The value specified for the ~x0 xarg must be a true ~
list of symbols and ~x1 is not."
kwd lst))
(t (er-progn
(case kwd
(:stobjs (chk-all-stobj-names lst
:STOBJS
(msg "... :stobjs ~x0 ..."
(cadr temp))
ctx wrld state))
(:dfs (value nil))
(t (value (er hard 'get-declared-kwd-names
"Implementation error: Unexpected keyword, ~
~x0"
kwd))))
(er-let*
((rst (get-declared-kwd-names kwd (cdr edcls) ctx wrld
state)))
(value (union-eq lst rst)))))))
(t (get-declared-kwd-names kwd (cdr edcls) ctx wrld state)))))
(t (get-declared-kwd-names kwd (cdr edcls) ctx wrld state))))
(defun get-stobjs-in-lst (lst defun-mode ctx wrld state)
; Lst is a list of ``fives'' as computed in chk-acceptable-defuns.
; Each element is of the form (fn args "doc" edcls body). We know the
; args are legal arg lists, but nothing else.
; Unless we cause an error, we return a list in 1:1 correspondence
; with lst containing the STOBJS-IN flags for each fn. This involves
; three steps. First we recover from the edcls the declared :stobjs.
; We augment those with STATE, if STATE is in formals, which is always
; implicitly a stobj, if STATE is in the formals. We confirm that all
; the declared stobjs are indeed stobjs in wrld. Then we compute the
; stobj flags using the formals and the declared stobjs.
(cond ((null lst) (value nil))
(t (let ((fn (first (car lst)))
(formals (second (car lst))))
(er-let* ((dcl-stobj-names
(get-declared-kwd-names :stobjs
(fourth (car lst))
ctx wrld state))
(dcl-df-names-1
(get-declared-kwd-names :dfs
(fourth (car lst))
ctx wrld state))
(dcl-df-names-2
(value (extend-known-dfs-with-declared-df-types
(fourth (car lst))
nil)))
(dcl-stobj-namesx
(cond ((and (member-eq 'state formals)
(not (member-eq 'state dcl-stobj-names)))
(er-progn
(cond
((and (eq defun-mode :logic)
(function-symbolp fn wrld))
; In this case, we skip the polite check that state can be a formal without
; declaring it a stobj. This way, verify-termination can succeed in the case
; that the original :program mode definition was evaluated in a world with
; state-ok but the current definition is not.
(value nil))
(t (chk-state-ok ctx wrld state)))
(value (cons 'state dcl-stobj-names))))
(t (value dcl-stobj-names)))))
(cond
((not (subsetp-eq dcl-stobj-namesx formals))
(er soft ctx
"The stobj name~#0~[ ~&0 is~/s ~&0 are~] ~
declared but not among the formals of ~x1. ~
This generally indicates some kind of ~
typographical error and is illegal. Declare ~
only those stobj names listed in the formals. ~
The formals list of ~x1 is ~x2."
(set-difference-equal dcl-stobj-namesx formals)
fn
formals))
((or (intersectp-eq dcl-stobj-namesx dcl-df-names-1)
(intersectp-eq dcl-stobj-namesx dcl-df-names-2))
(mv-let (n int)
(if (intersectp-eq dcl-stobj-namesx dcl-df-names-1)
(mv 0
(intersection-eq dcl-stobj-namesx dcl-df-names-1))
(mv 1
(intersection-eq dcl-stobj-namesx dcl-df-names-2)))
(er soft ctx
"The formal~#0~[ ~&0 of function ~x1 is~/s of ~
function ~x1 ~&0 are each~] declared both with ~
xargs :STOBJS and with ~#2~[xargs :DFS~/a ~
double-float type declaration~]. But a formal ~
cannot name both a stobj and a df."
int
fn
n)))
(t (er-let* ((others (get-stobjs-in-lst (cdr lst)
defun-mode
ctx wrld state)))
; Note: Wrld is irrelevant below because dcl-stobj-namesx is not T so
; we simply look for the formals that are in dcl-stobj-namesx.
(value
(cons (compute-stobj-flags formals
dcl-stobj-namesx
(union-eq dcl-df-names-1
dcl-df-names-2)
wrld)
others))))))))))
(defun chk-stobjs-out-bound (names bindings ctx state)
(cond ((null names) (value nil))
((translate-unbound (car names) bindings)
(er soft ctx
"Translate failed to determine the output signature of ~
~x0." (car names)))
(t (chk-stobjs-out-bound (cdr names) bindings ctx state))))
(defun store-stobjs-out (names bindings w)
(cond ((null names) w)
(t (store-stobjs-out
(cdr names)
bindings
(putprop (car names) 'stobjs-out
(translate-deref (car names)
bindings) w)))))
(defun unparse-signature (x)
; Suppose x is an internal form signature, i.e., (fn formals stobjs-in
; stobjs-out). Then we return an external version of it, e.g., ((fn
; . stobjs-in) => (mv . stobjs-out)). This is only used in error
; reporting.
(let* ((fn (car x))
(pretty-flags1 (prettyify-stobj-flags (caddr x)))
(output (prettyify-stobjs-out (cadddr x))))
`((,fn ,@pretty-flags1) => ,output)))
(defun chk-defun-mode (defun-mode ctx state)
(cond ((eq defun-mode :program)
(value nil))
((eq defun-mode :logic)
; We do the check against the value of state global 'program-fns-with-raw-code
; in redefinition-renewal-mode, so that we do it only when reclassifying.
(value nil))
(t (er soft ctx
"The legal defun-modes are :program and :logic. ~x0 is ~
not a recognized defun-mode."
defun-mode))))
(defun dcl-fields1 (lst)
(declare (xargs :guard (plausible-dclsp1 lst)))
(cond ((endp lst) nil)
((member-eq (caar lst) '(type ignore ignorable))
(add-to-set-eq (caar lst) (dcl-fields1 (cdr lst))))
(t (union-eq (evens (cdar lst)) (dcl-fields1 (cdr lst))))))
(defun dcl-fields (lst)
; Lst satisfies plausible-dclsp, i.e., is the sort of thing you would find
; between the formals and the body of a DEFUN. We return a duplicate-free list
; of all the "field names" used in lst, where 'comment indicates a string. Our
; answer is a subset of the union of the values of '(comment type ignore
; ignorable irrelevant) and *xargs-keywords*.
(declare (xargs :guard (plausible-dclsp lst)))
(cond ((endp lst) nil)
((stringp (car lst))
(add-to-set-eq 'comment (dcl-fields (cdr lst))))
(t (union-eq (dcl-fields1 (cdar lst))
(dcl-fields (cdr lst))))))
(defun set-equalp-eq (lst1 lst2)
(declare (xargs :guard (and (true-listp lst1)
(true-listp lst2)
(or (symbol-listp lst1)
(symbol-listp lst2)))))
(and (subsetp-eq lst1 lst2)
(subsetp-eq lst2 lst1)))
(defun non-identical-defp-chk-measures (name new-measures old-measures
justification)
(cond
((equal new-measures old-measures)
nil)
(t
; We could try harder, by translating the new measure and seeing if the set of
; free variables is the same as the old measured subset. But as Sandip Ray
; points out, it might be odd for the new "measure" to be allowed when in fact
; we have proved nothing about it! Also, the new measure would have to be
; translated in order to get its free variables, and we prefer not to pay that
; price (though perhaps it's quite minor). Bottom line: we see no reason for
; anyone to expect a definition to be redundant with an earlier one that has a
; different measure.
(let ((old-measured-subset
(assert$
justification
; Old-measured-subset is used only if chk-measure-p is true. In that case, if
; the existing definition is non-recursive then we treat the measured subset as
; nil.
(access justification justification :subset))))
(cond
((and (consp new-measures)
(null (cdr new-measures))
(let ((new-measure (car new-measures)))
(or (equal new-measure (car old-measures))
(and (true-listp new-measure)
(eq (car new-measure) :?)
(arglistp (cdr new-measure))
(set-equalp-eq old-measured-subset
(cdr new-measure))))))
nil)
(old-measures
(msg "the proposed and existing definitions for ~x0 differ on their ~
measures. The proposed measure is ~x1 but the existing measure ~
is ~x2. The proposed measure needs to be specified explicitly ~
in fully translated form with :measure (see :DOC xargs), either ~
to be identical to the existing measure or to be a call of :? ~
on the measured subset; for example, ~x3 will serve as the ~
proposed :measure."
name
(car new-measures)
(car old-measures)
(cons :? old-measured-subset)))
(t
(msg "the existing definition for ~x0 does not have an explicitly ~
specified measure. Either remove the :measure declaration from ~
your proposed definition, or else specify a :measure that ~
applies :? to the existing measured subset, for example, ~x1."
name
(cons :? old-measured-subset))))))))
(defun non-identical-defp (def1 def2 chk-measure-p wrld)
; This predicate is used in recognizing redundant definitions. In our intended
; application, def2 will have been successfully processed and def1 is merely
; proposed, where def1 and def2 are each of the form (fn args ...dcls... body)
; and everything is untranslated. Two such tuples are "identical" if their
; fns, args, bodies, types, stobjs, guards, and (if chk-measure-p is true)
; measures are equal -- except that the new measure can be (:? v1 ... vk) if
; (v1 ... vk) is the measured subset for the old definition. We return nil if
; def1 is thus redundant with ("identical" to) def2. Otherwise we return a
; message suitable for printing using " Note that ~@k.".
; Note that def1 might actually be syntactically illegal, e.g., it might
; specify two different :measures. But it is possible that we will still
; recognize it as identical to def2 because the args and body are identical.
; Thus, the syntactic illegality of def1 might not be discovered if def1 is
; avoided because it is redundant. This happens already in redundancy checking
; in defthm: a defthm event is redundant if it introduces an identical theorem
; with the same name -- even if the :hints in the new defthm are ill-formed.
; The idea behind redundancy checking is to allow books to be loaded even if
; they share some events.
; Should we do any checks here related to the :subversive-p field of the
; justification for def2? The concern is that def2 (the old definition) is
; subversive but local, and def1 (the new definition) is not subversive and is
; non-local. But the notion of "subversive" is handled just as well in pass2
; as in pass1, so ultimately def1 will be marked correctly on its
; subversiveness.
(let* ((justification (and chk-measure-p ; optimization
(getpropc (car def2) 'justification nil wrld)))
(all-but-body1 (butlast (cddr def1) 1))
(all-but-body2 (butlast (cddr def2) 1))
; We insist on the syntactic identity of the :ruler-extenders, and then check
; that the default ruler-extenders in the two cases do not ruin the equality of
; the two ruler-extenders. Default ruler-extenders may ruin that equality only
; if both definitions have no explicit ruler-extenders.
(ruler-extenders1-lst (fetch-dcl-field :ruler-extenders
all-but-body1))
(ruler-extenders2-lst (fetch-dcl-field :ruler-extenders
all-but-body2)))
(cond
((and justification
(or (not (equal ruler-extenders1-lst ruler-extenders2-lst))
(and (null ruler-extenders1-lst)
(not (equal (access justification
justification
:ruler-extenders)
(default-ruler-extenders wrld))))))
(msg "the proposed and existing definitions for ~x0 differ on their ~
ruler-extenders (see :DOC ruler-extenders). The proposed ~
ruler-extenders value does not match the existing ruler-extenders ~
for ~x0, namely, ~x1."
(car def1)
(access justification justification :ruler-extenders)))
((equal def1 def2) ; optimization
nil)
((not (eq (car def1) (car def2))) ; check same fn (can this fail?)
(msg "the name of the new event, ~x0, differs from the name of the ~
corresponding existing event, ~x1."
(car def1) (car def2)))
((not (equal (cadr def1) (cadr def2))) ; check same args
(msg "the proposed argument list for ~x0, ~x1, differs from the ~
existing argument list, ~x2."
(car def1) (cadr def1) (cadr def2)))
((not (equal (car (last def1)) (car (last def2)))) ; check same body
(msg "the proposed body for ~x0,~|~%~p1,~|~%differs from the existing ~
body,~|~%~p2.~|~%"
(car def1) (car (last def1)) (car (last def2))))
((not (equal (fetch-dcl-field :non-executable all-but-body1)
(fetch-dcl-field :non-executable all-but-body2)))
(msg "the proposed and existing definitions for ~x0 differ on their ~
:non-executable declarations."
(car def1)))
((not (equal (fetch-dcl-field :type-prescription all-but-body1)
(fetch-dcl-field :type-prescription all-but-body2)))
(msg "the proposed and existing definitions for ~x0 differ on their ~
:type-prescription declarations."
(car def1)))
((flet ((normalize-value
(x)
(cond ((equal x '(nil))
nil)
((or (equal x '(t))
(null x))
t)
(t (er hard 'non-identical-defp
"Internal error: Unexpected value when processing ~
:normalize xargs keyword, ~x0. Please contact ~
the ACL2 implementors."
x)))))
(not (equal (normalize-value
(fetch-dcl-field :normalize all-but-body1))
(normalize-value
(fetch-dcl-field :normalize all-but-body2)))))
(msg "the proposed and existing definitions for ~x0 differ on the ~
values supplied by :normalize declarations."
(car def1)))
((not (let ((stobjs1 (fetch-dcl-field :stobjs all-but-body1))
(stobjs2 (fetch-dcl-field :stobjs all-but-body2)))
(or (equal stobjs1 stobjs2) ; optimization
; Quoting :doc xargs: "The only exception to this rule is state: whether you
; include it or not, state is always treated as a single-threaded object."
; If the two definitions are identical except for how state is declared as a
; stobj, then since the old definition was acceptable, so is the new one.
(equal (remove1-eq 'state stobjs1)
(remove1-eq 'state stobjs2)))))
; We insist that the :STOBJS of the two definitions be identical. Vernon
; Austel pointed out the following bug.
; Define a :program mode function with a non-stobj argument.
; (defun stobjless-fn (stobj-to-be)
; (declare (xargs :mode :program))
; stobj-to-be)
; Use it in the definition of another :program mode function.
; (defun my-callee-is-stobjless (x)
; (declare (xargs :mode :program))
; (stobjless-fn x))
; Then introduce a the argument name as a stobj:
; (defstobj stobj-to-be
; (a-field :type integer :initially 0))
; And reclassify the first function into :logic mode.
; (defun stobjless-fn (stobj-to-be)
; (declare (xargs :stobjs stobj-to-be))
; stobj-to-be)
; If you don't notice the different use of :stobjs then the :program
; mode function my-callee-is-stobjless [still] treats the original
; function as though its argument were NOT a stobj! For example,
; (my-callee-is-stobjless 3) is a well-formed :program mode term
; that treats 3 as a stobj.
(msg "the proposed and existing definitions for ~x0 differ on their ~
:stobjs declarations."
(car def1)))
((not (equal (fetch-dcl-field :dfs all-but-body1)
(fetch-dcl-field :dfs all-but-body2)))
(msg "the proposed and existing definitions for ~x0 differ on their ~
:dfs declarations."
(car def1)))
((not (equal (fetch-dcl-field 'type all-but-body1)
(fetch-dcl-field 'type all-but-body2)))
; Once we removed the restriction that the type and :guard fields of the defs
; be equal. But imagine that we have a strong guard on foo in our current ACL2
; session, but that we then include a book with a much weaker guard. (Horrors!
; What if the new guard is totally unrelated!?) If we didn't make the tests
; below, then presumably the guard on foo would be unchanged by this
; include-book. Suppose that in this book, we have verified guards for a
; function bar that calls foo. Then after including the book, it will look as
; though correctly guarded calls of bar always generate only correctly guarded
; calls of foo, but now that foo has a stronger guard than it did when the book
; was certified, this might not always be the case.
(msg "the proposed and existing definitions for ~x0 differ on their ~
type declarations."
(car def1)))
((let* ((guards1 (fetch-dcl-field :guard all-but-body1))
(guards1-trivial-p (or (null guards1) (equal guards1 '(t))))
(guards2 (fetch-dcl-field :guard all-but-body2))
(guards2-trivial-p (or (null guards2) (equal guards2 '(t)))))
; See the comment above on type and :guard fields. Here, we comprehend the
; fact that omission of a guard is equivalent to :guard t. Of course, it is
; also equivalent to :guard 't and even to :guard (not nil), but we see no need
; to be that generous with our notion of redundancy.
(cond ((and guards1-trivial-p guards2-trivial-p)
nil)
((not (equal guards1 guards2))
(msg "the proposed and existing definitions for ~x0 differ on ~
their :guard declarations."
(car def1)))
; So now we know that the guards are equal and non-trivial. If the types are
; non-trivial too then we need to make sure that the combined order of guards
; and types for each definition are in agreement. The following example shows
; what can go wrong without that check.
; (encapsulate
; ()
; (local (defun foo (x)
; (declare (xargs :guard (consp x)))
; (declare (xargs :guard (consp (car x))))
; x))
; (defun foo (x)
; (declare (xargs :guard (consp (car x))))
; (declare (xargs :guard (consp x)))
; x))
;
; (foo 3) ; hard raw Lisp error!
((not (equal (fetch-dcl-fields '(type :guard) all-but-body1)
(fetch-dcl-fields '(type :guard)
all-but-body2)))
(msg "although the proposed and existing definitions for ~x0 ~
agree on the their type and :guard declarations, they ~
disagree on the combined orders of those declarations.")))))
((let ((split-types1 (fetch-dcl-field :split-types all-but-body1))
(split-types2 (fetch-dcl-field :split-types all-but-body2)))
(or (not (eq (all-nils split-types1) (all-nils split-types2)))
; Catch the case of illegal values in the proposed definition.
(not (boolean-listp split-types1))
(and (member-eq nil split-types1)
(member-eq t split-types1))))
(msg "the proposed and existing definitions for ~x0 differ on their ~
:split-types declarations."
(car def1)))
((not chk-measure-p)
nil)
((null justification)
; The old definition (def2) was non-recursive. Then since the names and bodies
; are identical (as checked above), the new definition (def1) is also
; non-recursive. In this case we don't care about the measures; see the
; comment above about "syntactically illegal".
nil)
(t
(non-identical-defp-chk-measures
(car def1)
(fetch-dcl-field :measure all-but-body1)
(fetch-dcl-field :measure all-but-body2)
justification)))))
(defun identical-defp (def1 def2 chk-measure-p wrld)
; This function is probably obsolete -- superseded by non-identical-defp -- but
; we leave it here for reference by comments.
(not (non-identical-defp def1 def2 chk-measure-p wrld)))
(defun redundant-or-reclassifying-defunp (defun-mode symbol-class
ld-skip-proofsp chk-measure-p def
wrld)
; Def is a defuns tuple such as (fn args ...dcls... body) that has been
; submitted to defuns with mode defun-mode. We determine whether fn is already
; defined in wrld and has an "identical" definition (up to defun-mode). We
; return either nil, a message (cons pair suitable for printing with ~@),
; 'redundant, 'reclassifying, or 'verify-guards. 'Redundant is returned if
; there is an existing definition for fn that is identical-defp to def and has
; mode :program or defun-mode, except that in this case 'verify-guards is
; returned if the symbol-class was :ideal but this definition indicates
; promotion to :common-lisp-compliant. 'Reclassifying is returned if there is
; an existing definition for fn that is identical-defp to def but in mode
; :program while defun-mode is :logic. Otherwise nil or an explanatory
; message, suitable for printing using " Note that ~@0.", is returned.
; Functions further up the call tree will decide what to do with a result of
; 'verify-guards. But a perfectly reasonable action would be to cause an error
; suggesting the use of verify-guards instead of defun.
(cond ((function-symbolp (car def) wrld)
(let* ((wrld1 (decode-logical-name (car def) wrld))
(name (car def))
(val (scan-to-cltl-command (cdr wrld1)))
(chk-measure-p
(and chk-measure-p
; If we are skipping proofs, then we do not need to check the measure. Why
; not? One case is that we are explicitly skipping proofs (with skip-proofs,
; rebuild, set-ld-skip-proofsp, etc.; or, inclusion of an uncertified book), in
; which case all bets are off. Otherwise we are including a certified book,
; where the measured subset was proved correct. This observation satisfies our
; concern, which is that the current redundant definition will ultimately
; become the actual definition because the earlier one is local.
(not ld-skip-proofsp)
; A successful redundancy check may require that the untranslated measure is
; identical to that of the earlier corresponding defun. Without such a check
; we can store incorrect induction information, as exhibited by the "soundness
; bug in the redundancy criterion for defun events" mentioned in :doc
; note-3-0-2. The following examples, which work with Version_3.0.1 but
; (fortunately) not afterwards, build on the aforementioned proof of nil given
; in :doc note-3-0-2, giving further weight to our insistence on the same
; measure if the mode isn't changing from :program to :logic.
; The following example involves redundancy only for :program mode functions.
; (encapsulate
; ()
;
; (local (defun foo (x y)
; (declare (xargs :measure (acl2-count y) :mode :program))
; (if (and (consp x) (consp y))
; (foo (cons x x) (cdr y))
; y)))
;
; (defun foo (x y)
; (declare (xargs :mode :program))
; (if (and (consp x) (consp y))
; (foo (cons x x) (cdr y))
; y))
;
; (verify-termination foo))
;
; (defthm bad
; (atom x)
; :rule-classes nil
; :hints (("Goal" :induct (foo x '(3)))))
;
; (defthm contradiction
; nil
; :rule-classes nil
; :hints (("Goal" :use ((:instance bad (x '(7)))))))
; Note that even though we do not store induction schemes for mutual-recursion,
; the following variant of the first example shows that we still need to check
; measures in that case:
; (set-bogus-mutual-recursion-ok t) ; ease construction of example
;
; (encapsulate
; ()
; (local (encapsulate
; ()
;
; (local (mutual-recursion
; (defun bar (x) x)
; (defun foo (x y)
; (declare (xargs :measure (acl2-count y)))
; (if (and (consp x) (consp y))
; (foo (cons x x) (cdr y))
; y))))
;
; (mutual-recursion
; (defun bar (x) x)
; (defun foo (x y)
; (if (and (consp x) (consp y))
; (foo (cons x x) (cdr y))
; y)))))
; (defun foo (x y)
; (if (and (consp x) (consp y))
; (foo (cons x x) (cdr y))
; y)))
;
; (defthm bad
; (atom x)
; :rule-classes nil
; :hints (("Goal" :induct (foo x '(3)))))
;
; (defthm contradiction
; nil
; :rule-classes nil
; :hints (("Goal" :use ((:instance bad (x '(7))))))) ; |
; After Version_3.4 we no longer concern ourselves with the measure in the case
; of :program mode functions, as we now explain.
; Since verify-termination is now just a macro for make-event, we may view the
; :measure of a :program mode function as nothing more than a hint for use by
; that make-event. So we need think only about definitions (defun, defuns).
; Note that the measure for a :logic mode definition will always come lexically
; from that definition. So for redundancy, soundness only requires that the
; measured subsets agree when the old and new definitions are both in :logic
; mode. We can even change the measure from an existing :program mode
; definition to produce a new :program mode definition, so as to provide a
; better hint for a later verify-termination call.
; One might think that we should do the measures check when the old definition
; is :logic and the new one is :program. But in that case, either the new one
; is redundant or ultimately in :program mode (if the first is local and the
; second is installed on a second pass). Either way, there is no concern: if
; the definition is installed, it will be in program mode and hence its measure
; presents no concern for soundness.
(eq (cadr val) :logic)
(eq defun-mode :logic))))
; The 'cltl-command val for a defun is (defuns :defun-mode ignorep . def-lst)
; where :defun-mode is a keyword (rather than nil which means this was an
; encapsulate or was :non-executable).
(cond ((null val) nil)
((and (consp val)
(eq (car val) 'defuns)
(keywordp (cadr val)))
(cond
((non-identical-defp def
(assoc-eq name (cdddr val))
chk-measure-p
wrld))
; Else, this cltl-command contains a member of def-lst that is identical to
; def.
((eq (cadr val) defun-mode)
(cond ((and (eq symbol-class :common-lisp-compliant)
(eq (symbol-class name wrld) :ideal))
; The following produced a hard error in v2-7, because the second defun was
; declared redundant on the first pass and then installed as
; :common-lisp-compliant on the second pass:
; (encapsulate nil
; (local
; (defun foo (x) (declare (xargs :guard t :verify-guards nil)) (car x)))
; (defun foo (x) (declare (xargs :guard t)) (car x)))
; (thm (equal (foo 3) xxx))
; The above example was derived from one sent by Jared Davis, who proved nil in
; an early version of v2-8 by exploiting this idea to trick ACL2 into
; considering guards verified for a function employing mbe.
; Here, we prevent such promotion of :ideal to :common-lisp-compliant.
'verify-guards)
; The next potential COND branch would avoid redundancy when downgrading from
; :common-lisp-compliant to :ideal. But it is commented out, because there
; were many regression failures; see GitHub Issue 582.
; ((and (eq symbol-class :ideal)
; (eq (symbol-class name wrld)
; :common-lisp-compliant))
;
; ; We have returned 'redundant in this case, but we now realize that doing so
; ; could be problematic. Consider a book with the following events. If the
; ; second definition of foo is redundant on the first pass of certify-book,
; ; then bar will produce an error on the second pass because foo is not
; ; :common-lisp-compliant at that time.
;
; ; (local
; ; (defun foo (x)
; ; (declare (xargs :guard t :verify-guards t))
; ; x))
; ;
; ; (defun foo (x)
; ; (declare (xargs :guard t :verify-guards nil))
; ; x)
; ;
; ; (defun bar (x)
; ; (declare (xargs :guard t))
; ; (foo x))
;
; ; Out of courtesy, given this change to long-standing behavior, we print an
; ; explanatory message.
;
; (msg "it is not redundant to provide a new ~
; definition that specifies the removal of ~
; guard-verified status."))
(t 'redundant)))
((and (eq (cadr val) :program)
(eq defun-mode :logic))
'reclassifying)
(t
; We allow "redefinition" from :logic to :program mode by treating the latter
; as redundant. At one time we thought it should be disallowed because of an
; example like this:
; (encapsulate nil
; (local (defun foo (x) x))
; (defun foo (x) (declare (xargs :mode :program)) x) ; redundant?
; (defthm foo-is-id (equal (foo x) x)))
; We clearly don't want to allow this encapsulation or analogous books. But
; this is prevented by pass 2 of the encapsulate (similarly, but at the book
; level, for certify-book), when ACL2 discovers that foo is now :program mode.
; We need to be careful to avoid similar traps elsewhere.
; It's important to allow such to be redundant in order to avoid the following
; problem, pointed out by Jared Davis. Imagine that one book defines a
; function in :logic mode, while another has an identical definition in
; :program mode followed by verify-termination. Also imagine that both books
; are independently certified. Now imagine, in a fresh session, including the
; first book and then the second. Inclusion of the second causes an error in
; Version_3.4 because of the "downgrade" from :logic mode to :program mode at
; the time the :program mode definition is encountered.
; Finally, note that we are relying on safe-mode! Imagine a book with a local
; :logic mode definition of f followed by a non-local :program mode definition
; of f, followed by a defconst that uses f. Also suppose that the guard of f
; is insufficient to verify its guards; to be specific, suppose (f x) is
; defined to be (car x) with a guard of t. If we call (f 3) in the defconst,
; there is a guard violation. In :logic mode that isn't a problem, because we
; are running *1* code. But in :program mode we could get a hard Lisp error.
; In fact, we won't in the case of defconst, because defconst forms are
; evaluated in safe mode. For a potentially related issue, see the comments in
; :DOC note-2-9 for an example of how we can get unsoundness, not merely a hard
; error, for the use of ill-guarded functions in defconst forms.
'redundant)))
((and (null (cadr val)) ; optimization
(fetch-dcl-field :non-executable
(butlast (cddr def) 1)))
(cond
((let* ((event-tuple (cddr (car wrld1)))
(event (access-event-tuple-form event-tuple)))
(non-identical-defp
def
(case (car event)
(mutual-recursion
(assoc-eq name (strip-cdrs (cdr event))))
(defuns
(assoc-eq name (cdr event)))
(otherwise
(cdr event)))
chk-measure-p
wrld)))
((and (eq (symbol-class name wrld) :program)
(eq defun-mode :logic))
'reclassifying)
(t
; We allow "redefinition" from :logic to :program mode by treating the latter
; as redundant. See the comment above on this topic.
'redundant)))
(t nil))))
(t nil)))
(defun redundant-or-reclassifying-defunsp1 (defun-mode symbol-class
ld-skip-proofsp chk-measure-p
def-lst wrld ans)
(cond ((null def-lst) ans)
(t (let ((x (redundant-or-reclassifying-defunp
defun-mode symbol-class ld-skip-proofsp chk-measure-p
(car def-lst) wrld)))
(cond
((consp x) x) ; a message
((eq ans x)
(redundant-or-reclassifying-defunsp1
defun-mode symbol-class ld-skip-proofsp chk-measure-p
(cdr def-lst) wrld ans))
(t nil))))))
(defun recover-defs-lst (fn wrld)
; Fn is a :program function symbol in wrld. Thus, it was introduced by defun.
; (Constrained and defchoose functions are :logic.) We return the defs-lst
; that introduced fn. We recover this from the cltl-command for fn.
; A special case is when fn is non-executable. We started allowing
; non-executable :program mode functions after Version_4.1, to provide an easy
; way to use defattach, especially during the boot-strap. We prohibit
; reclassifying such a function symbol into :logic mode, for at least the
; following reason: we store the true stobjs-out for non-executable :program
; mode functions, to match attachments that may be made; but we always store a
; stobjs-out of (nil) in the :logic mode case. We could perhaps allow
; reclassifying into :logic mode in cases where the stobjs-out is (nil) in the
; :program mode function, by recovering defuns from the event. But it seems
; most coherent simply to disallow the upgrade. We store a different value,
; :program, for the 'non-executablep property for :program mode functions than
; for :logic mode functions, where we store t.
(let ((err-str "For technical reasons, we do not attempt to recover the ~
definition of a ~s0 function such as ~x1. It is surprising ~
actually that you are seeing this message; please contact ~
the ACL2 implementors unless you have called ~x2 yourself.")
(ctx 'recover-defs-lst))
(cond
((getpropc fn 'non-executablep nil wrld)
; We shouldn't be seeing this message, as something between verify-termination
; and this lower-level function should be handling the non-executable case
; (which is disallowed for the reasons explained above, related to
; stobjs-out).
(er hard ctx
err-str
"non-executable" fn 'recover-defs-lst))
(t
(let ((val
(scan-to-cltl-command
(cdr (lookup-world-index 'event
(getpropc fn 'absolute-event-number
'(:error "See ~
RECOVER-DEFS-LST.")
wrld)
wrld)))))
(cond ((and (consp val)
(eq (car val) 'defuns))
; Val is of the form (defuns defun-mode-flg ignorep def1 ... defn). If
; defun-mode-flg is non-nil then the parent event was (defuns def1 ... defn)
; and the defun-mode was defun-mode-flg. If defun-mode-flg is nil, the parent
; was an encapsulate, defchoose, or :non-executable, but none of these cases
; should occur since presumably we are only considering :program mode functions
; that are not non-executable.
(cond ((cadr val) (cdddr val))
(t (er hard ctx
err-str
"non-executable or :LOGIC mode"
fn
'recover-defs-lst))))
(t (er hard ctx
"We failed to find the expected CLTL-COMMAND for the ~
introduction of ~x0."
fn))))))))
(defun get-clique (fn wrld)
; Fn must be a function symbol. We return the list of mutually recursive fns
; in the clique containing fn, according to their original definitions. If fn
; is :program we have to look for the cltl-command and recover the clique from
; the defs-lst. Otherwise, we can use the 'recursivep property.
(cond ((programp fn wrld)
(let ((defs (recover-defs-lst fn wrld)))
(strip-cars defs)))
(t (let ((recp (getpropc fn 'recursivep nil wrld)))
(cond ((null recp) (list fn))
(t recp))))))
(defun redundant-or-reclassifying-defunsp0 (defun-mode symbol-class
ld-skip-proofsp chk-measure-p
def-lst wrld)
; See redundant-or-reclassifying-defunsp. This function has the same behavior
; as that one, except in this one, if parameter chk-measure-p is nil, then
; measure checking is suppressed.
(cond
((null def-lst) 'redundant)
(t (let ((ans (redundant-or-reclassifying-defunp
defun-mode symbol-class ld-skip-proofsp chk-measure-p
(car def-lst) wrld)))
(cond ((consp ans) ans) ; a message
(t (let ((ans (redundant-or-reclassifying-defunsp1
defun-mode symbol-class ld-skip-proofsp
chk-measure-p (cdr def-lst) wrld ans)))
(cond ((eq ans 'redundant)
(cond
((set-equalp-eq (strip-cars def-lst)
(get-clique (caar def-lst) wrld))
ans)
(t (msg "for definitions to be redundant, if one ~
is defined with mutual-recursion then the ~
old and new definition must occur in ~
mutual-recursion events that define the ~
same set of function symbols. See :DOC ~
redundant-events.~|~%"))))
((and (eq ans 'reclassifying)
(not (set-equalp-eq (strip-cars def-lst)
(get-clique (caar def-lst)
wrld))))
(msg "for reclassifying :program mode definitions ~
to :logic mode, an entire mutual-recursion ~
clique must be reclassified. In this case, ~
the mutual-recursion that defined ~x0 also ~
defined the following, not included in the ~
present event: ~&1.~|~%"
(caar def-lst)
(set-difference-eq (get-clique (caar def-lst)
wrld)
(strip-cars def-lst))))
(t ans)))))))))
(defun strip-last-elements (lst)
(declare (xargs :guard (true-list-listp lst)))
(cond ((endp lst) nil)
(t (cons (car (last (car lst)))
(strip-last-elements (cdr lst))))))
(defun redundant-or-reclassifying-defunsp (defun-mode symbol-class
ld-skip-proofsp def-lst ctx wrld
ld-redefinition-action fives
non-executablep stobjs-in-lst
default-state-vars)
; We return 'redundant if the functions in def-lst are already identically
; defined with :mode defun-mode and class symbol-class. We return
; 'verify-guards if they are all identically defined with :mode :logic and class
; :ideal, but this definition indicates promotion to :common-lisp-compliant.
; Finally, we return 'reclassifying if they are all identically defined in
; :mode :program and defun-mode is :logic. We return nil otherwise.
; We start to answer this question by independently considering each def in
; def-lst. We then add additional requirements pertaining to mutual-recursion.
; The first is for :logic mode definitions (but see the Historical Plaque
; below): if the old and new definition are in different mutual-recursion nests
; (or if one is in a mutual-recursion with other definitions and the other is
; not), then the new definition is not redundant. To see why we make this
; additional restriction, consider the following example.
; (encapsulate
; ()
; (local
; (mutual-recursion
; (defun f (x y)
; (if (and (consp x) (consp y))
; (f (cons 3 x) (cdr y))
; (list x y)))
; (defun g (x y)
; (if (consp y)
; (f x (cdr y))
; (list x y)))))
;
; (defun f (x y)
; ;;; possible IMPLICIT (bad) measure of (acl2-count x)
; (if (and (consp x) (consp y))
; (f (cons 3 x) (cdr y))
; (list x y))))
; As the comment indicates, if ACL2 were to use the entire mutual-recursion to
; guess measures, then it might well guess a different measure (based on y) for
; the first definition of f than for the second (based on x), leaving us with
; an unsound induction scheme for f (based incorrectly on x). Although ACL2
; does not guess measures that way as of this writing (shortly after the
; Version_3.4 release), still one can imagine future heuristic changes of this
; sort. A more "practical" reason for this restriction is that it seems to
; make the underlying theory significantly easier to work out.
; A second requirement is that we do not reclassify from :program mode to
; :logic mode for a proper subset of a mutual-recursion nest. This restriction
; may be overly conservative, but then again, we expect it to be rare that it
; would affect anyone. While we do not have a definitive reason for this
; restriction, consider for example induction schemes, which are stored for
; single recursion but not mutual-recursion. Although this issue may be fully
; handled by the restriction on redundancy described above, we see this as just
; one possible pitfall, so we prefer to maintain the invariant that all
; functions in a mutual-recursion nest have the same defun-mode.
; Note: Our redundancy check for definitions is based on the untranslated
; terms. This is different from, say, theorems, where we compare translated
; terms. The reason is that we do not store the translated versions of
; :program definitions and don't want to go to the cost of translating
; what we did store. We could, I suppose. We handle theorems the way we do
; because we store the translated theorem on the property list, so it is easy.
; Our main concern vis-a-vis redundancy is arranging for identical definitions
; not to blow us up when we are loading books that have copied definitions and
; I don't think translation will make an important difference to the utility of
; the feature.
; Note: There is a possible bug lurking here. If the host Common Lisp expands
; macros before storing the symbol-function, then we could recognize as
; "redundant" an identical defun that, if actually passed to the underlying
; Common Lisp, would result in the storage of a different symbol-function
; because of the earlier redefinition of some macro used in the "redundant"
; definition. This is not a soundness problem, since redefinition is involved.
; But it sure might annoy somebody who didn't notice that his redefinition
; wasn't processed.
; Historical Plaque: The following comment was in place before we restricted
; redundancy to insist on identical mutual-recursion nests.
; We answer this question by answering it independently for each def in
; def-lst. Thus, every def must be 'redundant or 'reclassifying as
; appropriate. This seems really weak because we do not insist that only one
; cltl-command tuple is involved. But (defuns def1 ... defn) just adds one
; axiom for each defi and the axiom is entirely determined by the defi. Thus,
; if we have executed a defuns that added the axiom for defi then it is the
; same axiom as would be added if we executed a different defuns that
; contained defi. Furthermore, a cltl-command of the form (defuns :defun-mode
; ignorep def1 ... defn) means (defuns def1 ... defn) was executed in this
; world with the indicated defun-mode.
(let ((ans
(redundant-or-reclassifying-defunsp0 defun-mode symbol-class
ld-skip-proofsp t def-lst wrld)))
(cond ((and ld-redefinition-action
(member-eq ans '(redundant reclassifying verify-guards)))
; We do some extra checking, converting ans to nil, in order to consider there
; to be true redefinition (by returning nil) in cases where that seems possible
; -- in particular, because translated bodies have changed due to prior
; redefinition of macros or defconsts called in a new body. Our handling of
; this case isn't perfect, for example because it may reject reclassification
; when the order changes. But at least it forces some definitions to be
; considered as doing redefinition. Notice that this extra effort is only
; performed when redefinition is active, so as not to slow down the system in
; the normal case. If there has been no redefinition in the session, then we
; expect this extra checking to be unnecessary.
(let ((names (strip-cars fives))
(arglists (strip-cadrs fives))
(bodies (get-bodies fives)))
(mv-let (erp lst bindings)
(translate-bodies1 (eq non-executablep t) ; not :program
names bodies
(pairlis$ names names)
arglists
stobjs-in-lst
ctx wrld default-state-vars)
(declare (ignore bindings))
(cond (erp
; This error could be due to an untouchable variable or function in one of the
; bodies. In that case, we return the result returned by
; redundant-or-reclassifying-defunsp0 above, without possibly converting it to
; nil as may be done below. That's OK; then we will not make an additional
; check that we are truly doing redefinition. As discussed above, such
; perfection here is not required; in particular, we then simply consider the
; definition redundant here just as if redefinition were off. However, it
; should be perfectly OK to consider the definition not to be redundant in that
; case.
ans)
((eq (symbol-class (car names) wrld)
:program)
(let ((old-defs (recover-defs-lst (car names)
wrld)))
(and (equal names (strip-cars old-defs))
(mv-let
(erp old-lst bindings)
(translate-bodies1
; The old non-executablep is nil; see recover-defs-lst.
nil
names
(strip-last-elements old-defs)
(pairlis$ names names)
arglists
stobjs-in-lst
ctx wrld default-state-vars)
(declare (ignore bindings))
(cond ((and (null erp)
(equal lst old-lst))
ans)
(t
; If erp is true then we consider this to be true redefinition. That is the
; opposite decision from what is made in the case above when translate-bodies1
; returns an error. Which is the right decision: consider redefinition (vs.,
; say, redundancy) or not when there is an error in translate-bodies1? The
; answer is that either is acceptable, as discussed above. But as of this
; writing (August 2017) the code has probably been this way for a long time, we
; leave it alone, at least for now.
nil))))))
; Otherwise we expect to be dealing with :logic mode functions.
((equal lst
(get-unnormalized-bodies names wrld))
ans)
(t nil)))))
(t ans))))
(defun collect-when-cadr-eq (sym lst)
(cond ((null lst) nil)
((eq sym (cadr (car lst)))
(cons (car lst) (collect-when-cadr-eq sym (cdr lst))))
(t (collect-when-cadr-eq sym (cdr lst)))))
(defun all-programp (names wrld)
; Names is a list of function symbols. Return t iff every element of
; names is :program.
(cond ((null names) t)
(t (and (programp (car names) wrld)
(all-programp (cdr names) wrld)))))
; Essay on the Identification of Irrelevant Formals
; A formal is irrelevant if its value does not affect the value of the
; function. Of course, ignored formals have this property, but we here address
; ourselves to the much more subtle problem of formals that are used only in
; irrelevant ways. For example, y in
; (defun foo (x y) (if (zerop x) 0 (foo (1- x) (cons x y))))
; is irrelevant. Clearly, any formal mentioned outside of a recursive call is
; relevant -- provided that no previously introduced function has irrelevant
; arguments and no definition tests constants as in (if t x y). But a formal
; that is never used outside a recursive call may still be relevant, as
; illustrated by y in:
; (defun foo (x y) (if (< x 2) x (foo y 0)))
; Observe that (foo 3 1) = 1 and (foo 3 0) = 0; thus, y is relevant. (This
; function can be admitted with the measure (cond ((< x 2) 0) ((< y 2) 1) (t
; 2)).)
; Thus, we have to do a transitive closure computation based on which formals
; appear in which actuals of recursive calls. In the first pass we see that x,
; above, is relevant because it is used outside the recursion. In the next
; pass we see that y is relevant because it is passed into the x argument
; position of a recursive call.
; The whole thing is made somewhat more hairy by mutual recursion, though no
; new intellectual problems are raised. However, to cope with mutual recursion
; we stop talking about "formals" and start talking about "posns." A posn here
; is a natural number n that represents the nth formal for a function in the
; mutually recursive clique. We say a "posn is used" if the corresponding
; formal is used.
; A "recursive call" here means a call of any function in the clique. We
; generally use the variable clique-alist to mean an alist whose elements are
; each of the form (fn . posns).
; A second problem is raised by the presence of lambda expressions. We discuss
; them more below.
; Our algorithm iteratively computes the relevant posns of a clique by
; successively enlarging an initial guess. The initial guess consists of all
; the posns used outside of a recursive call, including the guard or measure or
; the lists of ignored or ignorable formals. Clearly, every posn so collected
; is relevant. We then iterate, sweeping into the set every posn used either
; outside recursion or in an actual used in a relevant posn. When this
; computation ceases to add any new posns we consider the uncollected posns to
; be irrelevant.
; For example, in (defun foo (x y) (if (zerop x) 0 (foo (1- x) (cons x y)))) we
; initially guess that x is relevant and y is not. The next iteration adds
; nothing, because y is not used in the x posn, so we are done.
; On the other hand, in (defun foo (x y) (if (< x 2) x (foo y 0))) we might
; once again guess that y is irrelevant. However, the second pass would note
; the occurrence of y in a relevant posn and would sweep it into the set. We
; conclude that there are no irrelevant posns in this definition.
; So far we have not discussed lambda expressions; they are unusual in this
; setting because they may hide recursive calls that we should analyze. We do
; not want to expand the lambdas away, for fear of combinatoric explosions.
; Instead, we expand the clique-alist, by adding, for each lambda-application a
; new entry that pairs that lambda expression with the appropriate terms.
; (That is, the "fn" of the new clique member is the lambda expression itself.)
; Thus, we actually use assoc-equal instead of assoc-eq when looking in
; clique-alist.
(defun formal-position (var formals i)
(cond ((null formals) i)
((eq var (car formals)) i)
(t (formal-position var (cdr formals) (1+ i)))))
(defun make-posns (formals vars)
(cond ((null vars) nil)
(t (cons (formal-position (car vars) formals 0)
(make-posns formals (cdr vars))))))
(mutual-recursion
(defun relevant-posns-term (fn formals term fns clique-alist posns)
; Term is a term occurring in the body of fn which has formals formals. We
; collect a posn into posns if it is used outside a recursive call (or in an
; already known relevant actual to a recursive call). See the Essay on the
; Identification of Irrelevant Formals.
(cond
((variablep term)
(add-to-set (formal-position term formals 0) posns))
((fquotep term) posns)
((equal (ffn-symb term) fn)
(relevant-posns-call fn formals (fargs term) 0 fns clique-alist :same
posns))
((member-equal (ffn-symb term) fns)
(relevant-posns-call fn formals (fargs term) 0 fns clique-alist
(cdr (assoc-equal (ffn-symb term) clique-alist))
posns))
(t
(relevant-posns-term-lst fn formals (fargs term) fns clique-alist posns))))
(defun relevant-posns-term-lst (fn formals lst fns clique-alist posns)
(cond ((null lst) posns)
(t
(relevant-posns-term-lst
fn formals (cdr lst) fns clique-alist
(relevant-posns-term fn formals (car lst) fns clique-alist posns)))))
(defun relevant-posns-call (fn formals actuals i fns clique-alist
called-fn-posns posns)
; See the Essay on the Identification of Irrelevant Formals.
; This function extends the set, posns, of posns for fn that are known to be
; relevant. It does so by analyzing the given (tail of the) actuals for a call
; of some function in the clique, which we denote as called-fn, where that call
; occurs in the body of fn (which has the given formals). Called-fn-posns is
; the set of posns for called-fn that are known to be relevant, except for the
; case that called-fn is fn, in which case called-fn-posns is :same. The
; formal i, which is initially 0, is the position in called-fn's argument
; list of the first element of actuals. We extend posns, the posns of fn known
; to be relevant, by seeing which posns are used in the actuals in the relevant
; posns of called-fn (i.e., called-fn-posns).
(cond
((null actuals) posns)
(t (relevant-posns-call
fn formals (cdr actuals) (1+ i) fns clique-alist
called-fn-posns
(if (member i
(if (eq called-fn-posns :same)
posns ; might be extended through recursive calls
called-fn-posns))
(relevant-posns-term fn formals (car actuals) fns clique-alist
posns)
posns)))))
)
(defun relevant-posns-clique1 (fns arglists bodies all-fns ans)
(cond
((null fns) ans)
(t (relevant-posns-clique1
(cdr fns)
(cdr arglists) ; nil, once we cdr down to the lambdas
(cdr bodies) ; nil, once we cdr down to the lambdas
all-fns
(let* ((posns (cdr (assoc-equal (car fns) ans)))
(new-posns
(cond ((flambdap (car fns))
(relevant-posns-term (car fns)
(lambda-formals (car fns))
(lambda-body (car fns))
all-fns
ans
posns))
(t
(relevant-posns-term (car fns)
(car arglists)
(car bodies)
all-fns
ans
posns)))))
(cond ((equal posns new-posns) ; optimization
ans)
(t (put-assoc-equal (car fns) new-posns ans))))))))
(defun relevant-posns-clique-recur (fns arglists bodies clique-alist)
(let ((clique-alist1 (relevant-posns-clique1 fns arglists bodies fns
clique-alist)))
(cond ((equal clique-alist1 clique-alist) clique-alist)
(t (relevant-posns-clique-recur fns arglists bodies
clique-alist1)))))
(defun relevant-posns-clique-init (fns arglists guards split-types-terms
measures ignores ignorables ans)
; We associate each function in fns, reversing the order in fns, with
; obviously-relevant formal positions.
(cond
((null fns) ans)
(t (relevant-posns-clique-init
(cdr fns)
(cdr arglists)
(cdr guards)
(cdr split-types-terms)
(cdr measures)
(cdr ignores)
(cdr ignorables)
(acons (car fns)
(make-posns
(car arglists)
(all-vars1 (car guards)
(all-vars1 (car split-types-terms)
(all-vars1 (car measures)
; Ignored formals are considered not to be irrelevant. Should we do similarly
; for ignorable formals?
; - If yes (ignorables are not irrelevant), then we may miss some irrelevant
; formals. Of course, it is always OK to miss some irrelevant formals, but
; we would prefer not to miss them needlessly.
; - If no (ignorables are irrelevant), then we may report an ignorable variable
; as irrelevant, which might annoy the user even though it really is
; irrelevant, if "ignorable" not only means "could be ignored" but also means
; "could be irrelevant".
; We choose "yes". If the user has gone through the trouble to label a
; variable as irrelevant, then the chance that irrelevance is due to a typo are
; dwarfed by the chance that irrelevance is due to being an ignorable var.
(union-eq (car ignorables)
(car ignores))))))
ans)))))
; We now develop the code to generate the clique-alist for lambda expressions.
(mutual-recursion
(defun relevant-posns-lambdas (term ans)
(cond ((or (variablep term)
(fquotep term))
ans)
((flambdap (ffn-symb term))
(relevant-posns-lambdas
(lambda-body (ffn-symb term))
(acons (ffn-symb term)
nil
(relevant-posns-lambdas-lst (fargs term) ans))))
(t (relevant-posns-lambdas-lst (fargs term) ans))))
(defun relevant-posns-lambdas-lst (termlist ans)
(cond ((endp termlist) ans)
(t (relevant-posns-lambdas-lst
(cdr termlist)
(relevant-posns-lambdas (car termlist) ans)))))
)
(defun relevant-posns-merge (alist acc)
(cond ((endp alist) acc)
((endp (cdr alist)) (cons (car alist) acc))
((equal (car (car alist))
(car (cadr alist)))
(relevant-posns-merge (acons (caar alist)
(union$ (cdr (car alist))
(cdr (cadr alist)))
(cddr alist))
acc))
(t (relevant-posns-merge (cdr alist) (cons (car alist) acc)))))
(defun relevant-posns-lambdas-top (bodies)
(let ((alist (merge-sort-lexorder (relevant-posns-lambdas-lst bodies nil))))
(relevant-posns-merge alist nil)))
(defun relevant-posns-clique (fns arglists guards split-types-terms measures
ignores ignorables bodies)
; We compute the relevant posns in an expanded clique alist (one in which the
; lambda expressions have been elevated to clique membership). The list of
; relevant posns includes the relevant lambda posns. We do it by iteratively
; enlarging an initial clique-alist until it is closed.
(let* ((clique-alist1 (relevant-posns-clique-init fns arglists guards
split-types-terms measures
ignores ignorables nil))
(clique-alist2 (relevant-posns-lambdas-top bodies)))
(relevant-posns-clique-recur (append fns (strip-cars clique-alist2))
arglists
bodies
(revappend clique-alist1 clique-alist2))))
(defun irrelevant-non-lambda-slots-clique2 (fn formals i posns acc)
(cond ((endp formals) acc)
(t (irrelevant-non-lambda-slots-clique2
fn (cdr formals) (1+ i) posns
(cond ((member i posns) acc)
(t (cons (list* fn i (car formals))
acc)))))))
(defun irrelevant-non-lambda-slots-clique1 (fns arglists clique-alist acc)
(cond ((endp fns)
(assert$ (or (null clique-alist)
(flambdap (caar clique-alist)))
acc))
(t (assert$ (eq (car fns) (caar clique-alist))
(irrelevant-non-lambda-slots-clique1
(cdr fns) (cdr arglists) (cdr clique-alist)
(irrelevant-non-lambda-slots-clique2
(car fns) (car arglists) 0 (cdar clique-alist)
acc))))))
(defun irrelevant-non-lambda-slots-clique (fns arglists guards
split-types-terms measures
ignores ignorables bodies)
; Let clique-alist be an expanded clique alist (one in which lambda expressions
; have been elevated to clique membership). Return all the irrelevant slots
; for the non-lambda members of the clique.
; A "slot" is a triple of the form (fn n . var), where fn is a function symbol,
; n is some nonnegative integer less than the arity of fn, and var is the nth
; formal of fn. If (fn n . var) is in the list returned by this function, then
; the nth formal of fn, namely var, is irrelevant to the value computed by fn.
(let ((clique-alist (relevant-posns-clique fns arglists guards
split-types-terms measures
ignores ignorables bodies)))
(irrelevant-non-lambda-slots-clique1 fns arglists clique-alist nil)))
(defun tilde-*-irrelevant-formals-msg1 (slots)
(cond ((null slots) nil)
(t (cons (cons "~n0 formal of ~x1, ~x2,"
(list (cons #\0 (list (1+ (cadar slots))))
(cons #\1 (caar slots))
(cons #\2 (cddar slots))))
(tilde-*-irrelevant-formals-msg1 (cdr slots))))))
(defun tilde-*-irrelevant-formals-msg (slots)
(list "" "~@*" "~@* and the " "~@* the " (tilde-*-irrelevant-formals-msg1 slots)))
(defun missing-irrelevant-slots1 (irrelevant-slots irrelevants-alist acc)
; Recall that a slot has the form (fn n . var); see
; irrelevant-non-lambda-slots-clique.
(cond ((endp irrelevant-slots) acc)
(t (missing-irrelevant-slots1
(cdr irrelevant-slots)
irrelevants-alist
(if (member-eq (cddr (car irrelevant-slots)) ; var
(cdr (assoc-eq (car (car irrelevant-slots)) ; fn
irrelevants-alist)))
acc
(cons (car irrelevant-slots) acc))))))
(defun missing-irrelevant-slots (irrelevant-slots irrelevants-alist)
(cond ((null irrelevant-slots) ; common case
nil)
((null irrelevants-alist) ; common case
irrelevant-slots)
(t (missing-irrelevant-slots1 irrelevant-slots irrelevants-alist
nil))))
(defun find-slot (fn var irrelevant-slots)
(cond ((endp irrelevant-slots) nil)
((let ((slot (car irrelevant-slots))) ; (fn n . var)
(or (and (eq fn (car slot))
(eq var (cddr slot))))))
(t (find-slot fn var (cdr irrelevant-slots)))))
(defun bogus-irrelevants-alist2 (irrelevant-slots fn vars)
(cond ((endp vars) nil)
((find-slot fn (car vars) irrelevant-slots)
(bogus-irrelevants-alist2 irrelevant-slots fn (cdr vars)))
(t
(cons (car vars)
(bogus-irrelevants-alist2 irrelevant-slots fn (cdr vars))))))
(defun bogus-irrelevants-alist1 (irrelevant-slots irrelevants-alist acc)
; Recall that a slot has the form (fn n . var); see
; irrelevant-non-lambda-slots-clique.
(cond ((endp irrelevants-alist) acc)
(t (bogus-irrelevants-alist1
irrelevant-slots
(cdr irrelevants-alist)
(let ((bogus-vars
(bogus-irrelevants-alist2 irrelevant-slots
(caar irrelevants-alist)
(cdar irrelevants-alist))))
(if bogus-vars
(acons (caar irrelevants-alist)
bogus-vars
acc)
acc))))))
(defun bogus-irrelevants-alist (irrelevant-slots irrelevants-alist)
(cond ((null irrelevants-alist) ; optimization for common case
irrelevant-slots)
(t (bogus-irrelevants-alist1 irrelevant-slots irrelevants-alist nil))))
(defun tilde-*-bogus-irrelevants-alist-msg1 (alist)
(cond ((endp alist) nil)
(t (cons (cons "formal~#0~[~/s~] ~&0 of ~x1"
(list (cons #\0 (cdar alist))
(cons #\1 (caar alist))))
(tilde-*-bogus-irrelevants-alist-msg1 (cdr alist))))))
(defun tilde-*-bogus-irrelevants-alist-msg (alist)
(list "" "~@*" "~@*; and the " "~@*; the "
(tilde-*-bogus-irrelevants-alist-msg1 alist)))
(defun chk-irrelevant-formals-msg (fns arglists guards split-types-terms
measures ignores ignorables
irrelevants-alist bodies rawp)
; This code was originally part of chk-irrelevant-formals. It has been split
; out in support of community book
; books/kestrel/utilities/irrelevant-formals.lisp.
(let ((irrelevant-slots
(irrelevant-non-lambda-slots-clique
fns arglists guards split-types-terms measures ignores ignorables
bodies)))
(cond
((and (null irrelevant-slots)
(null irrelevants-alist)) ; optimize for common case
nil)
(t
(let ((bogus-irrelevants-alist ; declared irrelevant but not
(bogus-irrelevants-alist irrelevant-slots irrelevants-alist))
(missing-irrelevant-slots ; irrelevant but not declared
(missing-irrelevant-slots irrelevant-slots
irrelevants-alist)))
(cond
((and (null bogus-irrelevants-alist)
(null missing-irrelevant-slots))
nil)
(rawp (list bogus-irrelevants-alist
missing-irrelevant-slots))
(t (msg
"~@0~@1See :DOC irrelevant-formals."
(if missing-irrelevant-slots
(msg "The ~*0 ~#1~[is~/are~] irrelevant but not declared ~
to be irrelevant. "
(tilde-*-irrelevant-formals-msg
missing-irrelevant-slots)
(if (cdr missing-irrelevant-slots) 1 0))
"")
(if bogus-irrelevants-alist
(msg "The ~*0 ~#1~[is~/are~] falsely declared irrelevant. "
(tilde-*-bogus-irrelevants-alist-msg
bogus-irrelevants-alist)
(if (or (cdr bogus-irrelevants-alist)
(cddr (car bogus-irrelevants-alist)))
1
0))
"")))))))))
(defun chk-irrelevant-formals (fns arglists guards split-types-terms measures
ignores ignorables irrelevants-alist bodies
ctx state)
(let ((irrelevant-formals-ok
(cdr (assoc-eq :irrelevant-formals-ok
(table-alist 'acl2-defaults-table (w state))))))
(cond
((or (eq irrelevant-formals-ok t)
(and (eq irrelevant-formals-ok :warn)
(warning-disabled-p "Irrelevant-formals")))
(value nil))
(t (let ((msg (chk-irrelevant-formals-msg fns arglists guards
split-types-terms
measures ignores ignorables
irrelevants-alist bodies nil)))
(cond ((null msg) (value nil))
((eq irrelevant-formals-ok :warn)
(pprogn (warning$ ctx ("Irrelevant-formals") "~@0" msg)
(value nil)))
(t (er soft ctx "~@0" msg))))))))
; Essay on the Use of :PROGRAM Mode Functions by :LOGIC Mode Functions
; This essay does not reveal anything that a user familiar with apply$ doesn't
; already know! It's not really about implementation details though it served
; as a sort of design document for allowing :program mode functions to be
; badged.
; Before the introduction of apply$, there was a simple rule: :logic mode
; functions are defined entirely in terms of :logic mode subfunctions. When
; apply$ was first introduced (in Version 8.0), every badged function had also
; a warrant (and all warranted functions are necessarily in :logic mode), so
; every badged function was in :logic mode. But it was possible to define an
; unbadged :logic mode function to appear to call a :program mode function (or
; even an undefined function!). The following commands were carried out in
; V8.3, which has the same restrictions as V8.0 but was more powerful mainly
; because all warrants are true in its evaluation theory (enabling top-level
; evaluation of apply$ forms), loop$ recursion was supported, and lambda object
; rewriting was done.
; Consider this sketch of a V8.3 session:
; ; Successful defun of :logic mode fn ``calling'' an undefined function:
; (defun foo (x y)
; (declare (xargs :mode :logic))
; (apply$ 'undefined-fn (list x y)))
;
; ; Two tests and results:
; (foo 1 2)
;
; ACL2 Error in TOP-LEVEL: The value of APPLY$-USERFN is not specified
; on UNDEFINED-FN because UNDEFINED-FN is not a known function symbol.
;
; (thm (consp (foo 1 2)))
; *** Key checkpoint at the top level: ***
; Goal'
; (CONSP (APPLY$ 'UNDEFINED-FN '(1 2)))
;
; ; The checkpoint could be further reduced, with hints to (CONSP (APPLY$-USERFN
; ; 'UNDEFINED-FN '(1 2))).
;
; ; Define the formerly undefined function:
; (defun undefined-fn (x y)(declare (xargs :mode :program)) (list 'hello x y))
;
; ; Repeating the two tests above produces virtually the same results, except the
; ; error message now complains about undefined-fn being in :program mode.
;
; ; So convert undefined-fn to :logic mode:
; (verify-termination undefined-fn)
;
; ; Repeating the two tests above produces virtually the same results, except the
; ; error message now complains about undefined-fn being unwarranted (which also
; ; means unbadged here).
;
; ; So badge and warrant it:
; (defwarrant undefined-fn)
;
; ; Repeating the two tests:
;
; (foo 1 2)
; ==> (HELLO 1 2)
;
; (thm (consp (foo 1 2)))
; q.e.d. (given one forced hypothesis)
; [1]Goal
; (APPLY$-WARRANT-UNDEFINED-FN)
;
; ; So supply the warrant:
; (thm (implies (warrant undefined-fn) (consp (foo 1 2))))
; Q.E.D.
; What we've seen is that ACL2 has always supported the idea that a :logic mode
; function might use apply$ to call an undefined quoted ``function symbol'' and
; the :logic mode function behaves reasonably (soundly) as the formerly
; undefined function becomes more acceptable to apply$.
; Furthermore, it not possible to determine all the functions a :logic
; mode function might call via apply$. Consider
; (defun foo (fn x y) (apply$ fn (list x y))).
; If you replace every call of (foo 1 2) in the session above with (foo
; 'undefined-fn 1 2) you get the same behavior as before. So it's simply
; hopeless to imagine enforcing a rule against :logic mode functions
; ``calling'' :program mode functions via apply$. The best we can do is
; guarantee that evaluation -- whether in the evaluation theory or the logic --
; handles non-:logic arguments correctly. And that's pretty straightforward
; simply because the logic doesn't know anything about userfns except what
; warrants tell it, and the doppelgangers for badge-userfn and apply$-userfn
; handle things correctly.
; Here is an example of an unbadgeable function that computes a new lambda
; expression on every recursion, and we reason about it perfectly well:
; (defun enumerate (fn lst)
; (if (endp lst)
; nil
; (cons (apply$ fn (list (car lst)))
; (enumerate `(lambda (e)
; (cons (binary-+ '1 ,(cadr (caddr fn)))
; ,(caddr (caddr fn))))
; (cdr lst)))))
; (thm (equal (enumerate '(lambda (e) (cons '0 e)) '(a b c d))
; '((0 . a) (1 . b) (2 . c) (3 . d))))
; In Version 8.4 we allowed :program mode functions to have badges so that
; loop$s, for example, could be run using :program mode functions in their
; bodies. Of course, :program mode functions cannot have warrants because
; warrants make a logical statement about the function. The session sketched
; above should still be valid, except that in 8.4 we'd expect (foo 1 2) to
; evaluate in the evaluation theory even if undefined-fn is just a :program
; mode function -- as long as it has a badge.
; The change from v8.3 to v8.4 really only raises source-code programming
; problems: prior to v8.4 we knew that if a userfn had a badge the function was
; in :logic mode and had a warrant. But in v8.4 we must explicitly test each
; of those conditions before carrying out certain acts. The challenge was
; finding all the places we tested for badges or tameness and implicitly relied
; on the existence of a warrant.
; It is also worth noting that rewrite-lambda-object invites trouble because it
; offers an opportunity to get into evaluation without explicitly going through
; apply$ or the doppelgangers. If a lambda object can have a body that
; mentions undefined or :program mode functions and the rewriter just barged
; into the body treating it like a logical term, we could easily get hard
; errors (or worse) because we'd be violating a basic invariant of our code:
; the theorem prover deals with :logic terms. The lambda rewriter checked that
; the body was well-formed, which included tame, which pre-v8.4 guaranteed
; warrants, and so we knew that (ev$ '(undefined-fn '1 '2) a) = (undefined-fn
; '1 '2). But now tameness doesn't guarantee that warrants exist.
; Some good guidelines to keep in mind:
; (1) Apply$ can encounter anything!
; (2) Badges are necessary to know that :FNs are respected.
; (3) Badges are necessary to run safely.
; (4) Not every function using apply$ will be badged.
; (5) Warrants are necessary to prove anything.
; (6) Warrants imply badges and :logic mode.
(defun chk-logic-subfunctions (names0 names terms wrld str ctx state)
; Assume we are defining names in terms of terms (1:1 correspondence). Assume
; also that the definitions are to be :logic. Then we insist that every
; function used in terms be :logic. Str is a string used in our error message
; and is either "guard", "split-types expression", or "body". But we allow
; quoted function symbols in :FN slots to be in :program mode. See the Essay
; on the Use of :PROGRAM Mode Functions by :LOGIC Mode Functions.
; WARNING: This function guarantees that a call of a :logic mode function
; cannot lead to a call of a :program mode function. This guarantee justifies
; the restriction, implemented in oneify-cltl-code, that only :program mode
; functions lay down *1* code that is sensitive to invariant-risk. It seems
; conceivable that without the guarantee, a :logic mode function could lead to
; a call of a :program mode function that violates stobj invariants or writes
; past the end of an array. So be careful when considering a relaxation of
; this guarantee! For more reasons why :logic mode functions must not call
; :program mode functions, see comments in raw-ev-fncall and
; raw-ev-fncall-simple and see :DOC program-only.
(cond
((null names) (value nil))
(t
(let
((bad (collect-programs
(set-difference-eq (all-fnnames (car terms))
names0)
wrld)))
(cond
(bad
; Before eliminating the error below, think carefully! In particular, consider
; the following problem involving trans-eval. A related concern, which points
; to the comment below, may be found in a comment in the definition of
; magic-ev-fncall.
; Sol Swords wondered whether there might be an issue when function takes and
; returns both a user-defined stobj and state, calling trans-eval to change the
; stobj even though the function doesn't actually change it. Below
; investigating whether Sol's idea can be exploited to destroy, perhaps with
; bad consequences, some sort of invariant related to the user-stobj-alist of
; the state. The answer seems to be no, but only because (as Sol pointed out,
; if memory serves) trans-eval is in :program mode -- and it stays there
; because trans-eval calls ev-for-trans-eval, which calls ev, which belongs to
; the list *initial-program-fns-with-raw-code* (and because :logic mode
; functions can't call :program mode functions). Below is an example that
; illustrates what could go wrong if trans-eval were in :logic mode.
; (defstobj st fld)
;
; (set-state-ok t)
;
; (defun f (st state)
; (declare (xargs :stobjs (st state)
; :mode :program))
; (let ((st (update-fld 2 st)))
; (mv-let (erp val state)
; (trans-eval '(update-fld 3 st) 'f state nil)
; (declare (ignore erp val))
; (mv state (fld st) st))))
;
; ; Logically, f sets (fld st) to 2, so the return value should be (mv _ 2
; ; _). But we get (mv _ 3 _). The only thing that saves us is that
; ; trans-eval is in :program mode, hence f is in :program mode. This gives
; ; us a good reason to be very cautious before allowing :program mode
; ; functions to be called from :logic mode functions. Note that even if we
; ; were to allow the return state to be somehow undefined, still the middle
; ; return value would be a problem logically!
;
; ; Succeeds
; (mv-let (state val st)
; (f st state)
; (assert$ (equal val 3)
; (mv state val st)))
;
; ; Fails
; (mv-let (state val st)
; (f st state)
; (assert$ (equal val 2)
; (mv state val st)))
(er
soft ctx
"The ~@0 for ~x1 calls the :program function~#2~[ ~&2~/s ~&2~]. We ~
require that :logic definitions be defined entirely in terms of ~
:logically defined functions. See :DOC defun-mode."
str (car names)
(reverse bad)))
(t (chk-logic-subfunctions names0 (cdr names)
(cdr terms)
wrld str ctx state)))))))
(defun collect-unbadged (fns wrld)
(cond ((endp fns) nil)
((executable-badge (car fns) wrld)
(collect-unbadged (cdr fns) wrld))
(t (cons (car fns)
(collect-unbadged (cdr fns) wrld)))))
(defun chk-badged-quoted-subfunctions (names0 names terms wrld str ctx state)
; Assume we are defining names in terms of terms (1:1 correspondence). Then we
; insist that every quoted symbol used as a function in a :FN or :EXPR slot be
; badged. Note that we do not require the subfunctions to be in :logic mode
; even if the functions being defined are in :logic mode. The only reason we
; check this condition is to cause an error on (apply$ 'foo ...) where foo is
; unbadged. And the reason we do that is because we cause such an error on
; (lambda$ (x) (foo ...)) and it just seems confusing to allow unbadged symbols
; to (apparently) be applied with apply$ but not evaluated by ev$.
; Warning: Do not call this function during boot-strap!
(cond
((null names) (value nil))
(t
(let
((bad (collect-unbadged
(set-difference-eq (all-fnnames! nil ; term flg
:inside ; collect inside quotes
nil ; don't start til inside
(car terms)
nil ; ilk
wrld nil)
names0)
wrld)))
(cond
(bad
(er
soft ctx
"The ~@0 for ~x1 uses the unbadged symbol~#2~[ ~&2~/s ~&2~] in one ~
or more :FN or :EXPR slots. We require that all such symbols be ~
badged function symbols. See :DOC defun and defbadge."
str (car names)
(reverse bad)))
(t (chk-badged-quoted-subfunctions names0 (cdr names)
(cdr terms)
wrld str ctx state)))))))
;; Historical Comment from Ruben Gamboa:
;; This function strips out the functions which are
;; non-classical in a chk-acceptable-defuns "fives" structure.
#+:non-standard-analysis
(defun get-non-classical-fns-from-list (names wrld fns-sofar)
(cond ((null names) fns-sofar)
(t (let ((fns (if (or (not (symbolp (car names)))
(classicalp (car names) wrld))
fns-sofar
(cons (car names) fns-sofar))))
(get-non-classical-fns-from-list (cdr names) wrld fns)))))
;; Historical Comment from Ruben Gamboa:
;; This function takes in a list of terms and returns any
;; non-classical functions referenced in the terms.
#+:non-standard-analysis
(defmacro get-non-classical-fns (lst wrld)
`(get-non-classical-fns-aux ,lst ,wrld nil))
#+:non-standard-analysis
(defun get-non-classical-fns-aux (lst wrld fns-sofar)
(cond ((null lst) fns-sofar)
(t (get-non-classical-fns-aux
(cdr lst)
wrld
(get-non-classical-fns-from-list
(all-fnnames (car lst)) wrld fns-sofar)))))
;; Historical Comment from Ruben Gamboa:
;; this function checks that the measures used to accept the definition
;; are classical. Note, *no-measure* is a signal that the default measure is
;; being used (see get-measures1) -- and in that case, we know it's classical,
;; since it's just the acl2-count of some tuple consisting of variables in the
;; defun.
#+:non-standard-analysis
(defun strip-missing-measures (lst accum)
(if (consp lst)
(if (equal (car lst) *no-measure*)
(strip-missing-measures (cdr lst) accum)
(strip-missing-measures (cdr lst) (cons (car lst) accum)))
accum))
#+:non-standard-analysis
(defun chk-classical-measures (measures names ctx wrld state)
(let ((non-classical-fns (get-non-classical-fns
(strip-missing-measures measures nil)
wrld)))
(cond ((null non-classical-fns)
(value nil))
(t
(er soft ctx
"It is illegal to use non-classical measures to justify a ~
recursive definition. However, there has been an ~
attempt to recursively define ~*0 using the ~
non-classical function~#1~[~/s~] ~*2 in the measure."
`("<MissingFunction>" "~x*," "~x* and " "~x*, " ,names)
non-classical-fns
`("<MissingFunction>" "~x*," "~x* and " "~x*, "
,non-classical-fns))))))
;; Historical Comment from Ruben Gamboa:
;; This function checks that non-classical functions only appear
;; on non-recursive functions.
#+:non-standard-analysis
(defun chk-no-recursive-non-classical (non-classical-fns names mp rel
measures
bodies ctx
wrld state)
(cond ((and (int= (length names) 1)
(not (ffnnamep-mod-mbe (car names) (car bodies))))
; Then there is definitely no recursion (see analogous computation in
; putprop-recursivep-lst). Note that with :bogus-mutual-recursion-ok, a clique
; of size greater than 1 might not actually have any recursion. But then it
; will be up to the user in this case to eliminate the appearance of possible
; recursion.
(value nil))
((not (null non-classical-fns))
(er soft ctx
"It is illegal to use non-classical functions in a ~
recursive definition. However, there has been an ~
attempt to recursively define ~*0 using the ~
non-classical function~#1~[~/s~] ~*2"
`("<MissingFunction>" "~x*," "~x* and " "~x*, " ,names)
non-classical-fns
`("<MissingFunction>." "~x*." "~x* and " "~x*, "
,non-classical-fns)))
((not (and (classicalp mp wrld)
(classicalp rel wrld)))
(er soft ctx
"It is illegal to use a non-classical function as a ~
well-ordering or well-ordered domain in a recursive ~
definition. However, there has been an ~
attempt to recursively define ~*0 using the ~
well-ordering function ~x1 and domain ~x2."
`("<MissingFunction>" "~x*," "~x* and " "~x*, " ,names)
mp
rel))
(t
(chk-classical-measures measures names ctx wrld state))))
(defun translate-measures (terms logic-modep ctx wrld state)
; WARNING: Keep this in sync with translate-term-lst. Here we allow (:? var1
; ... vark), where the vari are distinct variables.
(cond ((null terms) (value nil))
(t (er-let*
((term
(cond ((and (consp (car terms))
(eq (car (car terms)) :?))
(cond ((arglistp (cdr (car terms)))
(value (car terms)))
(t (er soft ctx
"A measure whose car is :? must be of the ~
form (:? v1 ... vk), where (v1 ... vk) is ~
a list of distinct variables. The measure ~
~x0 is thus illegal."
(car terms)))))
(t
(translate (car terms)
; One might use stobjs-out '(nil) below, if one felt uneasy about measures
; changing state. But we know no logical justification for this feeling, nor
; do we ever expect to execute the measures in Common Lisp. In fact it could
; be useful to be able to pass state into a measure even when its argument
; position isn't "state".
t ; stobjs-out
logic-modep t ctx wrld state))))
(rst (translate-measures (cdr terms) logic-modep ctx wrld state)))
(value (cons term rst))))))
(defun redundant-predefined-error-msg (name wrld)
(let ((pkg-name (and (symbolp name) ; probably always true
(symbol-package-name name))))
(msg "ACL2 is processing a redundant definition of the name ~s0 (package ~
~s1), which is ~#2~[already defined using special raw Lisp ~
code~/predefined in the ~x3 package~]. We disallow non-LOCAL ~
redundant definitions in such cases; for explanation, see :DOC ~
redundant-events. Consider wrapping this definition inside a call ~
of LOCAL.~@4"
(symbol-name name)
(symbol-package-name name)
(if (equal pkg-name *main-lisp-package-name*)
1
0)
*main-lisp-package-name*
(if (not (symbolp name))
""
(let* ((wrld (decode-logical-name name wrld))
(path (global-val 'include-book-path wrld))
(book-name (car path)))
(cond
((null book-name) "")
(t (msg " Alteratively, include the book that introduces the ~
proposed redundant definition:~|~x0."
(cond
((sysfile-p book-name)
`(include-book
,(remove-lisp-suffix (sysfile-filename book-name)
t)
:dir ,(sysfile-key book-name)))
(t `(include-book ,(remove-lisp-suffix book-name
t))))))))))))
(defun chk-acceptable-defuns-redundancy (names defun-mode ctx wrld state)
; The following comment is referenced in :doc redundant-events and in a comment
; in defmacro-fn. If it is removed or altered, consider modifying that
; documentation and comment (respectively).
; The definitions of names have tentatively been determined to be redundant.
; We cause an error if this is not allowed, else return (value 'redundant).
; Here we cause an error for non-local redundant built-in definitions. The
; reason is that some built-ins are defined using #-acl2-loop-only code. So
; consider what happens when such a built-in function has a definition
; occurring in the compiled file for a book. At include-book time, this new
; definition will be loaded from that compiled file, presumably without any
; #-acl2-loop-only.
; The following book certified in ACL2 Version_3.3 built on SBCL, where mv was
; values and also we load compiled files. In this case the problem was that
; while ACL2 defined prog2$ as a macro in #-acl2-loop-only, for proper
; multiple-value handling, nevertheless that definition was overridden by the
; compiled definition loaded by the compiled file associated with the book
; "prog2" (not shown here, but containing the redundant #+acl2-loop-only
; definition of prog2$).
; (in-package "ACL2")
;
; (include-book "prog2") ; redundant #+acl2-loop-only def. of prog2$
;
; (defun foo (x)
; (prog2$ 3 (mv x x)))
;
; (defthm foo-fact
; (equal (foo 4)
; (list 4 4))
; :rule-classes nil)
;
; (verify-guards foo)
;
; (defthm foo-fact-bogus
; (equal (foo 4)
; (list 4))
; :rule-classes nil)
;
; (defthm contradiction
; nil
; :hints (("Goal" :use (foo-fact foo-fact-bogus)))
; :rule-classes nil)
; After Version_4.1, prog2$ became just a macro whose calls expanded to forms
; (return-last 'progn ...). But the idea illustrated above is still relevant.
; We make this restriction for functions whose #+acl2-loop-only and
; #-acl2-loop-only definitions disagree. See
; fns-different-wrt-acl2-loop-only.
; By the way, it is important to include functions defined in #+acl2-loop-only
; that have no definition in #-acl2-loop-only. This becomes clear if you
; create a book with (in-package "ACL2") followed by the definition of LENGTH
; from axioms.lisp. In an Allegro CL build of ACL2 Version_3.3, you will get a
; raw Lisp error during the compilation phase when you apply certify-book to
; this book, complaining about redefining a function in the COMMON-LISP
; package.
; Note that we can avoid the restriction for local definitions, since those
; will be ignored in the compiled file.
(cond ((and (not (f-get-global 'in-local-flg state))
(not (global-val 'boot-strap-flg (w state)))
(not (f-get-global 'redundant-with-raw-code-okp state))
(let ((recp (getpropc (car names) 'recursivep nil wrld))
(bad-fns
; The test below isn't right if a built-in function with raw Lisp code has been
; promoted to logic mode after assigning state global
; 'verify-termination-on-raw-program-okp to t. However, that assignment may
; only be done with a trust tag, and the documentation warns that doing this
; promotion could be unsound. So we don't worry about that case here.
(if (eq (symbol-class (car names) wrld)
:program)
(f-get-global
'program-fns-with-raw-code
state)
(f-get-global
'logic-fns-with-raw-code
state))))
(if recp
(intersectp-eq recp bad-fns)
(member-eq (car names) bad-fns))))
(er soft ctx
"~@0"
(redundant-predefined-error-msg (car names) wrld)))
(t (value (cons 'redundant defun-mode)))))
(defun chk-acceptable-defuns-verify-guards-er (names ctx wrld state)
; The redundancy check during processing the definition(s) of names has
; returned 'verify-guards. We cause an error. If that proves to be too
; inconvenient for users, we could look into arranging for a call of
; verify-guards.
(let ((include-book-path
(global-val 'include-book-path wrld)))
(mv-let
(erp ev-wrld-and-cmd-wrld state)
(state-global-let*
((inhibit-output-lst
(cons 'error (f-get-global 'inhibit-output-lst state))))
; Keep the following in sync with pe-fn.
(let ((wrld (w state)))
(er-let*
((ev-wrld (er-decode-logical-name (car names) wrld :pe state))
(cmd-wrld (superior-command-world ev-wrld wrld :pe
state)))
(value (cons ev-wrld cmd-wrld)))))
(mv-let (erp1 val1 state)
(er soft ctx
"The definition of ~x0~#1~[~/ (along with the others in its ~
mutual-recursion clique)~]~@2 demands guard verification, ~
but there is already a corresponding existing definition ~
without its guard verified. ~@3Use verify-guards instead; ~
see :DOC verify-guards. ~#4~[Here is the existing ~
definition of ~x0:~/The existing definition of ~x0 appears ~
to precede this one in the same top-level command.~]"
(car names)
names
(cond
(include-book-path
(cons " in the book ~xa"
(list (cons #\a
(book-name-to-filename
(car include-book-path)
wrld
ctx)))))
(t ""))
(cond
((cddr include-book-path)
(cons "Note: The above book is included under the ~
following sequence of included books from outside ~
to inside, i.e., top-level included book ~
first:~|~&b.~|"
(list (cons #\b (book-name-lst-to-filename-lst
(reverse (cdr include-book-path))
(project-dir-alist wrld)
ctx)))))
((cdr include-book-path)
(cons "Note: The above book is included inside the book ~
~xb. "
(list (cons #\b (book-name-to-filename
(cadr include-book-path)
wrld
ctx)))))
(t ""))
(if erp 1 0))
(pprogn (if erp
state
(pe-fn1 wrld (standard-co state)
(car ev-wrld-and-cmd-wrld)
(cdr ev-wrld-and-cmd-wrld)
state))
(mv erp1 val1 state))))))
(defun chk-non-executablep (defun-mode non-executablep ctx state)
; We check that the value for keyword :non-executable is legal with respect to
; the given defun-mode.
(cond ((eq non-executablep nil)
(value nil))
((eq defun-mode :logic)
(cond ((eq non-executablep t)
(value nil))
(t (er soft ctx
"The :NON-EXECUTABLE flag for :LOGIC mode functions ~
must be ~x0 or ~x1, but ~x2 is neither."
t nil non-executablep))))
(t ; (eq defun-mode :program)
(cond ((eq non-executablep :program)
(value nil))
(t (er soft ctx
"The :NON-EXECUTABLE flag for :PROGRAM mode functions ~
must be ~x0 or ~x1, but ~x2 is neither."
:program nil non-executablep))))))
(defun chk-acceptable-defuns0 (fives ctx wrld state)
; This helper function for chk-acceptable-defuns factors out some computation,
; as requested by Daron Vroon for ACL2s purposes.
(er-let*
((defun-mode (get-unambiguous-xargs-flg :MODE
fives
(default-defun-mode wrld)
ctx state))
(stobjs-in-lst (get-stobjs-in-lst fives defun-mode ctx wrld state))
(non-executablep
(get-unambiguous-xargs-flg :NON-EXECUTABLE fives nil ctx state))
(verify-guards (get-unambiguous-xargs-flg :VERIFY-GUARDS
fives
'(unspecified)
ctx state)))
(er-progn
(chk-defun-mode defun-mode ctx state)
(chk-non-executablep defun-mode non-executablep ctx state)
(cond ((consp verify-guards)
; This means that the user did not specify a :verify-guards. We will default
; it appropriately.
(value nil))
((eq defun-mode :program)
(if (eq verify-guards nil)
(value nil)
(er soft ctx
"When the :MODE is :program, the only legal :VERIFY-GUARDS ~
setting is NIL. ~x0 is illegal."
verify-guards)))
((or (eq verify-guards nil)
(eq verify-guards t))
(value nil))
(t (er soft ctx
"The legal :VERIFY-GUARD settings are NIL and T. ~x0 is ~
illegal."
verify-guards)))
(let* ((symbol-class (cond ((eq defun-mode :program) :program)
((consp verify-guards)
(cond
((= (default-verify-guards-eagerness wrld)
0)
:ideal)
((= (default-verify-guards-eagerness wrld)
1)
(if (get-guardsp fives wrld)
:common-lisp-compliant
:ideal))
(t :common-lisp-compliant)))
(verify-guards :common-lisp-compliant)
((= (default-verify-guards-eagerness wrld)
3)
:common-lisp-compliant)
(t :ideal))))
(value (list* stobjs-in-lst defun-mode non-executablep symbol-class))))))
(defun get-boolean-unambiguous-xargs-flg-lst (key lst default ctx state)
(er-let* ((lst (get-unambiguous-xargs-flg-lst key lst default ctx state)))
(cond ((boolean-listp lst) (value lst))
(t (er soft ctx
"The value~#0~[ ~&0 is~/s ~&0 are~] illegal for XARGS key ~x1,
as ~x2 and ~x3 are the only legal values for this key."
lst key t nil)))))
(defun get-irrelevants-alist (fives)
(cond ((null fives) nil)
(t (acons (caar fives)
(irrelevant-vars (fourth (car fives)))
(get-irrelevants-alist (cdr fives))))))
(defun raw-lambda$s-to-lambdas (lst)
; Lst is a list of logically translated well-formed lambda objects whose bodies
; are tagged as having come from translated lambda$s. We create a set of pairs
; mapping ``raw Lisp lambda$s'' to their logic counterparts. This set of pairs
; will be added to the lambda$-alist by the function
; chk-acceptable-lambda$-translations.
; Here is a refresher course on the markings of macroexpanded lambda$s in raw
; Lisp and the logic translations of those lambda$s.
; Let x be a typical lambda$ expression, (lambda$ formals dcls* body). After
; macroexpansion in raw Lisp, x turns into
; (QUOTE (,*lambda$-marker* . (lambda$ formals dcls* body))).
; ---------------------------
; = x
; We've underlined the original lambda$ expression, x, in this raw Lisp quoted
; constant.
; Meanwhile, in the logic, the translation of the lambda$ expression is
; (QUOTE (lambda formals dcl' (return-last 'progn 'raw-x body'))).
; -------------------------------------------------------
; = x'
; The underlined evg is here named x'. X' is a typical element of lst, i.e., a
; logically translated well-formed lambda object whose body is tagged as having
; come from a lambda$.
; If x' is an element of lst then our set of pairs will contain the pair (x
; . x'). We can construct this pair from x' because x is embedded in x'.
(cond ((endp lst) nil)
(t (cons (cons (unquote (fargn (lambda-object-body (car lst)) 2))
(car lst))
(raw-lambda$s-to-lambdas (cdr lst))))))
(defun chk-acceptable-lambda$-translations1 (new-pairs ctx wrld state)
(cond
((null new-pairs) (value nil))
(t (let* ((key (car (car new-pairs)))
(val (cdr (car new-pairs))))
(mv-let (erp tkey bindings)
; Below we are setting stobjs-out = t and known-stobjs = t, since the more
; restrictive translation has already been made, if necessary.
(translate11-lambda-object key
t ; stobjs-out
nil ; bindings
t ; known-stobjs
nil ; flet-alist
key
ctx
wrld
(default-state-vars t)
nil)
(declare (ignore bindings))
(cond
(erp (er soft ctx
"The attempt to translate a lambda$ to be stored as a key ~
on lambda$-alist has caused an error, despite the fact ~
that this very same lambda$ was successfully translated ~
a moment ago! The error caused is:~%~@0~%~%The ~
offending lambda$ is ~x1. This is an implementation ~
error and you should contact the ACL2 developers."
tkey ; (really, a msg)
key))
((equal (unquote tkey) val)
(chk-acceptable-lambda$-translations1 (cdr new-pairs) ctx wrld state))
(t (er soft ctx
"Imperfect counterfeit translated lambda$, ~x0. Unless you ~
knowingly tried to construct a translated lambda$ (instead ~
of using lambda$ and letting ACL2 generate the ~
translation) this is an implementation error. Please ~
report such errors to the ACL2 developers.~%~%But if you ~
tried to counterfeit a lambda$ we should point out that we ~
don't understand why you would do such a thing! Your ~
counterfeit translated lambda$ won't enjoy the same ~
runtime support as our translated lambda$ even if you did ~
it perfectly. The lambda object you created would be ~
interpreted by *1*apply even in a guard-verified raw Lisp ~
function while our lambda$ translation would produce ~
compiled code.~%~%Nevertheless, here's what's wrong with ~
your counterfeit version: the lambda$ expression~%~Y01 ~
actually translates to~%~Y21 but your counterfeit claimed it ~
translates to~%~Y31."
key
nil
(unquote tkey)
val))))))))
(defun chk-acceptable-lambda$-translations2 (new-pairs lambda$-alist ctx state)
; We check that no key in new-pairs occurs with a different value in either
; lambda$-alist the rest of new-pairs.
(cond
((null new-pairs) (value nil))
(t (let* ((key (car (car new-pairs)))
(val (cdr (car new-pairs)))
(temp1 (assoc-equal key lambda$-alist)))
(cond
((and temp1 (not (equal val (cdr temp1))))
(er soft ctx
"A pair about to be added to lambda$-alist has the same key ~
associated with a different value on lambda$-alist already. ~
This is an implementation error. Please report it to the ACL2 ~
developers. The duplicate key is ~x0. On lambda$-alist that ~
key is mapped to the value ~x1. But we were about to map it ~
to the value ~x2. This shouldn't happen because both values ~
are allegedly the translation of the key!"
key
(cdr temp1)
val))
(t
(let ((temp2 (assoc-equal key (cdr new-pairs))))
(cond
((and temp2 (not (equal val (cdr temp2))))
(er soft ctx
"Two pairs about to be added to lambda$-alist have the same key ~
but different values. This is an implementation error. ~
Please report it to the ACL2 developers. The key is ~x0 and ~
the two values are ~x1 and ~x2. This shouldn't happen because ~
both values are allegedly the translation of the key!"
key
(cdr temp2)
val))
(t (chk-acceptable-lambda$-translations2 (cdr new-pairs)
lambda$-alist
ctx state))))))))))
(defun chk-acceptable-lambda$-translations
(symbol-class guards bodies ctx wrld state)
; This function computes, checks, and returns the new pairs we should add to
; lambda$-alist. It does not add them.
; To explain what this function does we first have to recap the world global
; 'lambda$-alist. Lambda$-alist maps the lambda expressions produced by the
; raw Lisp macroexpansion of lambda$ expressions to the logic translations of
; the lambda$ expressions. For example, if a defun mentioned (lambda$ (x) (+ 1
; x)) then the raw Lisp will contain (quote (,*lambda$-marker* . (lambda$ (x)
; (+ 1 x)))), and the lambda$-alist will map (lambda$ (x) (+ 1 x)) to (lambda
; (x) (binary-+ '1 x)). The idea is that when apply$-lambda sees the quoted,
; marked, untranslated lambda$ expression it will use lambda$-alist to map it
; to its logical counterpart so we can do guard verification etc.
; Lambda$-alist is also used by authenticate-tagged-lambda$ to confirm that a
; quoted LAMBDA object tagged as having come from a lambda$ is actually
; produced by translating the lambda$. That check can be made without
; re-translating if the quoted LAMBDA object appears as a cdr of a pair on
; lambda$-alist.
; Warning: Don't change the format or content of these pairs without inspecting
; authenticate-tagged-lambda$ since failure to find the LAMBDA object just
; causes a silent (slower) re-translation.
; The first formal above is the symbol-class of the defuns being processed.
; The next two are the guards and bodies, all of which ultimately get
; transferred into raw Lisp. We must find every translated lambda$ in these
; terms and map their untranslated raw Lisp lambda$ counterparts to their logic
; translations. These pairs will be added to lambda$-alist. But to guard
; against the possibility that the user has incorrectly counterfeited a
; translated lambda$, we must check that the alleged translations are actually
; correct!
; For example, the user could manufacture a tagged lambda object that alleges
; that (lambda$ (x) (+ 1 x)) translated to:
; (LAMBDA (X)
; (RETURN-LAST 'PROGN
; '(LAMBDA$ (X) (+ 1 X))
; '23))
; If we added such a pair to lambda$-alist then the raw Lisp apply$-lambda
; would give the wrong answer when applying the untranslated (lambda$ (x) (+ 1
; x)).
(cond
((and (not (eq symbol-class :program))
(not (global-val 'boot-strap-flg wrld)))
(let ((new-pairs
(raw-lambda$s-to-lambdas
(collect-certain-lambda-objects-lst
:lambda$
(append guards bodies)
wrld
nil))))
(er-progn
(chk-acceptable-lambda$-translations1 new-pairs ctx wrld state)
(chk-acceptable-lambda$-translations2 new-pairs
(global-val 'lambda$-alist wrld)
ctx state)
(value new-pairs))))
(t (value nil))))
(defrec loop$-alist-entry
; The :flg field is normally nil, but it is t when adding an entry during
; certification and we are not in the process of including a sub-book.
(term . flg)
; Some day we might change the cheap-flg from nil to t.
nil)
(defun loop$-alist-term (loop$-form loop$-alist)
(let ((pair (assoc-equal loop$-form loop$-alist)))
(and pair
(access loop$-alist-entry (cdr pair) :term))))
; The following is akin to untrans-table but limited to CLTL primitives that
; are translated into ACL2 versions. This table is used to transform logically
; translated terms into raw Lisp runnable terms. Those transformed terms are
; compiled and run, possibly during the reduction of ground subexpressions to
; constants. So it is important that this table be a trusted constant and not
; something the user might augment as with untrans-table.
(defconst *primitive-untranslate-alist*
; Warning: It is important that none of these functions return multiple values!
'((binary-+ . +)
(binary-* . *)
(binary-append . append)
(binary-logand . logand)
(binary-logior . logior)
(binary-logxor . logxor)
(binary-logeqv . logeqv)
(unary-- . -)
(unary-/ . /)))
(defun convert-to-dfs-fn (form stobjs-out index bound-vars-rev out-exprs-rev)
(cond ((endp stobjs-out)
`(mv-let ,(reverse bound-vars-rev)
,form
(mv ,@(reverse out-exprs-rev))))
((eq (car stobjs-out) :df)
(let ((var (packn (list 'x index))))
(convert-to-dfs-fn form (cdr stobjs-out) (1+ index)
(cons var bound-vars-rev)
(cons `(to-df ,var) out-exprs-rev))))
(t
(let ((var (packn (list 'x index))))
(convert-to-dfs-fn form (cdr stobjs-out) (1+ index)
(cons var bound-vars-rev)
(cons var out-exprs-rev))))))
(defun convert-to-dfs (form stobjs-out)
; Stobjs-out is the expected stobjs-out from the given form. We return a
; modification of form that converts each value in a df position to a df.
(cond ((not (member-eq :df stobjs-out))
form)
((null (cdr stobjs-out))
`(to-df ,form))
(t (convert-to-dfs-fn form stobjs-out 0 nil nil))))
(mutual-recursion
(defun logic-code-to-runnable-code (already-in-mv-listp term wrld)
; Note: This function used to be called ``twoify''.
; This function converts a translated term into something that can be executed
; in raw Lisp. The logic translation of an mv is a list, with the concomitant
; translation of mv-let into bindings expressed in terms of car/cdr nests. A
; minor problem is that + and other primitive arithmetic expressions are turned
; into binary-+, etc., which if allowed to persist would prevent the compiler
; from making routine optimizations if TYPE declarations allow. The fix here
; is to find every call of a multi-valued function and wrap it in a
; multiple-value-list which converts it to a list of values as the logic treats
; it. While we're at it we turn calls of binary-+ into + so Lisp can recognize
; the chance to optimize arithmetic. In a simple test of CCL we saw no
; significant difference between the assembly produced for a well-declared (+ x
; y z) versus (+ x (+ y z)), so we do not flatten +-nests -- both result in two
; calls of the machine's +.
(declare (xargs :guard (and (pseudo-termp term)
(plist-worldp wrld))))
(cond ((variablep term) term)
((fquotep term)
; Should we transform quoted LAMBDA objects? No. First, they may not be in
; :FN slots and changing them would simply be wrong. Second, even if they are
; in :FN slots they are applied with apply$, which can't execute raw Lisp.
; It's just wrongheaded to think about transforming quoted LAMBDA objects!
term)
((flambdap (ffn-symb term))
(cons (list 'lambda (lambda-formals (ffn-symb term))
(logic-code-to-runnable-code nil
(lambda-body (ffn-symb term))
wrld))
(logic-code-to-runnable-code-lst (fargs term) wrld)))
((eq (ffn-symb term) 'if)
`(if ,(logic-code-to-runnable-code nil (fargn term 1) wrld)
,(logic-code-to-runnable-code nil (fargn term 2) wrld)
,(logic-code-to-runnable-code nil (fargn term 3) wrld)))
((eq (ffn-symb term) 'return-last)
(logic-code-to-runnable-code nil (fargn term 3) wrld))
((eq (ffn-symb term) 'do$)
; We use ec-call here in case stobjs are involved, following the use of ec-call
; farther below.
(let* ((do$-stobjs-out (do$-stobjs-out (fargs term)))
(call `(ec-call (do$ ,@(logic-code-to-runnable-code-lst
(fargs term)
wrld)))))
(convert-to-dfs
(if (cdr (do$-stobjs-out (fargs term)))
`(values-list ,call)
call)
do$-stobjs-out)))
((eq (ffn-symb term) 'mv-list)
; Since term is a fully translated term, we know it is of the form (mv-list 'k
; expr) where k is an explicit integer greater than 1 and the out-arity of expr
; is k. Thus, there is no need to wrap another mv-list around expr! But we do
; have to transform its subterms.
`(mv-list ,(fargn term 1)
,(logic-code-to-runnable-code t (fargn term 2) wrld)))
(t (let* ((fn (ffn-symb term))
(stobjs-out (stobjs-out fn wrld))
(pair (assoc-eq fn *primitive-untranslate-alist*))
(ec-call-p
; We sometimes use ec-call to avoid issues with live stobjs. Consider the
; following example.
; (include-book "projects/apply/top" :dir :system)
; (defstobj st fld)
; (defwarrant fld)
; (apply$ '(lambda (x)
; (declare (xargs :guard (stp x) :split-types t))
; (fld x))
; '((17)))
; The lambda can be guard-verified and will thus be compiled. But if we are
; not careful, this will lead to calling fld on (17), which logically should
; return 17 but, at least in our experiments, does not. (It has returned
; 637624320 !) We address this below by wrapping ec-call around functions that
; take stobj arguments, since *1* functions can handle such non-stobj
; arguments; after all, they do so in proofs. We also want to prevent live
; stobjs from getting into computations that involve such lambdas, so we check
; not only the stobjs-in but also the stobjs-out, so as to avoid raw Lisp
; calls of stobj creators.
; There is a similar issue for dfs.
(and (not pair) ; optimization
; The check on stobjs-out is not redundant with the check on stobjs-in, because
; the former rules out stobj creators.
(not (and (all-nils stobjs-out)
(all-nils (stobjs-in fn wrld))))))
(args (logic-code-to-runnable-code-lst (fargs term) wrld))
(call (if pair ; hence not ec-call-p
(cons (cdr pair) args)
(let ((call0 (cons-with-hint fn args term)))
(if ec-call-p
(list 'ec-call call0)
call0)))))
(cond
((and (not already-in-mv-listp)
(cdr stobjs-out)) ; (not (int= out-arity 1))
`(mv-list ',(length stobjs-out) ,call))
(t call))))))
(defun logic-code-to-runnable-code-lst (terms wrld)
(declare (xargs :guard (and (pseudo-term-listp terms)
(plist-worldp wrld))))
(cond ((endp terms) nil)
(t (cons-with-hint (logic-code-to-runnable-code nil (car terms) wrld)
(logic-code-to-runnable-code-lst (cdr terms) wrld)
terms)))))
(defun authenticate-tagged-lambda$ (x state)
; X is a well-formed LAMBDA object. If it is tagged as having come from a
; lambda$, we check that it is authentic, i.e., that the lambda$ expression it
; allegedly comes from actually translates to the LAMBDA object x. In general
; we do this by re-translating the lambda$. But there is a short-cut. The
; world global lambda$-alist contains every lambda$ expression ever seen by
; defun and only lambda$ expressions seen by defun. Lambda$ expressions,
; paired with their tagged translations, are stored there by defun, basically
; immediately after translation. So we know that the cdr of every pair on
; lambda$-alist is an authentic translation of a lambda$. Of course, this
; short-cut does not handle lambda$s in top-level type-in, or in theorems or
; other events, just defuns. So we still have do re-translation on some tagged
; LAMBDA objects we encounter.
(cond
((let ((dcl (lambda-object-dcl x)))
(and dcl
(double-float-types-p dcl)))
; Authenticate-tagged-lambda$ is called by
; make-compileable-guard-and-body-lambdas to evaluate using the original
; untranslated code when the lambda$ has already been translated (using
; translate11-lambda-object, with stobjs-out set to (nil) to insist on
; returning a single ordinary value). The actuals for a call of apply$ are in
; a list, so those are not dfs; therefore we have a problem when there is a
; formal parameter is a df. Consider for example the following (untranslated)
; term.
; (lambda$ (y)
; (declare (type double-float y))
; (from-df (df- y)))
; If the translation of that lambda$ is successfully authenticated by
; authenticate-tagged-lambda$, then the following will be produced by
; make-compileable-guard-and-body-lambdas (eliding the declare form, since it's
; not relevant to the point here).
; (LAMBDA (Y)
; (DECLARE ...)
; (FROM-DF (DF- Y)))
; The reason we can get a raw Lisp error when applying this lambda to a list of
; ordinary ACL2 objects is that the definition of (unary-df- x) includes the
; term (- (the double-float x)), and Lisp can complain when the wrong type is
; supplied (i.e., integer instead of double-float). This error may not be
; signaled with all Lisps since ACL2 is built with safety 0, but we have seen a
; similar such error in GCL with safety 0.
; By returning nil here, we arrange that
; make-compileable-guard-and-body-lambdas does what is necessary to get past
; this issue.
nil)
((lambda$-bodyp (lambda-object-body x))
; X is tagged as having been a lambda$. If we find it among the cdrs of
; lambda$-alist, we know it is authentic. Otherwise, we translate the lambda$
; and check.
(cond
((assoc-equal-cdr x (global-val 'lambda$-alist (w state)))
t)
(t (mv-let (erp obj bindings)
(translate11-lambda-object
(unquote (fargn (lambda-object-body x) 2))
'(nil) ; stobjs-out
nil ; bindings
nil ; known-stobjs
nil ; flet-alist
nil ; cform
'authenticate-tagged-lambda$-expression
(w state)
(default-state-vars state)
t) ; allow-counterfeitsp
(declare (ignore bindings))
(cond (erp nil)
((equal (unquote obj) x) t)
(t nil))))))
(t nil)))
(defun make-compileable-guard-and-body-lambdas (x state)
; X is a well-formed LAMBDA object. We want to create two new LAMBDA objects,
; one that, when applied in raw Lisp, will test the guard of x and the other to
; run the body of x. These created lambdas will be compiled and executed in
; raw Lisp when the guards of x have been verified and checked. So, in so far
; as possible, we want them to be fast raw Lisp lambda expressions. The
; challenge is that x is in translated form. Thus, for example, LOOP$s will
; have been converted to scion calls and multiple-value functions will be
; handled like they return lists. If x was generated by translating a lambda$
; we can recover the original type-in from the tagging -- after authenticating
; it. But if x was a quoted LAMBDA we have no choice but to do our best to
; convert the translated logic code into runnable lisp with
; logic-code-to-runnable-code.
(let ((formals (lambda-object-formals x))
(dcl (lambda-object-dcl x))
(body (lambda-object-body x))
(wrld (w state)))
; Note: dcl and body are in fully translated form.
(cond
((authenticate-tagged-lambda$ x state)
; X came from a lambda$, i.e., the quoted lambda$ expression found (in the
; second arg of the return-last) in the lambda-object-body really does
; translate to the quotation of x. So we can trust the original lambda$ to
; give us the user's code for this object.
(let* ((lambda$-expr
(unquote (fargn (lambda-object-body x) 2)))
(edcls
(edcls-from-lambda-object-dcls-short-cut (cddr lambda$-expr)))
(guard-lst
(get-guards2 edcls '(types guards) nil wrld nil nil nil)))
; Guard-lst is the list of untranslated conjuncts in the guard (plus any TYPE
; declarations) typed in the original lambda$. It can be run directly in raw
; Lisp (when x is guard verified). Note that the guard of the guard is T, but
; we don't bother to declare it below because the compiler can't make use of
; that.
; We'll create the lambda expression for the body by just re-using the whole
; lambda$-expr, after replacing the 'lambda$ by 'lambda. The right
; declarations are already in it.
(mv `(LAMBDA ,formals
(DECLARE (IGNORABLE ,@formals))
,(cond ((null guard-lst) 'T)
((null (cdr guard-lst)) (car guard-lst))
(t `(AND ,@guard-lst))))
`(LAMBDA ,@(cdr lambda$-expr)))))
(t
; If x is not an authentic lambda$ all we can do is use
; logic-code-to-runnable-code to transform the already-translated guard and
; body of the x. There are two main drawbacks. One is that the results of
; multiple valued functions are coerced to lists and then torn apart with
; car/cdr -- less efficiently than our raw Lisp can handle multiple values.
; The other is that LOOP$ statements run as calls of loop$ scions on LAMBDA
; objects rather than as raw Lisp loops. But if x is not from a lambda$ it was
; typed by the user and he or she couldn't have used mv-let or loop$ in it
; anyway because quoted LAMBDA objects are not translated.
(mv `(LAMBDA ,formals
(DECLARE (IGNORABLE ,@formals)
,@(remove-double-float-types (cdr dcl)))
,(logic-code-to-runnable-code
nil
(remove-guard-holders
(or (cadr (assoc-keyword :guard
(cdr (assoc-eq 'xargs
(cdr dcl)))))
*t*)
wrld)
wrld))
`(LAMBDA ,formals
,@(let ((d (remove-double-float-types (cdr dcl))))
(and d `((declare ,@d))))
,(logic-code-to-runnable-code
nil
(remove-guard-holders body wrld)
wrld)))))))
(defun convert-tagged-loop$s-to-pairs (lst flg wrld)
; Lst is a list of marked loop$ expressions and we return the list of pairs
; mapping the loop$s to loop$-alist-entry records that record their
; translations.
(cond ((endp lst) nil)
(t (cons (cons (unquote (fargn (car lst) 2))
(make loop$-alist-entry
:term (logic-code-to-runnable-code
nil
(fargn (car lst) 3)
wrld)
:flg flg))
(convert-tagged-loop$s-to-pairs (cdr lst) flg wrld)))))
(defun chk-acceptable-loop$-translations1 (new-pairs ctx wrld state)
(cond
((null new-pairs) (value nil))
(t (let* ((key (car (car new-pairs)))
(val (cdr (car new-pairs)))
(val-term (access loop$-alist-entry val :term)))
(mv-let (erp tkey bindings)
; Below we use stobjs-out = t and known-stobjs = t, since suitable translation
; (insisting on single-threadedness) has already been performed to create the
; defun bodies that we are exploring.
(translate11-loop$ key
t ; stobjs-out
nil ; bindings
t ; known-stobjs
nil ; known-floats (irrelevant since stobjs-out = t)
nil ; flet-alist
key
ctx
wrld
(default-state-vars t))
(declare (ignore bindings))
(cond
(erp (er soft ctx
"The attempt to translate a loop$ to be stored as a key ~
on loop$-alist has caused an error, despite the fact ~
that this very same loop$ was successfully translated ~
a moment ago! The error caused is:~%~@0~%~%The ~
offending loop$ is ~x1. This is an implementation ~
error and you should contact the ACL2 developers."
tkey ; (really, a msg)
key))
((and (tagged-loop$p tkey)
(equal (logic-code-to-runnable-code
nil
(fargn tkey 3)
wrld)
val-term))
; The error message below makes it seem like the loop$ is expected to translate
; to val-term. But that's not true. The translation of (loop$ ...) is,
; actually, (RETURN-LAST 'PROGN '(LOOP$ ...) meaning). And val-term here is
; supposed to be meaning.
(chk-acceptable-loop$-translations1 (cdr new-pairs) ctx wrld state))
(t (er soft ctx
"Imperfect counterfeit translated loop$, ~x0. Unless you ~
knowingly tried to construct a translated loop$ (instead ~
of using loop$ and letting ACL2 generate the translation) ~
this is an implementation error. Please report such ~
errors to the ACL2 developers.~%~%But if you tried to ~
counterfeit a loop$ we should point out that we don't ~
understand why you would do such a thing! Your ~
counterfeit translated loop$ won't enjoy the same runtime ~
support as our translated loop$ even if you did it ~
perfectly. Had you written a loop$ it would enter raw ~
Lisp as a CLTL loop statement and run fast when guard ~
verified. But the counterfeit term will just use the ~
logically translated term you claimed was the semantics ~
of the loop$.~%~%Nevertheless, here's what's wrong with ~
your counterfeit version: the loop$ expression~%~Y01 ~
actually translates to~%~Y21, which when converted to ~
runnable raw Lisp is~%~Y31, but your counterfeit claimed the ~
runnable raw Lisp of its translation is~%~Y41."
key
nil
tkey
(logic-code-to-runnable-code
nil
(fargn tkey 3)
wrld)
val-term))))))))
(defun chk-acceptable-loop$-translations2 (new-pairs loop$-alist ctx state)
; We check that no key in new-pairs occurs with a different value in either
; loop$-alist the rest of new-pairs.
(cond
((null new-pairs) (value nil))
(t (let* ((key (car (car new-pairs)))
(val (cdr (car new-pairs)))
(val-term (access loop$-alist-entry val :term))
(loop$-term (loop$-alist-term key loop$-alist)))
(cond
((and loop$-term (not (equal val-term loop$-term)))
(er soft ctx
"A pair about to be added to loop$-alist has the same key ~
associated with a different value on loop$-alist already. ~
This is an implementation error. Please report it to the ACL2 ~
developers. The duplicate key is ~x0. On loop$-alist that ~
key is mapped to the value ~x1. But we were about to map it ~
to the value ~x2. This shouldn't happen because both values ~
are allegedly the translation of the key!"
key
loop$-term
val-term))
(t
(let ((temp2 (assoc-equal key (cdr new-pairs))))
(cond
((and temp2 (not (equal val-term (cdr temp2))))
(er soft ctx
"Two pairs about to be added to loop$-alist have the same ~
key but different values. This is an implementation ~
error. Please report it to the ACL2 developers. The key ~
is ~x0 and the two values are ~x1 and ~x2. This shouldn't ~
happen because both values are allegedly the translation ~
of the key!"
key
(cdr temp2)
val-term))
(t (chk-acceptable-loop$-translations2 (cdr new-pairs)
loop$-alist
ctx state))))))))))
(defun chk-acceptable-loop$-translations
(symbol-class guards bodies ctx wrld state)
; This function computes, checks, and returns the new pairs we should add to
; loop$-alist. It does not add them.
; To explain what this function does we first have to recap the world global
; 'loop$-alist. Loop$-alist maps the loop$ expressions to their logic
; translations. For example, if a defun mentioned (loop$ for x in lst collect
; (cadr x)), the translation -- modulo our use of lambda$ for brevity -- is
; (collect$ (lambda$ (x) (car (cdr x))) lst). The raw Lisp of the defun will
; contain
; (if *aokp*
; (loop for x in lst collect (cadr x))
; (collect$ (lambda$ (x) (car (cdr x))) lst))
; But where did the Lisp compiler get the collect$ term, given that all that
; was in the source code was the loop$ and that the compiler doesn't have
; access to the ACL2 world? The answer is, it got it (indirectly) from the
; loop$-alist created by the defun and transferred into the .cert file when the
; book was certified.
; The first formal above is the symbol-class of the defuns being processed.
; The next two are the guards and bodies, all of which ultimately get
; transferred into raw Lisp. We must find every :top level translated loop$ in
; these terms and map them to their logic translations. (We do not need to add
; sub-loop$s of loop$s since they will never be encountered by the compiler.)
; These pairs will be added to loop$-alist. But to guard against the
; possibility that the user has incorrectly counterfeited a translated loop$,
; we must check that the alleged translations are actually correct!
(cond
((and (not (eq symbol-class :program))
(not (global-val 'boot-strap-flg wrld)))
(let* ((certify-book-info (f-get-global 'certify-book-info state))
(new-pairs
(convert-tagged-loop$s-to-pairs
(collect-certain-tagged-loop$s-lst
:top
(append guards bodies)
nil)
(and certify-book-info
(let ((path (global-val 'include-book-path wrld)))
(if (consp path)
(and (null (cdr path))
(equal (car path)
(access certify-book-info
certify-book-info
:full-book-name)))
(null path))))
wrld)))
(er-progn
; Note that we check the terms in new-pairs, not the :flg fields of the
; associated loop$-alist-entry. To understand how :flg is used, see Part 3 of
; the Essay on Hash Table Support for Compilation.
(chk-acceptable-loop$-translations1 new-pairs ctx wrld state)
(chk-acceptable-loop$-translations2 new-pairs
(global-val 'loop$-alist wrld)
ctx state)
(value new-pairs))))
(t (value nil))))
(mutual-recursion
(defun state-globals-set-by (term acc)
(cond ((or (variablep term)
(fquotep term))
acc)
((flambda-applicationp term)
(state-globals-set-by
(lambda-body (ffn-symb term))
(state-globals-set-by-lst (fargs term) acc)))
((member-eq (ffn-symb term) '(put-global makunbound-global))
(let ((qvar (fargn term 1)))
(cond ((and (quotep qvar)
(symbolp (unquote qvar)))
(cons (unquote qvar) acc)))))
(t (state-globals-set-by-lst (fargs term) acc))))
(defun state-globals-set-by-lst (termlist acc)
(cond ((endp termlist) acc)
(t (state-globals-set-by-lst
(cdr termlist)
(state-globals-set-by (car termlist) acc)))))
)
(mutual-recursion
(defun chk-lambdas-for-loop$-recursion1 (fn lambda-flg term fn-seenp
wrld ctx state)
; Fn is being defined with :loop$-recursion t. Initially, term is the body and
; we explore it recursively. Lambda-flg is non-nil if this occurrence of term
; is in the loop$ scion slot requiring a lambda expression -- in fact,
; lambda-flg is set to the name of that loop$ scion. We know that every loop$
; scion expects its :FN arg to be the first argument. Fn-seenp is a boolean
; indicating whether fn is called in the body of some well-formed lambda
; already seen in a loop$ scion. We either cause an error or return (value
; fn-seenp).
; We confirm that: (1) every quoted lambda-like object in the body is a
; well-formed lambda object, (2) every lambda in the body is in the :FN slot of
; a loop$ scion or else doesn't call fn, and (3) there is at least one lambda
; in a loop$ scion that calls fn.
(cond
(lambda-flg
; Term must be a lambda expression of the right arity for the loop$ scion
; named by lambda-flg.
(cond
((and (quotep term)
(consp (unquote term))
(eq (car (unquote term)) 'lambda))
(cond
((well-formed-lambda-objectp (unquote term) wrld)
(let ((style (loop$-scion-style lambda-flg))
(formals (lambda-object-formals (unquote term)))
(body (lambda-object-body (unquote term))))
(cond
((eq style :do)
(assert$
(eq lambda-flg 'do$)
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because there is a call of the loop$ scion, ~x1: calls of ~
~x1 are typically generated by expressions of the form ~
(LOOP$ ... DO ...), and we do not (yet) support recursion ~
inside such expressions. The offending lambda object is ~
~x2."
fn
lambda-flg
(unquote term))))
((eql (length formals) (if (eq style :plain) 1 2))
(chk-lambdas-for-loop$-recursion1
fn
nil
body
(or fn-seenp
(ffnnamep fn body))
wrld ctx state))
(t (er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~
~x0 because the loop$ scion ~x1 is called with a lambda ~
object of arity ~x2 where a lambda of arity ~x3 is ~
required. The offending lambda object is ~x4."
fn
lambda-flg
(length formals)
(if (eql style :plain) 1 2)
(unquote term))))))
(t
; This error cannot arise! If lambda-flg is set, it means we're in the :FN
; argument of a loop$ scion. If the actual is a quoted LAMBDA, as it is here,
; translate rejects it if it is not well-formed. Translate notes the
; gratuitous restriction which allows you to cons up an ill-formed lambda, but
; then the form wouldn't be quoted.
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because the loop$ scion ~x2 is called with an ill-formed ~
lambda object ~x1. We cannot generate measure conjectures for ~
ill-formed terms!"
fn term lambda-flg))))
((and (quotep term)
(symbolp (unquote term)))
; The only way this would have gotten past translation is if the quoted symbol
; is either fn itself or a previously admitted and badged function symbol. The
; error message explains why we reject it. However, if we ever change the
; expansion of (loop$ for e in x collect (fn e)) to expand simply to (collect$
; 'fn x), we'll have to permit this case.
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 because ~
it calls the loop$ scion ~x2 with ~x1 as the :FN argument. This ~
is equivalent to an admissible LOOP$ statement but it doesn't ~
execute as efficiently and admitting it would complicate the ~
generation of measure conjectures. Please rewrite this defun to ~
use the equivalent LOOP$!"
fn term lambda-flg))
(t
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because it calls the loop$ scion ~x2 with something other than a ~
lambda object, namely ~x1, as its :FN argument. We cannot ~
generate measure conjectures for computed terms."
fn term lambda-flg))))
((variablep term)
(value fn-seenp))
((fquotep term)
(cond ((and (consp (unquote term))
(eq (car (unquote term)) 'lambda))
(cond
((not (well-formed-lambda-objectp (unquote term) wrld))
; This is exactly the same error as the first one mentioned in this function.
; The simple fact is that every quoted lambda in a :loop$-recursion t defun
; must be well-formed, whether it occurs in a loop$ scion or not!
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because the body contains the ill-formed lambda object ~x1. ~
We cannot generate measure conjectures for ill-formed terms."
fn (unquote term)))
((loop$-recursion-ffnnamep fn (lambda-object-body (unquote term)))
(er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because the lambda object ~x1, which calls ~x0, occurs in ~
the body of ~x0 but not as the lambda object of a ~
translated loop$ statement. We cannot generate measure ~
conjectures since we cannot tell where or to what this ~
lambda object might be applied!"
fn (unquote term)))
(t (chk-lambdas-for-loop$-recursion1
fn
nil
(lambda-object-body (unquote term))
fn-seenp
wrld ctx state))))
(t (value fn-seenp))))
((lambda-applicationp term)
(er-let*
((fn-seenp (chk-lambdas-for-loop$-recursion1
fn
nil
(lambda-body (ffn-symb term))
fn-seenp wrld ctx state)))
(chk-lambdas-for-loop$-recursion1-lst
fn
nil
(fargs term)
fn-seenp wrld ctx state)))
(t (let ((style (loop$-scion-style (ffn-symb term))))
; If style is nil, the function being called is not a loop$ scion and so as we
; sweep through the arguments we provide lambda-flg = nil to each argument. We
; do this by providing the list nil, which is effectively a list of nils as
; long as we need. If the function being called is a loop$ scion, then style
; is :plain or :fancy, but in either case the functional argument is the first.
; So we provide ``(t)'' as the list of lambda-flgs, which is effectively a t
; followed by as many nils as needed. However, actually, instead of t we
; provide the particular loop$ scion involved so we can report it in the
; subsequent error message, if any.
(chk-lambdas-for-loop$-recursion1-lst
fn
(cond
((null style) nil)
(t (list (ffn-symb term))))
(fargs term)
fn-seenp wrld ctx state)))))
(defun chk-lambdas-for-loop$-recursion1-lst (fn lambda-flg-lst term-lst fn-seenp
wrld ctx state)
(cond
((endp term-lst) (value fn-seenp))
(t (er-let*
((fn-seenp
(chk-lambdas-for-loop$-recursion1
fn
(car lambda-flg-lst)
(car term-lst)
fn-seenp wrld ctx state)))
(chk-lambdas-for-loop$-recursion1-lst
fn
(cdr lambda-flg-lst)
(cdr term-lst)
fn-seenp wrld ctx state)))))
)
(defun chk-lambdas-for-loop$-recursion (fn body wrld ctx state)
; This function is only called by defun processing if xargs :loop$-recursion t
; was declared by the user in the defun of fn with the given body. We confirm
; that every quoted lambda object in the body is well-formed, every quoted
; lambda in the body is in the :FN slot of a loop$ scion or else doesn't call
; fn, and that every loop$ scion's :FN argument is in fact a quoted lambda. We
; also check that there is at least one quoted lambda in a loop$ scion that
; calls fn -- a check that is only there to de-confuse the user who
; unnecessarily declared :loop$-recursion t.
; Motivation: When generating measure conjectures for fn it is unnecessary to
; look at quoted lambdas, unless :loop$-recursion t is declared. If it is
; declared, one must only inspect the first (the :FN) arg of calls of loop$
; scions. No other quoted object is relevant to termination. Furthermore, you
; know the first argument of every loop$ scion is a well-formed quoted lambda,
; so you can just dive into the body of the lambda under the assumption that
; the first formal of the lambda is a member of the target of the loop$ scion.
; formed. The body may or may not call fn.
(er-let*
((fn-seenp
(chk-lambdas-for-loop$-recursion1 fn nil body nil wrld ctx state)))
(cond
(fn-seenp (value nil))
(t (er soft ctx
"It is illegal to use :loop$-recursion t in the defun of ~x0 ~
because ~x0 is never called in a loop$! We cause an error ~
simply because we expect you've made a mistake."
fn)))))
; ; It can be hard to understand why we check each of the cases above. The
; ; events below (most of which cause errors) provide an illustration of
; ; each of the six possible error messages generated by
; ; chk-lambdas-for-loop$-recursion.
; (include-book "projects/apply/top" :dir :system)
; (defun my-scion (fn x)(apply$ fn (list x))) ; no defwarrant yet
; (defun bar (x) (cons 'hi-from-bar x))
; (defwarrant bar)
; ; Now we provoke the successive errors.
; ; (1) ill-formed lambda in loop$ scion call -- can't happen
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) 23)
; (t (collect$ '(lambda (e) e . 47) x))))
; ; The error you see is caused by translate, not
; ; chk-lambdas-for-loop$-recursion.
; ; (2) loop$ scion is called with quoted symbol
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) t)
; (t (collect$ 'foo x))))
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) t)
; (t (collect$ 'bar x))))
; ; (3) loop$ scion is called with something other than a lambda object
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) t)
; (t (collect$ `(lambda (e) (,x e)) x))))
; ; (4) ill-formed lambda in non-:FN slot
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) (my-scion '(lambda (x) (foo x) . 47) x))
; (t (loop$ for e in x collect (foo e)))))
; ; (5) lambda object in non-loop$ scion calls fn
; (defun$ my-scion (fn x)
; (if (consp x)
; (cons (apply$ fn (list (car x)))
; (my-scion fn (cdr x)))
; nil))
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) (my-scion '(lambda (x) (foo x)) x))
; (t (loop$ for e in x collect (foo e)))))
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) (my-scion (lambda$ (x) (loop$ for e in x collect (foo e))) x))
; (t (loop$ for e in x collect (foo e)))))
; But well-formed lambda objects in non-loop$ scions are permitted if they
; don't call fn. We have to warrant my-scion first. The reason has nothing
; to do with chk-lambdas-for-loop$-recursion. If we don't warrant my-scion
; then translation and checking still approve this but the final check that
; foo is warrantable fails for lack of a badge for my-scion. You can see
; this by trying this without warranting my-scion:
; (defun$ bar (x) x)
; (defun foo (x)
; (declare (xargs :loop$-recursion t
; :measure (acl2-count x)))
; (cond ((atom x) (my-scion '(lambda (x) (bar x)) x))
; (t (loop$ for e in x collect (foo e)))))
; ; (6) unnecessary :loop$-recursion declaration:
; (defun foo (x)
; (declare (xargs :loop$-recursion t))
; (cond ((atom x) (my-scion '(lambda (x) (bar x)) x))
; (t (loop$ for e in x collect (bar e)))))
; The following record collects information related to the use of lambda,
; lambda$ and loop$ forms in defuns. We document what the items are below.
; But here we explain why we pack them together. These items are extracted and
; returned (ultimately) by chk-acceptable-defuns as part of its 2nd result, a
; list of over 20 items. When lambda objects were added, it would have been
; natural for us to add these items to the list. However, the number of
; lambda-related items keeps growing and some user books call
; chk-acceptable-defuns expecting the list to be of a certain length (whatever
; it was when the book was created). So we added one new item to
; chk-acceptable-defuns list, this record, and changed the user books to expect
; that new length. And now we're free to collect additional information during
; checking without having to mess with user books (unless they begin to use the
; lambda information here).
(defrec lambda-info
(loop$-recursion ; T or NIL indicating that recursive calls of the
; (single) function being defined are allowed
; inside LOOP$ statements. The function must
; be tame and return only one result!
new-lambda$-alist-pairs ; Maps the obvious untranslated terms to their
; respective translations
new-loop$-alist-pairs ; Maps untranslated loop$ statements to
; loop$-alist-entry records containing those
; translations after converting them from logic
; to runnable code.
)
nil)
; We need some machinery about type-prescriptions, which was in defthm.lisp
; before April 2021, in support of xargs :type-prescription for defun.
(mutual-recursion
(defun remove-lambdas1 (term)
(declare (xargs :guard (pseudo-termp term)
:verify-guards nil))
(cond ((or (variablep term)
(fquotep term))
(mv nil term))
(t (mv-let (changedp args)
(remove-lambdas-lst (fargs term))
(let ((fn (ffn-symb term)))
(cond ((flambdap fn)
(mv-let (changedp body)
(remove-lambdas1 (lambda-body fn))
(declare (ignore changedp))
(mv t
(subcor-var (lambda-formals fn)
args
body))))
(changedp (mv t (cons-term fn args)))
(t (mv nil term))))))))
(defun remove-lambdas-lst (termlist)
(declare (xargs :guard (pseudo-term-listp termlist)))
(cond ((consp termlist)
(mv-let (changedp1 term)
(remove-lambdas1 (car termlist))
(mv-let (changedp2 rest)
(remove-lambdas-lst (cdr termlist))
(cond ((or changedp1 changedp2)
(mv t (cons term rest)))
(t (mv nil termlist))))))
(t (mv nil nil))))
)
(defun remove-lambdas (term)
; Remove-lambdas returns the result of applying beta-reductions in term. This
; function preserves quote-normal form: if term is in quote-normal form, then
; so is the result.
(declare (xargs :guard (pseudo-termp term)
:verify-guards nil))
(mv-let (changedp ans)
(remove-lambdas1 term)
(declare (ignore changedp))
ans))
(defun type-prescription-disjunctp (var term)
; Warning: Keep this function in sync with
; subst-nil-into-type-prescription-disjunct.
; Var is a variable and term is a term. Essentially we are answering
; the question, ``is term a legal disjunct in the conclusion of a
; type-prescription about pat'' for some term pat. However, by this
; time all occurrences of the candidate pat in the conclusion have
; been replaced by some new variable symbol and that symbol is var.
; Furthermore, we will have already checked that the resulting
; generalized concl contains no variables other than var and the
; variables occurring in pat. So what this function actually checks
; is that term is either (equal var other-var), (equal other-var var),
; or else is some arbitrary term whose all-vars is identically the
; singleton list containing var.
; If term is one of the two equality forms above, then we know
; other-var is a variable in pat and that term is one of the disjuncts
; that says ``pat sometimes returns this part of its input.'' If term
; is of the third form, then it might have come from a
; type-restriction on pat, e.g., (and (rationalp pat) (<= pat 0)) or
; (compound-recognizerp pat), or it might be some pretty arbitrary
; term. However, we at least know that it contains no variables at
; all outside the occurrences of pat and that means that we can trust
; type-set-implied-by-term to tell us what this term implies about
; pat.
(cond ((variablep term)
; This could be a type-prescription disjunct in the generalized concl
; only if term is var, i.e., the original disjunct was equivalent to
; (not (equal pat 'nil)).
(eq term var))
((fquotep term) nil)
((flambda-applicationp term) nil)
(t (or (and (eq (ffn-symb term) 'equal)
(or (and (eq var (fargn term 1))
(variablep (fargn term 2))
(not (eq (fargn term 1) (fargn term 2))))
(and (eq var (fargn term 2))
(variablep (fargn term 1))
(not (eq (fargn term 2) (fargn term 1))))))
(equal (all-vars term) (list var))))))
(defun type-prescription-conclp (var concl)
; Warning: Keep this function in sync with
; subst-nil-into-type-prescription-concl.
; Var is a variable and concl is a term. We recognize those concl
; that are the macroexpansion of (or t1 ... tk) where every ti is a
; type-prescription-disjunctp about var.
; In the grand scheme of things, concl was obtained from the
; conclusion of an alleged :TYPE-PRESCRIPTION lemma about some term,
; pat, by replacing all occurrences of pat with some new variable,
; var. We also know that concl involves no variables other than var
; and those that occurred in pat.
(cond ((variablep concl) (type-prescription-disjunctp var concl))
((fquotep concl) nil)
((flambda-applicationp concl) nil)
((eq (ffn-symb concl) 'if)
(cond ((equal (fargn concl 1) (fargn concl 2))
(and (type-prescription-disjunctp var (fargn concl 1))
(type-prescription-conclp var (fargn concl 3))))
(t (type-prescription-disjunctp var concl))))
(t (type-prescription-disjunctp var concl))))
(defun subst-nil-into-type-prescription-disjunct (var term)
; Warning: Keep this function in sync with type-prescription-disjunctp.
; We assume var and term are ok'd by type-prescription-disjunctp.
; If term is of the form (equal var other-var) or (equal other-var var)
; we replace it by nil, otherwise we leave it alone.
(cond ((variablep term) term)
; The next two cases never happen, but we leave them in just to make
; sure we copy term modulo this substitution.
((fquotep term) term)
((flambda-applicationp term) term)
((and (eq (ffn-symb term) 'equal)
(or (and (eq var (fargn term 1))
(variablep (fargn term 2))
(not (eq (fargn term 1) (fargn term 2))))
(and (eq var (fargn term 2))
(variablep (fargn term 1))
(not (eq (fargn term 2) (fargn term 1))))))
*nil*)
(t term)))
(defun subst-nil-into-type-prescription-concl (var concl)
; Warning: Keep this function in sync with type-prescription-conclp.
; We know that var and concl are ok'd by type-prescription-conclp. So
; concl is a disjunction of terms, some of which are of the form
; (equal var other-var). We replace each of those disjuncts in concl
; with nil so as to produce that part of concl that is a disjunct of
; type restrictions. That is, if our answer is basic-term and vars is
; the list of all the other-vars in concl, then concl is equivalent to
; basic-term disjoined with the equality between var and each variable
; in vars.
(cond
((variablep concl) (subst-nil-into-type-prescription-disjunct var concl))
; The next two cases never happen.
((fquotep concl) concl)
((flambda-applicationp concl) concl)
((eq (ffn-symb concl) 'if)
(cond ((equal (fargn concl 1) (fargn concl 2))
(let ((temp (subst-nil-into-type-prescription-disjunct var
(fargn concl 1))))
(fcons-term* 'if
temp
temp
(subst-nil-into-type-prescription-concl var
(fargn concl 3)))))
(t (subst-nil-into-type-prescription-disjunct var concl))))
(t (subst-nil-into-type-prescription-disjunct var concl))))
(defun unprettyify-tp (term)
; This variant of unprettyify avoids giving special treatment to conjunctions,
; and hence is suitable for parsing terms into type-prescription rules. Unlike
; unprettyify, it returns (mv hyps concl).
(case-match term
(('implies t1 t2)
(mv-let (hyps concl)
(unprettyify-tp t2)
(mv (append? (flatten-ands-in-lit t1)
hyps)
concl)))
((('lambda vars body) . args)
(unprettyify-tp (subcor-var vars args body)))
(& (mv nil (remove-lambdas term)))))
(defun destructure-type-prescription (name typed-term term ens wrld)
; Warning: Keep this in sync with the :BACKCHAIN-LIMIT-LST case of
; translate-rule-class-alist.
; Note: This function does more than "destructure" term into a
; :TYPE-PRESCRIPTION rule, it checks a lot of conditions too and
; computes type-sets. However, it doesn't actually cause errors --
; note that state is not among its arguments -- but may return an
; error message suitable for printing with ~@. We return many
; results. The first is nil or an error message. The rest are
; relevant only if the first is nil and are described below. We code
; this way because the destructuring and checking are inextricably
; intertwined and when we destructure in order to add the rule, we do
; not have state around.
; We determine whether term is a suitable :TYPE-PRESCRIPTION lemma
; about the term typed-term. Term is suitable as a :TYPE-
; PRESCRIPTION lemma about typed-term if the conclusion of term,
; concl, is a disjunction of type-prescription disjuncts about
; typed-term. Each disjunct must either be an equality between
; typed-term and one of the variables occurring in typed-term, or else
; must be some term, such as (and (rationalp typed-term) (<=
; typed-term 0)) or (compound-recognizerp typed-term), that mentions
; typed-term and contains no variables outside those occurrences of
; typed-term.
; If term is unsuitable we return an error msg and nils. Otherwise we
; return nil and four more things: the list of hyps, a basic type
; set, a list of variables, and a ttree. In that case, term implies
; that when hyps are true, the type-set of typed-term is the union of the
; basic type-set together with the type-sets of the variables listed.
; The ttree records our dependencies on compound recognizers or other
; type-set lemmas in wrld. The ttree returned contains no 'assumption
; tags.
(let ((term (remove-guard-holders term wrld)))
(mv-let
(hyps concl)
(unprettyify-tp term)
(cond
((or (variablep typed-term)
(fquotep typed-term)
(flambda-applicationp typed-term))
(mv (msg "The :TYPED-TERM, ~x0, provided in the :TYPE-PRESCRIPTION ~
rule class for ~x1 is illegal because it is a variable, ~
constant, or lambda application. See :DOC type-prescription."
typed-term name)
nil nil nil nil nil))
((dumb-occur-lst typed-term hyps)
(mv (msg "The :TYPED-TERM, ~x0, of the proposed :TYPE-PRESCRIPTION ~
rule ~x1 occurs in the hypotheses of the rule. This would ~
cause ``infinite backchaining'' if we permitted ~x1 as a ~
:TYPE-PRESCRIPTION. (Don't feel reassured by this check: ~
infinite backchaining may occur anyway since it can be ~
caused by the combination of several rules.)"
typed-term
name)
nil nil nil nil nil))
(t
(let ((all-vars-typed-term (all-vars typed-term))
(all-vars-concl (all-vars concl)))
(cond
((not (subsetp-eq all-vars-concl all-vars-typed-term))
(mv (msg "~x0 cannot be used as a :TYPE-PRESCRIPTION rule as ~
described by the given rule class because the ~
:TYPED-TERM, ~x1, does not contain the ~#2~[variable ~&2 ~
which is~/variables ~&2 which are~] mentioned in the ~
conclusion. See :DOC type-prescription."
name
typed-term
(reverse
(set-difference-eq all-vars-concl all-vars-typed-term)))
nil nil nil nil nil))
(t (let* ((new-var (genvar (find-pkg-witness typed-term)
"TYPED-TERM" nil all-vars-typed-term))
(concl1 (subst-expr new-var typed-term concl)))
(cond
((not (type-prescription-conclp new-var concl1))
(mv (msg "~x0 is an illegal :TYPE-PRESCRIPTION lemma of the ~
class indicated because its conclusion is not a ~
disjunction of type restrictions about the ~
:TYPED-TERM ~x1. See :DOC type-prescription."
name typed-term)
nil nil nil nil nil))
(t (let ((vars (remove1-eq new-var (all-vars concl1)))
(basic-term
(subst-nil-into-type-prescription-concl new-var concl1)))
; Once upon a time, briefly, we got the type-set implied by (and hyps
; basic-term), thinking that we might need hyps to extract type
; information from basic-term. But the only var in basic-term is new
; so the hyps don't help much. The idea was to permit lemmas like
; (implies (rationalp x) (<= 0 (* x x))). Note that the guard for <=
; is satisfied only if we know that the product is rational, which we
; can deduce from the hyp. But when we try to process that lemma, the
; typed-term in generalized away, e.g., (implies (rationalp x) (<= 0
; Z)). Thus, the hyps don't help: the only var in basic-term is
; new-var. You could conjoin hyps and concl1 and THEN generalize the
; typed-term to new-var, thereby linking the occurrences of typed-term
; in the hyps to those in the concl. But this is very unhelpful
; because it encourages the creation of lemmas that contain the
; typed-term in the hyps. That is bad because type-set then
; infinitely backchains. In the face of these difficulties, we have
; reverted back to the simplest treatment of type-prescription lemmas.
(mv-let
(ts ttree)
(type-set-implied-by-term new-var nil basic-term ens wrld
nil)
(cond ((ts= ts *ts-unknown*)
(mv (msg "~x0 is a useless :TYPE-PRESCRIPTION ~
lemma because we can deduce no type ~
restriction about its :TYPED-TERM ~
(below represented by ~x1) from the ~
generalized conclusion, ~p2. See :DOC ~
type-prescription."
name
new-var
(untranslate concl1 t wrld))
nil nil nil nil nil))
((not (assumption-free-ttreep ttree))
; If type-set-implied-by-term requires that we force some assumptions,
; it is not clear what to do. For example, it is possible that the
; assumptions involve new-var, which makes no sense in the context of
; an application of this rule. My intuition tells me this error will
; never arise because for legal concls, basic-term is guard free. If
; there are :TYPE-PRESCRIPTION lemmas about the compound recognizers
; in it, they could have forced hyps. I think it unlikely, since the
; recognizers are Boolean. Well, I guess I could add a
; :TYPE-PRESCRIPTION lemma that said that under some forced hyp the
; compound-recognizer was actually t. In that case, the forced hyp
; would necessarily involve new-var, since that is the only argument
; to a compound recognizer. It would be interesting to see a living
; example of this situation.
(mv
(if (tagged-objectsp 'fc-derivation ttree)
(er hard 'destructure-type-prescription
"Somehow an 'fc-derivation, ~x0, has ~
found its way into the ttree returned ~
by type-set-implied-by-term."
(car (tagged-objects 'fc-derivation
ttree)))
(msg "~x0 is an illegal :TYPE-PRESCRIPTION ~
lemma because in determining the ~
type-set implied for its :TYPED-TERM, ~
~x1, by its conclusion the ~
~#2~[assumption ~&2 was~/assumptions ~
~&2 were~] and our :TYPE-PRESCRIPTION ~
preprocessor, ~
CHK-ACCEPTABLE-TYPE-PRESCRIPTION-RULE, ~
does not know how to handle this ~
supposedly unusual situation. It would ~
be very helpful to report this error to ~
the authors."
name typed-term
(tagged-objects 'assumption ttree)))
nil nil nil nil nil))
(t (mv nil hyps concl ts vars ttree))))))))))))))))
(defun get-xargs-type-prescription-lst (fives ctx state)
; Returns a list corresponding to fives that represents the unique value of the
; :type-prescription xargs (nil, when not supplied).
(cond
((endp fives) (value nil))
(t
(er-let* ((rest (get-xargs-type-prescription-lst (cdr fives) ctx state)))
(let* ((five (car fives))
(lst (fetch-dcl-fields1 '(:type-prescription) (fourth five))))
(cond
((null lst) (value (cons nil rest)))
((cdr lst)
(er soft ctx
"The :type-prescription keyword for xargs must occur at most ~
once in the declare forms for a function."))
(t (value (cons (car lst) rest)))))))))
(defun chk-type-prescription-lst (names arglists type-prescription-lst
ens wrld ctx state)
; Names and type-prescription-lst are true lists of the same length. We check
; that each name in names has a type-prescription in wrld that implies (is a
; subset of) the type-prescription represented by the pair (basic-ts . vars) at
; the corresponding position in type-prescription-lst, but we skip the check
; when there is nil rather than a pair at that position. We issue a warning
; when these are not equal.
; Note that wrld is a world constructed during admission of definitions of
; names, after type-prescriptions for names have been placed into wrld. So we
; expect these type-prescriptions to be hypothesis-free.
(cond
((endp names) (value nil))
((null (car type-prescription-lst))
(chk-type-prescription-lst (cdr names)
(cdr arglists)
(cdr type-prescription-lst)
ens wrld ctx state))
(t
(er-let* ((term (translate (car type-prescription-lst)
t
t ; logic-modep
t ; known-stobjs
ctx wrld state)))
(mv-let (erp hyps concl basic-ts vars ttree)
(destructure-type-prescription
(car names)
(cons (car names) (car arglists))
term ens wrld)
(declare (ignore concl ttree))
(cond
(erp (er soft ctx
"Illegal :type-prescription specified in xargs for name ~
~x0. Here is the error message one would see upon trying ~
to submit it as a :type-prescription rule:~|~%~@1"
(car names)
erp))
(hyps (er soft ctx
"Illegal :type-prescription specified in xargs for name ~
~x0: hypotheses are not allowed."
(car names)))
(t
(let* ((name (car names))
(old-tp (car (getpropc name 'type-prescriptions nil wrld)))
(uncertified-str
"This warning may occur when including an uncertified book, ~
when a :type-prescription declaration depends on ~
a type-prescription from a locally included book."))
(er-progn
(cond
((null old-tp)
(let ((str "It is illegal to specify a non-nil ~
:type-prescription in an xargs declaration for ~
~x0, because ACL2 computed no built-in type ~
prescription for ~x0."))
(cond ((f-get-global 'including-uncertified-p state)
; See comment below about including uncertified books.
(pprogn (warning$ ctx "Uncertified"
"~@0 ~@1"
(msg str name)
uncertified-str)
(value nil)))
(t
(er soft ctx str name)))))
(t
(let* ((old-basic-ts (access type-prescription old-tp :basic-ts))
(old-vars (access type-prescription old-tp :vars))
(subsetp-vars (subsetp-eq old-vars vars)))
(assert$
(null (access type-prescription old-tp :hyps))
(cond
((and (ts= old-basic-ts basic-ts)
(or (equal vars old-vars) ; optimization
(and subsetp-vars
(subsetp-eq vars old-vars))))
(value nil))
((and (ts-subsetp old-basic-ts basic-ts)
(subsetp-eq old-vars vars))
(pprogn
(warning$ ctx "Type-prescription"
"The type-prescription specified by the xargs ~
:type-prescription for ~x0 is strictly weaker ~
than the computed type of ~x0~@1."
name
(if (or (member-eq
'event
(f-get-global 'inhibit-output-lst
state))
(ld-skip-proofsp state))
""
" that is noted above"))
(value nil)))
(t
(let ((msg
(msg "The type-prescription specified by the xargs ~
:type-prescription for ~x0 is not implied by ~
the computed type of ~x0~@1.~|OLD:~| ~
~x2~|NEW:~| ~x3~|"
name
(if (or (member-eq
'event
(f-get-global 'inhibit-output-lst
state))
(ld-skip-proofsp state))
""
" that is noted above")
(access type-prescription old-tp :corollary)
term)))
(cond
((f-get-global 'including-uncertified-p state)
; We skip some checks for :type-prescription xargs when including an
; uncertified book, because they are presumably intended to pass but actually
; might not pass. In a certified book we expect type-prescription information
; to be preserved in the certificate's cert-data, from local include-book
; forms, but there is no such expectation when including an uncertified book
; (where cert-data is unavailable). It was a bit tempting to exclude the check
; whenever (ld-skip-proofsp state) holds, but the check seems necessary during
; the second pass of a non-trivial encapsulate. We could presumably check for
; such an environment, but the check seems inexpensive so for robustness, we
; make it except for the problematic case of including an uncertified book. An
; example in August 2021, pointed out by Eric Smith, was including the
; community book "kestrel/bv/defs-bitwise" as an uncertified book.
(pprogn (warning$ ctx "Uncertified"
"~@0~@1"
msg
uncertified-str)
(value nil)))
(t
(er soft ctx "~@0" msg))))))))))
(chk-type-prescription-lst (cdr names)
(cdr arglists)
(cdr type-prescription-lst)
ens wrld ctx state))))))))))
(defun global-stobjs-prop (names bodies guards wrld)
(mv-let (reads writes fns-seen)
(collect-global-stobjs-lst bodies wrld nil nil names)
(mv-let (reads writes fns-seen)
(collect-global-stobjs-lst guards wrld reads writes fns-seen)
(declare (ignore fns-seen))
(cond (writes (cons (set-difference-eq reads writes)
writes))
(reads (cons reads nil)) ; (cons reads writes)
(t nil)))))
(defun translate-guards (guards arglists stobjs-in-lst ctx wrld state
state-vars)
(cond ((endp guards) (value nil))
(t (mv-let (erp term bindings)
(translate11 (car guards)
nil
; Stobjs-out:
; Each guard returns one, non-stobj result. This arg is used for each guard.
; By using stobjs-out '(nil) we enable the thorough checking of the use of
; state. Thus, this call ensures that guards do not modify (or return)
; state. We are taking the conservative position because intuitively there is
; a confusion over the question of whether, when, and how often guards are run.
; By prohibiting them from modifying state we don't have to answer the
; questions about when they run.
'(nil)
nil
(if (eq stobjs-in-lst t)
t
(collect-non-*-df (car stobjs-in-lst)))
(if (eq stobjs-in-lst t)
nil
(collect-by-position '(:df)
(car stobjs-in-lst)
(car arglists)))
nil (car guards) ctx wrld state-vars)
(declare (ignore bindings))
(cond
(erp ; see cmp-to-error-triple
(er-soft ctx "Translate" "~@0" term))
(t (er-let* ((rst (translate-guards (cdr guards)
(cdr arglists)
(if (eq stobjs-in-lst t)
t
(cdr stobjs-in-lst))
ctx wrld state
state-vars)))
(value (cons term rst)))))))))
(defun chk-acceptable-defuns1 (names fives stobjs-in-lst defun-mode
symbol-class rc non-executablep ctx wrld
state
#+:non-standard-analysis std-p)
; WARNING: This function installs a world, hence should only be called when
; protected by a revert-world-on-error (a condition that should be inherited
; when called by chk-acceptable-defuns).
(let ((docs (get-docs fives))
(big-mutrec (big-mutrec names))
(arglists (strip-cadrs fives))
(default-hints (default-hints wrld))
(assumep (or (eq (ld-skip-proofsp state) 'include-book)
(eq (ld-skip-proofsp state) 'include-book-with-locals)))
(reclassifying-all-programp (and (eq rc 'reclassifying)
(all-programp names wrld)))
(state-vars (default-state-vars t)))
(er-let*
((wrld1 (chk-just-new-names names 'function rc ctx wrld state))
(wrld2 (update-w
big-mutrec
(store-stobjs-ins
names stobjs-in-lst
(putprop-x-lst2
; Warning: We rely on these 'formals properties, placed in reverse order from
; names, in the function termination-theorem-fn-subst (and its supporting
; functions).
names 'formals arglists
(putprop-x-lst1
names 'symbol-class symbol-class
wrld1)))))
(untranslated-measures
; If the defun-mode is :program, or equivalently, the symbol-class is :program,
; then we don't need the measures, other than to check that non-recursive
; functions aren't given measures.
(get-measures fives ctx state))
(measures (translate-measures untranslated-measures
(not (eq symbol-class :program))
ctx wrld2
state))
(ruler-extenders-lst (get-ruler-extenders-lst symbol-class fives
ctx
; Warning: If you move this binding of ruler-extenders-lst, then consider
; whether the 'formals property is still set on the new functions in wrld2.
wrld2 state))
(rel (get-unambiguous-xargs-flg
:WELL-FOUNDED-RELATION
fives
(default-well-founded-relation wrld2)
ctx state))
(do-not-translate-hints
(value (or assumep
(eq (ld-skip-proofsp state) 'initialize-acl2))))
(hints (if (or do-not-translate-hints
(eq defun-mode :program))
(value nil)
(let ((hints (get-hints fives)))
(if hints
(translate-hints+
(cons "Measure Lemma for" (car names))
hints
default-hints
ctx wrld2 state)
(value nil)))))
(guard-hints (if (or do-not-translate-hints
(eq defun-mode :program))
(value nil)
; We delay translating the guard-hints until after the definition is installed,
; so that for example the hint setting :in-theory (enable foo), where foo is
; being defined, won't cause an error.
(value (append (get-guard-hints fives)
default-hints))))
(std-hints #+:non-standard-analysis
(cond
((and std-p (not assumep))
(translate-hints+
(cons "Std-p for" (car names))
(get-std-hints fives)
default-hints
ctx wrld2 state))
(t (value nil)))
#-:non-standard-analysis
(value nil))
(otf-flg (if do-not-translate-hints
(value nil)
(get-unambiguous-xargs-flg :OTF-FLG
fives t ctx state)))
(guard-debug (get-unambiguous-xargs-flg :GUARD-DEBUG
fives
; Note: If you change the following default for guard-debug, then consider
; changing it in verify-guards as well, and fix the "Otherwise" message about
; :guard-debug in prove-guard-clauses.
nil ; guard-debug default
ctx state))
(measure-debug (get-unambiguous-xargs-flg :MEASURE-DEBUG
fives
nil ; measure-debug default
ctx state))
(guard-simplify (get-unambiguous-xargs-flg :GUARD-SIMPLIFY
fives
t ; guard-simplify default
ctx state))
(guard-simplify
; Keep this in sync with guard-simplify-p and a similar check in
; chk-acceptable-verify-guards-cmp. Unlike that function, here we return
; guard-simplify unchanged. That will require us to call guard-simplify-p in
; out utilities to convert :limited to nil, but that's a cheap thing to do and
; it maintains the use of the symbol GUARD-SIMPLIFY to denote a legal value for
; :guard-simplify.
(cond ((member-eq guard-simplify '(t :limited))
(value guard-simplify))
(t (er soft ctx "~@0" (guard-simplify-msg guard-simplify)))))
(split-types-lst (get-boolean-unambiguous-xargs-flg-lst
:SPLIT-TYPES fives nil ctx state))
(normalizeps (get-boolean-unambiguous-xargs-flg-lst
:NORMALIZE fives t ctx state))
(loop$-recursion-lst
; It will be illegal to specify a non-nil :loop$-recursion setting in any of
; the defuns of a mutually recursive clique. But to tell whether that has
; happened we first need to collect all the :loop$-recursion settings...
(get-unambiguous-xargs-flg-lst
:LOOP$-RECURSION
fives
nil ; :loop$-recursion default value
ctx state))
(type-prescription-lst
(get-xargs-type-prescription-lst fives ctx state)))
(er-progn
(cond
((and (consp (cdr loop$-recursion-lst))
(not (all-nils loop$-recursion-lst)))
(er soft ctx
"We do not support the declaration of non-nil :LOOP$-RECURSION ~
settings in MUTUAL-RECURSION."))
((and (null (cdr loop$-recursion-lst))
(car loop$-recursion-lst)
(not (eq (car loop$-recursion-lst) t)))
(er soft ctx
"The only legal values for the XARGS key :LOOP$-RECURSION are T ~
and NIL. ~x0 is not allowed."
(car loop$-recursion-lst)))
((and (car loop$-recursion-lst)
(global-val 'boot-strap-flg wrld))
(er soft ctx
"Implementors are not allowed to use :LOOP$-RECURSION in system ~
code!"))
(t (value nil)))
(er-let*
((wrld2a
(if (car loop$-recursion-lst)
; When (car loop$-recursion-lst) is non-nil we know it is actually T and that
; this is a singly-recursive defun of a fn in which the user has declared xargs
; :loop$-recursion t. The user means to use fn recursively inside one or more
; loop$ statements in the defun. This means we have to badge fn before
; translating! The user is supposed to understand that fn must be tame and
; must return 1 result -- we will check that. Just as we have already
; installed a (temporary) world, wrld2, in which the formals of fn, etc., have
; been stored in preparation for translation, we now also store a badge that
; asserts that fn is tame and returns 1 result. We will confirm this assertion
; before the defun is admitted. Note also that we know that fn is new and thus
; has no pre-existing badge so we can just cons a new one on. The resulting
; world is wrld2a and will be used henceforth.
(let* ((badge-table
(table-alist 'badge-table wrld2))
(userfn-structure
(cdr (assoc-eq :badge-userfn-structure badge-table)))
(fn (car names))
(badge (make apply$-badge
:arity (length (car arglists))
:out-arity 1 ; see note below
:ilks t)))
; Note: We have thought about allowing multi-valued functions here. The
; trouble is that we can't know the :out-arity until we've translated and we
; can't translate without a badge. One way to solve this would be to allow
; non-boolean values for :loop$-recursion, e.g., (xargs :loop$-recursion 3).
; But upon minimal reflection we don't see much use for multi-valued functions
; inside our loop$s because our loop$ scions all expect single-valued functions
; and return single values themselves. So if a multi-valued function were
; called inside the body, the multiple values would have to be combined inside
; the body to produce a single result and then some other expression would have
; to be used to supply the rest of the function's values. So we just await
; user complaints!
(update-w
t
(putprop
fn
'stobjs-out
'(NIL)
(putprop
'badge-table
'table-alist
(put-assoc-eq :badge-userfn-structure
(put-badge-userfn-structure-tuple-in-alist
(make-badge-userfn-structure-tuple
fn nil badge)
userfn-structure
ctx)
badge-table)
wrld2))))
(value wrld2))))
(er-progn
(cond
((not (and (symbolp rel)
(assoc-eq
rel
(global-val 'well-founded-relation-alist
wrld2a))))
(er soft ctx
"The :WELL-FOUNDED-RELATION specified by XARGS must be a ~
symbol which has previously been shown to be a well-founded ~
relation. ~x0 has not been. See :DOC well-founded-relation."
rel))
(t (value nil)))
(let ((mp (cadr (assoc-eq
rel
(global-val 'well-founded-relation-alist
wrld2a)))))
(er-let*
((bodies-and-bindings
; Keep this call in sync with the call of translate-bodies near the end of this
; definition.
(translate-bodies non-executablep ; t or :program
names
arglists
(get-bodies fives)
; bindings0 =
(if (car loop$-recursion-lst)
(list (cons (car names) '(NIL)))
(pairlis$ names names))
stobjs-in-lst
reclassifying-all-programp
ctx wrld2a state)))
(let* ((bodies (car bodies-and-bindings))
(bindings
(super-defun-wart-bindings
(cdr bodies-and-bindings)))
#+:non-standard-analysis
(non-classical-fns
(get-non-classical-fns bodies wrld2a)))
(er-progn
(if assumep
(value nil)
(er-progn
(chk-stobjs-out-bound names bindings ctx state)
#+:non-standard-analysis
(chk-no-recursive-non-classical
non-classical-fns
names mp rel measures bodies ctx wrld2a state)))
(if (car loop$-recursion-lst)
; If loop$-recursion was allowed, we have to check every lambda object in the
; body is well-formed and that the only ones that call fn are in loop$ scions,
; and that all of those lambdas have the right arity (1 or 2) for the loop$
; scion using them. We also make sure at least on loop$ scion has a lambda
; that calls fn, just to confirm that :loop$-recursion t is necessary.
(chk-lambdas-for-loop$-recursion
(car names)
(car bodies)
wrld2a ctx state)
(value nil))
(let* ((wrld30 (store-super-defun-warts-stobjs-in
names wrld2a))
(wrld31 (store-stobjs-out names bindings wrld30))
(wrld3 #+:non-standard-analysis
(if (or std-p
(null non-classical-fns))
wrld31
(putprop-x-lst1 names 'classicalp
nil wrld31))
#-:non-standard-analysis
wrld31)
(wrld4 (if (store-cert-data t bodies wrld state)
(update-translate-cert-data
(car names) wrld wrld3
:type :translate-bodies
:inputs names
:value bodies-and-bindings
:fns (all-fnnames-lst bodies)
:vars (state-globals-set-by-lst bodies nil))
wrld3))
(guards0 (get-guards fives split-types-lst nil
wrld2a)))
; Note: If :loop$-recursion t was specified for fn then wrld3 now contains the
; output arity of fn (in the stobjs-out property of fn). Thus, translation
; will not try to infer the output signature. Wrld3 is used everywhere a world
; is needed below, except in the optimizations using wrld2a as noted in the
; next Note.
(er-let* ((guards
; Keep this call in sync with the call of translate-term-list near the end of
; this definition.
(translate-guards
guards0
; Note: The use of wrld2a in get-guards above is just an optimization. We
; should more properly use wrld3 or even wrld4, but we know the presence of the
; stobjs-out properties won't affect get-guards because the only use it makes
; of the given world is to map from stobj names to the corresponding
; recognizers terms, e.g., from STATE to (STATE-P STATE).
arglists stobjs-in-lst ctx
; Note the use below of wrld3 instead of wrld2a. It is important that the
; proper stobjs-out be put on the new functions before we translate the guards!
; When we first allowed the functions being defined to be used in their guards
; (in v3-6), we introduced a soundness bug found by Sol Swords just after the
; release of v4-0, as follows.
; (defun foo (x)
; (declare (xargs :guard (or (consp x)
; (atom (foo '(a . b))))))
; (mv (car x)
; (mbe :logic (consp x)
; :exec t)))
;
; (defthm bad
; nil
; :hints (("goal" :use ((:instance foo (x nil)))))
; :rule-classes nil)
wrld3 state state-vars))
(split-types-terms
(translate-guards
(get-guards fives split-types-lst t wrld2a)
; Note: Wrld2a above is just the same optimization noted after the previous use
; of get-guards above.
; The arguments below are the same as those for the preceding call of
; translate-guards.
arglists stobjs-in-lst ctx wrld3
state state-vars)))
(er-progn
(if (eq defun-mode :logic)
; Although translate checks for inappropriate calls of :program functions,
; translate11 and translate1 do not.
(er-progn
(chk-logic-subfunctions
names names
guards wrld3 "guard"
ctx state)
(chk-logic-subfunctions
names names
split-types-terms wrld3
"split-types expression"
ctx state)
(chk-logic-subfunctions
names names
bodies wrld3 "body"
ctx state))
(value nil))
(if (global-val 'boot-strap-flg (w state))
(value nil)
(er-progn
(chk-badged-quoted-subfunctions
names names
guards wrld3 "guard"
ctx state)
(chk-badged-quoted-subfunctions
names names
split-types-terms wrld3
"split-types expression"
ctx state)
(chk-badged-quoted-subfunctions
names names
bodies wrld3 "body"
ctx state)))
(if (eq symbol-class :common-lisp-compliant)
(er-progn
(chk-common-lisp-compliant-subfunctions
names names guards wrld3 "guard" ctx state)
(chk-common-lisp-compliant-subfunctions
names names split-types-terms wrld3
"split-types expression" ctx state)
(chk-common-lisp-compliant-subfunctions
names names bodies wrld3 "body" ctx state))
(value nil))
(mv-let
(erp val state)
; This mv-let is just an aside that lets us conditionally check a bunch of
; conditions we needn't do in assumep mode.
(cond
(assumep (mv nil nil state))
(t
(let ((ignores (get-ignores fives))
(ignorables (get-ignorables fives))
(irrelevants-alist (get-irrelevants-alist fives)))
(er-progn
(chk-free-and-ignored-vars-lsts names
arglists
guards
split-types-terms
measures
ignores
ignorables
bodies
ctx state)
(chk-irrelevant-formals names arglists
guards
split-types-terms
measures
ignores
ignorables
irrelevants-alist
bodies ctx state)
(chk-mutual-recursion names bodies ctx
state)))))
(cond
(erp (mv erp val state))
(t (er-let* ((new-lambda$-alist-pairs
(if non-executablep
(value nil)
(chk-acceptable-lambda$-translations
symbol-class
guards bodies
ctx wrld3 state)))
(new-loop$-alist-pairs
(if non-executablep
(value nil)
(chk-acceptable-loop$-translations
symbol-class
guards bodies
ctx wrld3 state)))
; We conclude by extending the world with the 'global-stobjs property when
; appropriate, in which case we check that this doesn't cause the bodies or
; guards to violate the global-stobjs invariants. We start by obtaining that
; property from the currently installed world, wrld2a, to support efficiency of
; the re-translations below. Those re-translations are nevertheless
; heavy-handed, as we are only checking for errors pertaining to the
; global-stobjs invariants; but we don't expect this to be a big drain on
; overall efficiency, and of course there is no re-translation at all
; with-global-stobj isn't used (or, as we see below if there is no recursion).
(global-stobjs-prop
(value
(global-stobjs-prop names bodies guards
wrld2a)))
(wrld5 (value (if global-stobjs-prop
(putprop-x-lst1
names
'global-stobjs
global-stobjs-prop
wrld4)
wrld4)))
(ignore ; see comment above
(cond
((and global-stobjs-prop
(not assumep)
(or (ffnnamesp-lst
names bodies)
(ffnnamesp-lst
names guards)))
; There is recursion, so we will retranslate the guards and bodies as discussed
; above.
(er-progn
(update-w t wrld5)
; Keep this call in sync with the call of translate-bodies earlier in this
; definition.
(translate-bodies
non-executablep
names
arglists
(get-bodies fives)
(if (car loop$-recursion-lst)
(list (cons (car names) '(NIL)))
(pairlis$ names names))
stobjs-in-lst
reclassifying-all-programp
ctx wrld5 state)
(translate-guards
guards0
arglists stobjs-in-lst ctx
wrld5 state state-vars)))
(t (value nil)))))
(value (list 'chk-acceptable-defuns
names
arglists
docs
nil ; doc-pairs
guards
measures
ruler-extenders-lst
mp
rel
hints
guard-hints
std-hints ;nil for non-std
otf-flg
bodies
symbol-class
normalizeps
reclassifying-all-programp
wrld5
non-executablep
guard-debug
measure-debug
split-types-terms
(make lambda-info
:loop$-recursion
(car loop$-recursion-lst)
:new-lambda$-alist-pairs
new-lambda$-alist-pairs
:new-loop$-alist-pairs
new-loop$-alist-pairs)
guard-simplify
type-prescription-lst)))))))))))))))))))
(defun conditionally-memoized-fns (fns memoize-table)
(declare (xargs :guard (and (symbol-listp fns)
(alistp memoize-table))))
(cond ((endp fns) nil)
(t
(let ((alist (cdr (assoc-eq (car fns) memoize-table))))
(cond
((and alist ; optimization
(let ((condition-fn (cdr (assoc-eq :condition-fn alist))))
(and condition-fn
(not (eq condition-fn t)))))
(cons (car fns)
(conditionally-memoized-fns (cdr fns) memoize-table)))
(t (conditionally-memoized-fns (cdr fns) memoize-table)))))))
;; Historical Comment from Ruben Gamboa:
;; I modified the function below to check for recursive
;; definitions using non-classical predicates.
(defun chk-acceptable-defuns (lst ctx wrld state
#+:non-standard-analysis std-p)
; WARNING: This function installs a world, hence should only be called when
; protected by a revert-world-on-error.
; Rockwell Addition: We now also return the non-executable flag.
; This function does all of the syntactic checking associated with defuns. It
; causes an error if it doesn't like what it sees. It returns the traditional
; 3 values of an error-producing, output-producing function. However, the
; "real" value of the function is a list of items extracted from lst during the
; checking. These items are:
; names - the names of the fns in the clique, in order (as that order is
; expected by partial-functions-table-guard; see
; partial-functions-table-guard-msg)
; arglists - their formals
; docs - their documentation strings
; pairs - the (section-symbol . citations) pairs parsed from docs
; guards - their translated guards
; measures - their translated measure terms
; ruler-extenders-lst
; - their ruler-extenders
; mp - the domain predicate (e.g., o-p) for well-foundedness
; rel - the well-founded relation (e.g., o<)
; hints - their translated hints, to be used during the proofs of
; the measure conjectures, all flattened into a single list
; of hints of the form ((cl-id . settings) ...).
; guard-hints
; - like hints but to be used for the guard conjectures and
; untranslated
; std-hints (always returned, but only of interest when
; #+:non-standard-analysis)
; - like hints but to be used for the std-p conjectures
; otf-flg - t or nil, used as "Onward Thru the Fog" arg for prove
; bodies - their translated bodies
; symbol-class
; - :program, :ideal, or :common-lisp-compliant
; normalizeps
; - list of Booleans, used to determine for each fn in the clique
; whether its body is to be normalized
; reclassifyingp
; - t or nil, t if this is a reclassifying from :program
; with identical defs.
; wrld - a modified wrld in which the following properties
; may have been stored for each fn in names:
; 'formals, 'stobjs-in and 'stobjs-out
; non-executablep - t, :program, or nil according to whether these defuns
; are to be non-executable. Defuns with non-executable t may
; violate the translate conventions on stobjs.
; guard-debug
; - t or nil, used to add calls of EXTRA-INFO to guard conjectures
; measure-debug
; - t or nil, used to add calls of EXTRA-INFO to measure conjectures
; split-types-terms
; - list of translated terms, each corresponding to type
; declarations made for a definition with XARGS keyword
; :SPLIT-TYPES T
; lambda-info
; - a lambda-info record (q.v.) containing information about
; lambda objects gleaned during the acceptability check. The
; information includes whether recursive calls of the
; (singly-defined) new function is allowed in lambdas and about
; translations of lambda$ and loop$ forms encountered.
; guard-simplify
; - t or :limited, determining whether to simplify fully while
; generating the guard conjectures
; type-prescription-lst
; - collect all values of :type-prescription xargs, where each
; value is a type-prescription record derived from the formula
; supplied if any, else nil
(er-let*
((fives (chk-defuns-tuples lst nil ctx wrld state))
; Fives is a list in 1:1 correspondence with lst. Each element of
; fives is a 5-tuple of the form (name args doc edcls body). Consider the
; element of fives that corresponds to
; (name args (DECLARE ...) "Doc" (DECLARE ...) body)
; in lst. Then that element of fives is (name args "Doc" (...) body),
; where the ... is the cdrs of the DECLARE forms appended together.
; No translation has yet been applied to them. The newness of name
; has not been checked yet either, though we know it is all but new,
; i.e., is a symbol in the right package. We do know that the args
; are all legal.
(names (value (strip-cars fives))))
(er-progn
(chk-no-duplicate-defuns names ctx state)
(chk-xargs-keywords fives *xargs-keywords* ctx state)
(er-let*
((tuple (chk-acceptable-defuns0 fives ctx wrld state)))
(let* ((stobjs-in-lst (car tuple))
(defun-mode (cadr tuple))
(non-executablep (caddr tuple))
(symbol-class (cdddr tuple))
(rc (redundant-or-reclassifying-defunsp
defun-mode symbol-class (ld-skip-proofsp state) lst
ctx wrld
(ld-redefinition-action state)
fives non-executablep stobjs-in-lst
(default-state-vars t))))
(cond
((eq rc 'redundant)
(chk-acceptable-defuns-redundancy names defun-mode ctx wrld state))
((eq rc 'verify-guards)
; We avoid needless complication by simply causing a polite error in this
; case. If that proves to be too inconvenient for users, we could look into
; arranging for a call of verify-guards here.
(chk-acceptable-defuns-verify-guards-er names ctx wrld state))
((and (eq rc 'reclassifying)
(conditionally-memoized-fns names
(table-alist 'memoize-table wrld)))
; We no longer recall exactly why we have this restriction. However, after
; discussing this with Sol Swords we think it's because we tolerate all sorts
; of guard violations when dealing with :program mode functions, but we expect
; guards to be handled properly with :logic mode functions, including the
; condition function. If we verify termination and guards for the memoized
; function but not the condition, that could present a problem. Quite possibly
; we can relax this check somewhat after thinking things through -- e.g., if
; the condition function is a guard-verified :logic mode function -- if there
; is demand for such an enhancement.
(er soft ctx
"It is illegal to verify termination (i.e., convert from ~
:program to :logic mode) for function~#0~[~/s~] ~&0, because ~
~#0~[it is~/they are~] currently memoized with conditions; you ~
need to unmemoize ~#0~[it~/them~] first. See :DOC memoize."
(conditionally-memoized-fns names
(table-alist 'memoize-table wrld))))
(t
(chk-acceptable-defuns1 names fives
stobjs-in-lst defun-mode symbol-class rc
non-executablep ctx wrld state
#+:non-standard-analysis std-p))))))))
#+:non-standard-analysis
(defun build-valid-std-usage-clause (arglist body)
(cond ((null arglist)
(list (mcons-term* 'standardp body)))
(t (cons (mcons-term* 'not
(mcons-term* 'standardp (car arglist)))
(build-valid-std-usage-clause (cdr arglist) body)))))
#+:non-standard-analysis
(defun verify-valid-std-usage (names arglists bodies hints otf-flg
ttree0 ctx ens wrld state)
(cond
((null (cdr names))
(let* ((name (car names))
(arglist (car arglists))
(body (car bodies)))
(mv-let
(cl-set cl-set-ttree)
(clean-up-clause-set
(list (build-valid-std-usage-clause arglist body))
ens
wrld ttree0 state)
(pprogn
(increment-timer 'other-time state)
(let ((displayed-goal (prettyify-clause-set
cl-set
(let*-abstractionp state)
wrld)))
(mv-let
(col state)
(io? event nil (mv col state)
(cl-set displayed-goal name)
(cond ((null cl-set)
(fmt "~%The admission of ~x0 as a classical function ~
is trivial."
(list (cons #\0 name))
(proofs-co state)
state
nil))
(t
(fmt "~%The admission of ~x0 as a classical function ~
with non-classical body requires that it return ~
standard values for standard arguments. That ~
is, we must prove~%~%Goal~%~Q12."
(list (cons #\0 name)
(cons #\1 displayed-goal)
(cons #\2 (term-evisc-tuple nil state)))
(proofs-co state)
state
nil))))
(pprogn
(increment-timer 'print-time state)
(cond
((null cl-set)
(value (cons col cl-set-ttree)))
(t
(mv-let (erp ttree state)
(prove (termify-clause-set cl-set)
(make-pspv ens wrld state
:displayed-goal displayed-goal
:otf-flg otf-flg)
hints ens wrld ctx state)
(cond (erp (mv t nil state))
(t
(mv-let
(col state)
(io? event nil (mv col state)
(name)
(fmt "That completes the proof that ~x0 ~
returns standard values for standard ~
arguments."
(list (cons #\0 name))
(proofs-co state)
state
nil))
(pprogn
(increment-timer 'print-time state)
(value (cons col
(cons-tag-trees
cl-set-ttree
ttree)))))))))))))))))
(t (er soft ctx
"It is not permitted to use MUTUAL-RECURSION to define non-standard ~
predicates. Use MUTUAL-RECURSION to define standard versions of ~
these predicates, then use DEFUN-STD to generalize them, if that's ~
what you mean."))))
(defun union-eq1-rev (x y)
; This is like (union-eq x y) but is tail recursive and
; reverses the order of the new elements.
(cond ((endp x) y)
((member-eq (car x) y)
(union-eq1-rev (cdr x) y))
(t (union-eq1-rev (cdr x) (cons (car x) y)))))
(defun collect-hereditarily-constrained-fnnames (names wrld ans)
(cond ((endp names) ans)
(t (let ((name-fns (getpropc (car names)
'hereditarily-constrained-fnnames
nil
wrld)))
(cond
(name-fns
(collect-hereditarily-constrained-fnnames
(cdr names)
wrld
(union-eq1-rev name-fns ans)))
(t (collect-hereditarily-constrained-fnnames
(cdr names) wrld ans)))))))
(defun putprop-hereditarily-constrained-fnnames-lst (names fnnames-bodies
wrld)
; Names is a non-empty list of defined function names, and fnnames-bodies is in
; 1:1 correspondence with names. We set the hereditarily-constrained-fnnames
; property of each name in names, by collecting all the function names
; belonging to fnnames-bodies and filtering for the hereditarily constrained
; ones. We also add each name in names to the world global
; defined-hereditarily-constrained-fns.
; A ``hereditarily constrained function'' is either a constrained
; function, e.g., one introduced with defchoose or encapsulate, or
; else a defun'd function whose definition involves a hereditarily
; constrained function. The value of the
; hereditarily-constrained-fnnames property of a function symbol, fn,
; is a list of all the hereditarily constrained functions involved
; (somehow) in the definition of fn. If the list is nil, the symbol
; is in no sense constrained, but is either a primitive, e.g., car, or
; an ordinary defun'd function. If the list is a singleton, then its
; only element must necessarily be the fn itself and we know therefore
; that fn is constrained. Otherwise, the list has at least two
; elements and that fn is a defined but hereditarily constrained
; function. For example, if h is constrained and map-h is defined in
; terms of h, then the property for h will be '(h) and that for map-h
; will be '(map-h h). Mutually recursive cliques will list all the
; fns in the clique. One cannot assume the car of the list is fn.
(let ((fnnames (collect-hereditarily-constrained-fnnames
fnnames-bodies wrld nil)))
(cond
(fnnames
(global-set
'defined-hereditarily-constrained-fns
(append names
(global-val 'defined-hereditarily-constrained-fns wrld))
(putprop-x-lst1 names 'hereditarily-constrained-fnnames
(append names fnnames)
wrld)))
(t wrld))))
(defun split-inlines (names inlines not-inlines)
(declare (xargs :guard (symbol-listp names)))
(cond ((endp names)
(mv inlines not-inlines))
((let ((sname (symbol-name (car names))))
(terminal-substringp *inline-suffix*
sname
*inline-suffix-len-minus-1*
(1- (length sname))))
(split-inlines (cdr names)
(cons (car names) inlines)
not-inlines))
(t
(split-inlines (cdr names)
inlines
(cons (car names) not-inlines)))))
(defun put-ext-gen-info (ext-gens ext-gen-barriers names fnnames-bodies guards
wrld)
; Ext-gens and ext-gen-barriers are the values of world globals 'ext-gen and
; 'ext-gen-barriers in wrld, respectively, except that ext-gen-barriers may be
; :skip. We extend wrld by extending those globals as per the Essay on
; Attachable Stobjs, except that if input ext-gen-barriers is :skip then we do
; not extend the global, 'ext-gen-barriers. The extensions correspond to
; names, whose guards are the input, guards, and whose bodies call the function
; symbols in input fnnames-bodies.
(cond
((null ext-gens) ; optimization
(assert$ (eq ext-gen-barriers nil)
wrld))
((or (intersectp-eq ext-gens fnnames-bodies)
(intersectp-eq ext-gens (all-fnnames-lst guards)))
(mv-let (inlines not-inlines)
(split-inlines names nil nil)
(let* ((wrld (if inlines
(global-set 'ext-gens
(append inlines ext-gens)
wrld)
wrld))
(wrld (if not-inlines
(global-set 'ext-gen-barriers
(append not-inlines ext-gen-barriers)
wrld)
wrld)))
wrld)))
(t wrld)))
(defun defuns-fn1 (tuple ens big-mutrec names arglists docs pairs guards
guard-hints std-hints otf-flg guard-debug
guard-simplify bodies symbol-class normalizeps
split-types-terms lambda-info non-executablep
type-prescription-lst
#+:non-standard-analysis std-p
ctx state)
; See defuns-fn0.
; WARNING: This function installs a world. That is safe at the time of this
; writing because this function is only called by defuns-fn0, which is only
; called by defuns-fn, where that call is protected by a revert-world-on-error.
#-:non-standard-analysis
(declare (ignore std-hints))
(declare (ignore docs pairs))
(let* ((col (car tuple))
(subversive-p (cdddr tuple))
(wrld0 (w state))
(ext-gens (global-val 'ext-gens wrld0))
(ext-gen-barriers (global-val 'ext-gen-barriers wrld0)))
(er-let*
((wrld1 (update-w big-mutrec (cadr tuple)))
(wrld2 (update-w big-mutrec
(putprop-defun-runic-mapping-pairs names t wrld1)))
(wrld3 (update-w big-mutrec
(putprop-x-lst2-unless names 'guard guards *t*
wrld2)))
(wrld4 (update-w big-mutrec
(putprop-x-lst2-unless names 'split-types-term
split-types-terms *t* wrld3)))
#+:non-standard-analysis
(assumep
(value (or (eq (ld-skip-proofsp state) 'include-book)
(eq (ld-skip-proofsp state)
'include-book-with-locals))))
#+:non-standard-analysis
(col/ttree1 (if (and std-p (not assumep))
(verify-valid-std-usage names arglists bodies
std-hints otf-flg
(caddr tuple)
ctx ens wrld4 state)
(value (cons col (caddr tuple)))))
#+:non-standard-analysis
(col (value (car col/ttree1)))
(ttree1 #+:non-standard-analysis
(value (cdr col/ttree1))
#-:non-standard-analysis
(value (caddr tuple))))
(mv-let
(wrld5 ttree2)
(putprop-body-lst names arglists bodies normalizeps
(getpropc (car names) 'recursivep nil wrld4)
(make-controller-alist names wrld4)
#+:non-standard-analysis std-p
ens wrld4 wrld4 nil)
(er-progn
(update-w big-mutrec wrld5)
(mv-let
(wrld6 ttree2 state)
(putprop-type-prescription-lst names
subversive-p
(fn-rune-nume (car names)
t nil wrld5)
ens wrld5 ttree2 state)
(er-progn
; We want to defer printing the :type-prescription xargs warning, if any, until
; after printing the message about the type prescription (see
; print-defun-msg/type-prescriptions).
(update-w big-mutrec wrld6)
(let* ((wrld6a
(if (access lambda-info
lambda-info
:loop$-recursion)
; If loop$-recursion is non-nil, then names is a singleton list and the defun
; has an xargs :loop$-recursion t declaration. We store that fact under the
; loop$-recursion property of the function. We don't bother to store anything
; if loop$-recursion is nil.
(putprop (car names) 'loop$-recursion T wrld6)
wrld6))
(lambda$-alist-wrld6a
(global-val 'lambda$-alist wrld6a))
(new-lambda$-alist-pairs (access lambda-info
lambda-info
:new-lambda$-alist-pairs))
(wrld6b
(if (subsetp-equal new-lambda$-alist-pairs
lambda$-alist-wrld6a)
wrld6a
(global-set 'lambda$-alist
(union-equal new-lambda$-alist-pairs
lambda$-alist-wrld6a)
wrld6a)))
(loop$-alist-wrld6b
(global-val 'loop$-alist wrld6b))
(new-loop$-alist-pairs (access lambda-info
lambda-info
:new-loop$-alist-pairs))
(wrld6c
(if (subsetp-equal new-loop$-alist-pairs
loop$-alist-wrld6b)
wrld6b
(global-set 'loop$-alist
(union-equal new-loop$-alist-pairs
loop$-alist-wrld6b)
wrld6b)))
(fnnames-bodies (all-fnnames1 t bodies nil)))
(er-progn
(update-w big-mutrec wrld6c)
(er-let*
((wrld7 (update-w big-mutrec
(putprop-level-no-lst names wrld6c)))
(wrld8 (update-w big-mutrec
(putprop-primitive-recursive-defunp-lst
names wrld7)))
(wrld9 (update-w big-mutrec
(putprop-hereditarily-constrained-fnnames-lst
names fnnames-bodies wrld8)))
(wrld10 (update-w big-mutrec
(put-invariant-risk
names
bodies
non-executablep
symbol-class
guards
wrld9)))
(wrld11 (update-w big-mutrec
(putprop-x-lst1
names 'pequivs nil
(putprop-x-lst1 names 'congruences nil
wrld10))))
(wrld12 (update-w big-mutrec
(putprop-x-lst1 names 'coarsenings nil
wrld11)))
(wrld13 (update-w big-mutrec
(if non-executablep
(putprop-x-lst1
names 'non-executablep
non-executablep
wrld12)
wrld12)))
(wrld14 (update-w big-mutrec
(put-ext-gen-info
ext-gens ext-gen-barriers
names fnnames-bodies guards wrld13))))
(let ((wrld15
#+:non-standard-analysis
(if std-p
(putprop-x-lst1
names 'unnormalized-body nil
(putprop-x-lst1 names 'def-bodies nil wrld14))
wrld14)
#-:non-standard-analysis
wrld14))
(pprogn
(print-defun-msg names ttree2 wrld15 col state)
(set-w 'extension wrld15 state)
(er-progn
(chk-type-prescription-lst names arglists
type-prescription-lst
ens wrld15 ctx state)
(cond
((eq symbol-class :common-lisp-compliant)
(er-let*
((guard-hints (if guard-hints
(translate-hints
(cons "Guard for" (car names))
guard-hints
ctx wrld15 state)
(value nil)))
(pair (verify-guards-fn1 names guard-hints otf-flg
guard-debug guard-simplify
ctx state)))
; Pair is of the form (wrld . ttree3) and we return a pair of the same
; form, but we must combine this ttree with the ones produced by the
; termination proofs and type-prescriptions.
(value
(cons (car pair)
(cons-tag-trees ttree1
(cons-tag-trees
ttree2
(cdr pair)))))))
(t (value
(cons wrld15
(cons-tag-trees ttree1
ttree2))))))))))))))))))
(defun defuns-fn0 (names arglists docs pairs guards measures
ruler-extenders-lst mp rel hints guard-hints std-hints
otf-flg guard-debug guard-simplify measure-debug bodies
symbol-class normalizeps split-types-terms
lambda-info
non-executablep
type-prescription-lst
#+:non-standard-analysis std-p
ctx wrld state)
; WARNING: This function installs a world. That is safe at the time of this
; writing because this function is only called by defuns-fn, where that call is
; protected by a revert-world-on-error.
(cond
((eq symbol-class :program)
(defuns-fn-short-cut
t ; loop$-recursion-checkp, because chk-acceptable-defuns has approved.
(access lambda-info lambda-info :loop$-recursion)
names docs pairs guards measures split-types-terms
bodies non-executablep ctx wrld state))
(t
(let ((ens (ens state))
(big-mutrec (big-mutrec names)))
(er-let*
((tuple (put-induction-info (access lambda-info
lambda-info
:loop$-recursion)
names arglists
measures
ruler-extenders-lst
bodies
mp rel
hints
otf-flg
big-mutrec
measure-debug
ctx ens wrld state)))
(defuns-fn1
tuple
ens
big-mutrec
names
arglists
docs
pairs
guards
guard-hints
std-hints
otf-flg
guard-debug
guard-simplify
bodies
symbol-class
normalizeps
split-types-terms
lambda-info
non-executablep
type-prescription-lst
#+:non-standard-analysis std-p
ctx
state))))))
(defun strip-non-hidden-package-names (known-package-alist)
(declare (xargs :guard (known-package-alistp known-package-alist)))
(cond
((endp known-package-alist) nil)
(t
(let ((package-entry (car known-package-alist)))
(cond
((package-entry-hidden-p package-entry)
(strip-non-hidden-package-names (cdr known-package-alist)))
(t
(cons (package-entry-name package-entry)
(strip-non-hidden-package-names (cdr known-package-alist)))))))))
(defun in-package-fn (str state)
; Important Note: Don't change the formals of this function without
; reading the *initial-event-defmacros* discussion in axioms.lisp.
(cond ((not (stringp str))
(er soft 'in-package
"The argument to IN-PACKAGE must be a string, but ~
~x0 is not."
str))
((not (find-non-hidden-package-entry str (known-package-alist state)))
(er soft 'in-package
"The argument to IN-PACKAGE must be a known package ~
name, but ~x0 is not. The known packages are ~*1~@2"
str
(tilde-*-&v-strings
'&
(strip-non-hidden-package-names (known-package-alist state))
#\.)
(if (global-val 'include-book-path (w state))
(msg "~%NOTE: This error might be eliminated by certifying ~
the book,~|~x0.~|See :DOC certify-book."
(book-name-to-filename
(car (global-val 'include-book-path (w state)))
(w state)
'in-package))
"")))
(t (let ((state (f-put-global 'current-package str state)))
(value str)))))
(defun defstobj-functionsp (names embedded-event-lst)
; This function determines whether all the names in names are being defined as
; part of a defstobj or defabsstobj event. If so, it returns the name of the
; stobj; otherwise, nil.
; Explanation of the context: Defstobj and defabsstobj use defun to define the
; recognizers, accessors and updaters. But these events must install their own
; versions of the raw lisp code for these functions, to take advantage of the
; single-threadedness of their use. So what happens when defstobj or
; defabsstobj executes (defun name ...), where name is say an updater?
; Defuns-fn is run on the singleton list '(name) and the axiomatic def of name.
; At the end of the normal processing, defuns-fn computes a CLTL-COMMAND for
; name. When this command is installed by add-trip, it sets the
; symbol-function of name to the given body. Add-trip also installs a *1*name
; definition by oneifying the given body. But in the case of a defstobj (or
; defabsstobj) function we do not want the first thing to happen: we will
; compute a special body for the name and install it with its own CLTL-COMMAND.
; So to handle defstobj and defabsstobj, defuns-fn tells add-trip not to set
; the symbol-function. This is done by setting the ignorep flag in the defun
; CLTL-COMMAND. So the question arises: how does defun know that the name it
; is defining is being introduced by defstobj or defabsstobj? This function
; answers that question.
; Note that *1*name should still be defined as the oneified axiomatic body, as
; with any defun. Before v2-9 we introduced the *1* function at defun time.
; (We still do so if the function is being reclassified with an identical body,
; from :program mode to :logic mode, since there is no need to redefine its
; symbol-function -- -- indeed its installed symbol-function might be
; hand-coded as part of these sources -- but add-trip must generate a *1*
; body.) Because stobj functions can be inlined as macros (via the :inline
; keyword of defstobj), we need to defer definition of the *1* function until
; after the raw Lisp def (which may be a macro) has been added. We failed to
; do this in v2-8, which caused an error in CCL as reported by John
; Matthews:
; (defstobj tiny-state
; (progc :type (unsigned-byte 10) :initially 0)
; :inline t)
;
; (update-progc 3 tiny-state)
; Note: At the moment, defstobj and defabsstobj do not introduce any mutually
; recursive functions. So every name is handled separately by defuns-fns.
; Hence, names, here, is always a singleton, though we do not exploit that.
; Also, embedded-event-lst is always a list ee-entries, each being a cons with
; the name of some superevent like ENCAPSULATE, INCLUDE-BOOK, or DEFSTOBJ
; (which is also used for DEFABSSTOBJ), in the car. The ee-entry for the most
; immediate superevent is the first on the list. At the moment, defstobj and
; defabsstobj do not use encapsulate or other structuring mechanisms. Thus,
; the defstobj ee-entry will be first on the list. But we look up the list,
; just in case. The ee-entry for a defstobj or defabsstobj is of the form
; (defstobj name names) where name is the name of the stobj and names is the
; list of recognizers, accessors and updaters and their helpers.
(let ((temp (assoc-eq 'defstobj embedded-event-lst)))
(cond ((and temp
(subsetp-equal names (caddr temp)))
(cadr temp))
(t nil))))
; The following definition only supports non-standard analysis, but it seems
; reasonable to allow it in the standard version too.
; #+:non-standard-analysis
(defun index-of-non-number (lst)
(cond
((endp lst) nil)
((acl2-numberp (car lst))
(let ((temp (index-of-non-number (cdr lst))))
(and temp (1+ temp))))
(t 0)))
#+:non-standard-analysis
(defun non-std-error (fn index formals actuals)
(er hard fn
"Function ~x0 was called with the ~n1 formal parameter, ~x2, bound to ~
actual parameter ~x3, which is not a (standard) number. This is illegal, ~
because the arguments of a function defined with defun-std must all be ~
(standard) numbers."
fn (list index) (nth index formals) (nth index actuals)))
#+:non-standard-analysis
(defun non-std-body (name formals body)
; The body below is a bit inefficient in the case that we get an error.
; However, we do not expect to get errors very often, and the alternative is to
; bind a variable that we have to check is not in formals.
`(if (index-of-non-number (list ,@formals))
(non-std-error ',name
(index-of-non-number ',formals)
',formals
(list ,@formals))
,body))
#+:non-standard-analysis
(defun non-std-def-lst (def-lst)
(if (and (consp def-lst) (null (cdr def-lst)))
(let* ((def (car def-lst))
(fn (car def))
(formals (cadr def))
(body (car (last def))))
`((,@(butlast def 1)
,(non-std-body fn formals body))))
(er hard 'non-std-def-lst
"Unexpected call; please contact ACL2 implementors.")))
; Rockwell Addition: To support non-executable fns we have to be able,
; at defun time, to introduce an undefined function. So this stuff is
; moved up from other-events.lisp.
(defun make-udf-insigs (names wrld)
(cond
((endp names) nil)
(t (cons (list (car names)
(formals (car names) wrld)
(stobjs-in (car names) wrld)
; The following would normally be (stobjs-out (car names) wrld). However, we
; get a build-time error from read-user-stobj-alist in that case.
(getpropc (car names) 'stobjs-out '(nil) wrld))
(make-udf-insigs (cdr names) wrld)))))
(defun intro-udf (insig wrld)
; This function is called during pass 2 of an encapsulate. See the comment
; below about guards.
(case-match
insig
((fn formals stobjs-in stobjs-out)
(putprop
fn 'coarsenings nil
(putprop
fn 'congruences nil
(putprop
fn 'pequivs nil
(putprop
fn 'constrainedp t ; 'constraint-lst comes later
(putprop
fn 'hereditarily-constrained-fnnames (list fn)
(putprop
fn 'symbol-class :COMMON-LISP-COMPLIANT
(putprop-unless
fn 'stobjs-out stobjs-out nil
(putprop-unless
fn 'stobjs-in stobjs-in nil
(putprop
fn 'formals formals
(putprop fn 'guard
; We are putting a guard of t on a signature function, even though a :guard
; other than t might have been specified for this function. This may seem to
; be an error. However, proofs are skipped during that pass, so an incorrect
; guard proof obligation will not be noticed anyhow. Instead, guard
; verification takes place during the first pass of the encapsulate, which
; could indeed present a problem if we are not careful. However, we call
; function bogus-exported-compliants to check that we are not making that sort
; of mistake; see bogus-exported-compliants.
*t*
wrld)))))))))))
(& (er hard 'store-signature "Unrecognized signature!"))))
(defun intro-udf-lst1 (insigs wrld)
(cond ((null insigs) wrld)
(t (intro-udf-lst1 (cdr insigs)
(intro-udf (car insigs)
wrld)))))
(defun intro-udf-lst2 (insigs kwd-value-list-lst)
; Warning: Keep this in sync with oneify-cltl-code.
; Insigs is a list of internal form signatures, e.g., ((fn1 formals1 stobjs-in1
; stobjs-out1) ...), and we convert it to a "def-lst" suitable for giving the
; Common Lisp version of defuns, ((fn1 formals1 body1) ...), where each bodyi
; is just a throw to 'raw-ev-fncall with the signal that says there is no body.
; Note that the body we build (in this ACL2 code) is a Common Lisp body but not
; an ACL2 expression!
; kwd-value-list-lst is normally a list that corresponds by position to insigs,
; each of whose elements associates keywords with values; in particular it can
; associate :guard with the guard for the corresponding element of insigs.
; However, kwd-value-list-lst can be the atom 'non-executable-programp, which
; we use for proxy functions (see :DOC defproxy), i.e., :program mode functions
; with the xarg declaration :non-executable :program.
(cond
((null insigs) nil)
(t (cons `(,(caar insigs)
,(cadar insigs)
,@(cond
((eq kwd-value-list-lst 'non-executable-programp)
'((declare (xargs :non-executable :program))))
(t (let ((guard
(cadr (assoc-keyword :guard
(car kwd-value-list-lst)))))
(and guard
`((declare (xargs :guard ,guard)))))))
,(null-body-er (caar insigs)
(cadar insigs)
t))
(intro-udf-lst2 (cdr insigs)
(if (eq kwd-value-list-lst 'non-executable-programp)
'non-executable-programp
(cdr kwd-value-list-lst)))))))
(defun intro-udf-lst (insigs kwd-value-list-lst in-local-flg wrld state)
; Insigs is a list of internal form signatures. We know all the function
; symbols are new in wrld. We declare each of them to have the given formals,
; stobjs-in, and stobjs-out, symbol-class :common-lisp-compliant, a guard of t
; and constrainedp of t. We also arrange to execute a defun in the underlying
; Common Lisp so that each function is defined to throw to an error handler if
; called from ACL2.
(if (null insigs)
wrld
(put-cltl-command `(defuns nil nil
,@(intro-udf-lst2 insigs
(and (not (eq kwd-value-list-lst t))
kwd-value-list-lst)))
in-local-flg
(intro-udf-lst1 insigs wrld)
wrld
state)))
(defun defun-ctx (def-lst #+:non-standard-analysis std-p)
(cond ((atom def-lst)
(msg "( DEFUNS ~x0)"
def-lst))
((atom (car def-lst))
(cons 'defuns (car def-lst)))
((null (cdr def-lst))
#+:non-standard-analysis
(if std-p
(cons 'defun-std (caar def-lst))
(cons 'defun (caar def-lst)))
#-:non-standard-analysis
(cons 'defun (caar def-lst)))
(t (msg *mutual-recursion-ctx-string*
(caar def-lst)))))
(defun install-event-defuns (names event-form def-lst0 symbol-class
reclassifyingp non-executablep pair ctx wrld
state)
; Warning: Before changing the cltl-cmd argument of the call below of
; install-event, see the comment in cltl-def-from-name about how the
; cltl-command is used in cltl-def-from-name.
; See defuns-fn.
(install-event (cond ((null (cdr names)) (car names))
(t names))
event-form
(cond ((null (cdr names)) 'defun)
(t 'defuns))
(cond ((null (cdr names)) (car names))
(t names))
(cdr pair)
(cond
(non-executablep
`(defuns nil nil
,@(intro-udf-lst2
(make-udf-insigs names wrld)
(and (eq non-executablep :program)
'non-executable-programp))))
(t `(defuns ,(if (eq symbol-class :program)
:program
:logic)
,(if reclassifyingp
'reclassifying
(if (defstobj-functionsp names
(global-val 'embedded-event-lst
(car pair)))
(cons 'defstobj
; The following expression computes the stobj name, e.g., $S, for
; which this defun is supportive. The STOBJS-IN of this function is
; built into the expression created by oneify-cltl-code
; namely, in the throw-raw-ev-fncall expression (see
; oneify-fail-form). We cannot compute the STOBJS-IN of the function
; accurately from the world because $S is not yet known to be a stobj!
; This problem is a version of the super-defun-wart problem.
(defstobj-functionsp names
(global-val
'embedded-event-lst
(car pair))))
nil))
,@def-lst0)))
t
ctx
(car pair)
state))
(defun defuns-fn (def-lst state event-form #+:non-standard-analysis std-p)
; Important Note: Don't change the formals of this function without
; reading the *initial-event-defmacros* discussion in axioms.lisp.
; On Guards
; When a function symbol fn is defund the user supplies a guard, g, and a
; body b. Logically speaking, the axiom introduced for fn is
; (fn x1...xn) = b.
; After admitting fn, the guard-related properties are set as follows:
; prop after defun
; body b*
; guard g
; unnormalized-body b
; type-prescription computed from b
; symbol-class :ideal
; * We actually normalize the above. During normalization we may expand some
; boot-strap non-rec fns.
; In addition, we magically set the symbol-function of fn
; symbol-function b
; and the symbol-function of *1*fn as a program which computes the logical
; value of (fn x). However, *1*fn is quite fancy because it uses the raw body
; in the symbol-function of fn if fn is :common-lisp-compliant, and may signal
; a guard error if 'guard-checking-on is set to other than nil or :none. See
; oneify-cltl-code for the details.
; Observe that the symbol-function after defun may be a form that
; violates the guards on primitives. Until the guards in fn are
; checked, we cannot let raw Common Lisp evaluate fn.
; Intuitively, we think of the Common Lisp programmer intending to defun (fn
; x1...xn) to be b, and is declaring that the raw fn can be called only on
; arguments satisfying g. The need for guards stems from the fact that there
; are many Common Lisp primitives, such as car and cdr and + and *, whose
; behavior outside of their guarded domains is unspecified. To use these
; functions in the body of fn one must "guard" fn so that it is never called in
; a way that would lead to the violation of the primitive guards. Thus, we
; make a formal precondition on the use of the Common Lisp program fn that the
; guard g, along with the tests along the various paths through body b, imply
; each of the guards for every subroutine in b. We also require that each of
; the guards in g be satisfied. This is what we mean when we say fn is
; :common-lisp-compliant.
; It is, however, often impossible to check the guards at defun time. For
; example, if fn calls itself recursively and then gives the result to +, we
; would have to prove that the guard on + is satisfied by fn's recursive
; result, before we admit fn. In general, induction may be necessary to
; establish that the recursive calls satisfy the guards of their masters;
; hence, it is probably also necessary for the user to formulate general lemmas
; about fn to establish those conditions. Furthermore, guard checking is no
; longer logically necessary and hence automatically doing it at defun time may
; be a waste of time.
(with-ctx-summarized
(defun-ctx def-lst #+:non-standard-analysis std-p)
(let ((wrld (w state))
(def-lst0
#+:non-standard-analysis
(if std-p
(non-std-def-lst def-lst)
def-lst)
#-:non-standard-analysis
def-lst)
(event-form (or event-form (list 'defuns def-lst))))
(revert-world-on-error
(er-let*
((tuple (chk-acceptable-defuns def-lst ctx wrld state
#+:non-standard-analysis std-p)))
; Chk-acceptable-defuns puts the 'formals, 'stobjs-in and 'stobjs-out
; properties (which are necessary for the translation of the bodies).
; All other properties are put by the defuns-fn0 call below.
(cond
((eq (car tuple) 'redundant)
(stop-redundant-event ctx state
:name (caar def-lst0)
:defun-mode (cdr tuple)
:def-lst def-lst0))
(t
(enforce-redundancy
event-form ctx wrld
(let ((names (nth 1 tuple))
(arglists (nth 2 tuple))
(docs (nth 3 tuple))
(pairs (nth 4 tuple))
(guards (nth 5 tuple))
(measures (nth 6 tuple))
(ruler-extenders-lst (nth 7 tuple))
(mp (nth 8 tuple))
(rel (nth 9 tuple))
(hints (nth 10 tuple))
(guard-hints (nth 11 tuple))
(std-hints (nth 12 tuple))
(otf-flg (nth 13 tuple))
(bodies (nth 14 tuple))
(symbol-class (nth 15 tuple))
(normalizeps (nth 16 tuple))
(reclassifyingp (nth 17 tuple))
(wrld (nth 18 tuple))
(non-executablep (nth 19 tuple))
(guard-debug (nth 20 tuple))
(measure-debug (nth 21 tuple))
(split-types-terms (nth 22 tuple))
(lambda-info (nth 23 tuple))
(guard-simplify (nth 24 tuple))
(type-prescription-lst (nth 25 tuple)))
(with-useless-runes
(car names)
wrld
(er-let*
((pair (defuns-fn0
names
arglists
docs
pairs
guards
measures
ruler-extenders-lst
mp
rel
hints
guard-hints
std-hints
otf-flg
guard-debug
guard-simplify
measure-debug
bodies
symbol-class
normalizeps
split-types-terms
lambda-info
non-executablep
type-prescription-lst
#+:non-standard-analysis std-p
ctx
wrld
state)))
; Pair is of the form (wrld . ttree).
(er-progn
(chk-assumption-free-ttree (cdr pair) ctx state)
(if (access lambda-info lambda-info :loop$-recursion)
; If loop$-recursion is set we know this is a singly-recursive (not mutually
; recursive) defun that the user alleged was tame. We check that now.
(let ((wrld (car pair)))
(mv-let (erp msg-and-badge)
(ev-fncall-w 'badger
(list (car names) wrld)
wrld nil nil nil t t)
; If erp is t, then msg-and-badge is actually an error msg. Otherwise,
; msg-and-badge is (msg badge), where msg is either an error message or nil.
; When msg is nil, badge is the computed badge.
(let ((msg1 msg-and-badge)
(msg2 (if erp
nil
(car msg-and-badge)))
(badge (if erp
nil
(cadr msg-and-badge))))
(cond
((or erp msg2)
(er soft 'defun
"When :LOOP$-RECURSION T is declared for a ~
function, as it was for ~x0, we must assign ~
it a badge before we translate its body. ~
That assigned badge asserts that ~x0 returns ~
a single value and is tame. We then check ~
that assumption after translation by ~
generating the badge using the technique ~
that DEFWARRANT would use. But the attempt ~
to generate the badge has failed, indicating ~
that it is illegal to declare ~
:LOOP$-RECURSION T for this function. ~
~#1~[Our attempt to generate a badge ~
produced the following error:~/The error ~
message that would be reported by DEFWARRANT ~
is:~]~%~%~@2"
(car names) (if erp 0 1) (if erp msg1 msg2)))
((not (equal (access apply$-badge badge :out-arity) 1))
; This error can't happen! The world -- wrld3 of chk-acceptable-defuns1 -- has
; the stobjs-out property of fn set to a list of length 1. And the badger just
; looks there to find the :out-arity.
(er soft 'defun
"Impossible error! The final badger check in ~
DEFUNS-FN has failed on the :OUT-ARITY. ~
This is impossible given ~
chk-acceptable-defuns1. Please show the ~
implementors this bug!"))
((not (eq (access apply$-badge badge :ilks) t))
(er soft 'defun
"When :LOOP$-RECURSION T is declared for a ~
function the function must be tame. But ~x0 ~
is not! Its ilks are actually ~x1."
(car names)
(access apply$-badge badge :ilks)))
(t (value nil))))))
(value nil))
(install-event-defuns names event-form def-lst0 symbol-class
reclassifyingp non-executablep pair ctx wrld
state)))))))))))
:event-type 'defun
:event event-form))
(defun defun-fn (def state event-form #+:non-standard-analysis std-p)
; Important Note: Don't change the formals of this function without
; reading the *initial-event-defmacros* discussion in axioms.lisp.
; The only reason this function exists is so that the defmacro for
; defun is in the form expected by primordial-event-defmacros.
(defuns-fn (list def) state
(or event-form (cons 'defun def))
#+:non-standard-analysis std-p))
; Here we develop the :args keyword command that will print all that
; we know about a function.
(defun args-fn (name state)
(io? temporary nil (mv erp val state)
(name)
(let ((wrld (w state))
(channel (standard-co state)))
(cond
((member-eq name *stobjs-out-invalid*)
(pprogn (fms "Special form, basic to ACL2. See :DOC ~x0.~|~%"
(list (cons #\0 name))
channel state nil)
(value name)))
((and (symbolp name)
(function-symbolp name wrld))
(let* ((formals (formals name wrld))
(stobjs-in (stobjs-in name wrld))
(stobjs-out (stobjs-out name wrld))
(guard (untranslate (guard name nil wrld) t wrld))
(tp (find-runed-type-prescription
(list :type-prescription name)
(getpropc name 'type-prescriptions nil wrld)))
(tpthm (cond (tp (untranslate
(access type-prescription tp :corollary)
t wrld))
(t nil)))
(badge (executable-badge name wrld))
; If we're in boot-strap, executable-badge just caused a hard error. So if
; we're here we know that (getpropc '*badge-prim-falist* 'const nil wrld) is
; non-nil and is, in fact, a quoted constant. But we can't just use
; *badge-prim-falist* because it will not be known to the compiler during the
; boot-strapping.
(warrant (find-warrant-function-name name wrld))
(constraint-msg
(mv-let
(some-name constraint-lst)
(constraint-info name wrld)
(cond ((unknown-constraints-p constraint-lst)
"[UNKNOWN-CONSTRAINTS]")
(t (let ((constraint
(if some-name
(untranslate (conjoin constraint-lst)
t wrld)
t)))
(if (eq constraint t)
""
(msg "~y0" constraint))))))))
(pprogn
(fms "Function ~x0~|~
Formals: ~y1~|~
Signature: ~y2~|~
~ => ~y3~|~
Guard: ~q4~|~
Guards Verified: ~y5~|~
Defun-Mode: ~@6~|~
Type: ~#7~[built-in (or unrestricted)~/~q8~]~|~
Badge: ~#b~[~yc~/none~]~|~
Warrant: ~#d~[built-in~/~ye~/none~]~|~
~#9~[~/Constraint: ~@a~|~]~
~%"
(list (cons #\0 name)
(cons #\1 formals)
(cons #\2 (cons name
(prettyify-stobj-flags stobjs-in)))
(cons #\3 (prettyify-stobjs-out stobjs-out))
(cons #\4 guard)
(cons #\5 (eq (symbol-class name wrld)
:common-lisp-compliant))
(cons #\6 (defun-mode-string (fdefun-mode name wrld)))
(cons #\7 (if tpthm 1 0))
(cons #\8 tpthm)
(cons #\9 (if (equal constraint-msg "") 0 1))
(cons #\a constraint-msg)
(cons #\b (if badge 0 1))
(cons #\c badge)
(cons #\d (if (eq warrant t) 0 (if warrant 1 2)))
(cons #\e warrant))
channel state nil)
(value name))))
((and (symbolp name)
(getpropc name 'macro-body nil wrld))
(let ((args (macro-args name wrld))
(guard (untranslate (guard name nil wrld) t wrld)))
(pprogn
(fms "Macro ~x0~|~
Macro Args: ~y1~|~
Guard: ~Q23~|~%"
(list (cons #\0 name)
(cons #\1 args)
(cons #\2 guard)
(cons #\3 (term-evisc-tuple nil state)))
channel state nil)
(value name))))
((member-eq name '(let lambda declare quote))
(pprogn (fms "Special form, basic to the Common Lisp language. ~
See for example CLtL."
nil channel state nil)
(value name)))
(t (er soft :args
"~x0 is neither a function symbol nor a macro name known to ~
ACL2."
name))))))
(defmacro args (name)
(list 'args-fn name 'state))
; We now develop the code for verify-termination, a macro that is essentially
; a form of defun.
(defun make-verify-termination-def (old-def new-dcls wrld)
; Old-def is a def tuple that has previously been accepted by defuns. For
; example, if is of the form (fn args ...dcls... body), where dcls is a list of
; at most one doc string and possibly many DECLARE forms. New-dcls is a new
; list of dcls (known to satisfy plausible-dclsp). We create a new def tuple
; that uses new-dcls instead of ...dcls... but which keeps any member of the
; old dcls not specified by the new-dcls except for the :mode (if any), which
; is replaced by :mode :logic.
(let* ((fn (car old-def))
(args (cadr old-def))
(body (car (last (cddr old-def))))
(dcls (butlast (cddr old-def) 1))
(new-fields (dcl-fields new-dcls))
(modified-old-dcls (strip-dcls
(add-to-set-eq :mode new-fields)
dcls)))
(assert$
(not (getpropc fn 'non-executablep nil wrld))
`(,fn ,args
,@new-dcls
,@(if (not (member-eq :mode new-fields))
; At one time we also required (eq (default-defun-mode wrld) :program) here.
; But it seems safest to eliminate that condition, thus guaranteeing that the
; defun form specifies a :logic-mode definition. (Perhaps that was already
; guaranteed, but this way there is no doubt.)
'((declare (xargs :mode :logic)))
nil)
,@modified-old-dcls
,body))))
(defun make-verify-termination-defs-lst (defs-lst lst wrld)
; Defs-lst is a list of def tuples as previously accepted by defuns. Lst is
; a list of tuples supplied to verify-termination. Each element of a list is
; of the form (fn . dcls) where dcls satisfies plausible-dclsp, i.e., is a list
; of doc strings and/or DECLARE forms. We copy defs-lst, modifying each member
; by merging in the dcls specified for the fn in lst. If some fn in defs-lst
; is not mentioned in lst, we don't modify its def tuple except to declare it
; of :mode :logic.
(cond
((null defs-lst) nil)
(t (let ((temp (assoc-eq (caar defs-lst) lst)))
(cons (make-verify-termination-def (car defs-lst) (cdr temp) wrld)
(make-verify-termination-defs-lst (cdr defs-lst) lst wrld))))))
(defun chk-acceptable-verify-termination1 (lst clique fn1 ctx wrld state)
; Lst is the input to verify-termination. Clique is a list of function
; symbols, fn1 is a member of clique (and used for error reporting only). Lst
; is putatively of the form ((fn . dcls) ...) where each fn is a member of
; clique and each dcls is a plausible-dclsp, as above. That means that each
; dcls is a list containing documentation strings and DECLARE forms mentioning
; only TYPE, IGNORE, and XARGS. We do not check that the dcls are actually
; legal because what we will ultimately do with them in verify-termination-fn
; is just create a modified definition to submit to defuns. Thus, defuns will
; ultimately approve the dcls. By construction, the dcls submitted to
; verify-termination will find their way, whole, into the submitted defuns. We
; return nil or cause an error according to whether lst satisfies the
; restrictions noted above.
(cond ((null lst) (value nil))
((not (and (consp (car lst))
(symbolp (caar lst))
(function-symbolp (caar lst) wrld)
(plausible-dclsp (cdar lst))))
(er soft ctx
"Each argument to verify-termination must be of the form (name ~
dcl ... dcl), where each dcl is either a DECLARE form or a ~
documentation string. The DECLARE forms may contain TYPE, ~
IGNORE, and XARGS entries, where the legal XARGS keys are ~&0. ~
The argument ~x1 is illegal. See :DOC verify-termination."
*xargs-keywords*
(car lst)))
((not (member-eq (caar lst) clique))
(er soft ctx
"The function symbols whose termination is to be verified must ~
all be members of the same clique of mutually recursive ~
functions. ~x0 is not in the clique of ~x1. The clique of ~x1 ~
consists of ~&2. See :DOC verify-termination."
(caar lst) fn1 clique))
(t (chk-acceptable-verify-termination1 (cdr lst) clique fn1 ctx wrld
state))))
(defun uniform-defun-modes (defun-mode clique wrld)
; Defun-Mode should be a defun-mode. Clique is a list of fns. If defun-mode is
; :program then we return :program if every element of clique is
; :program; else nil. If defun-mode is :logic we return :logic if
; every element of clique is :logic; else nil.
(cond ((null clique) defun-mode)
((programp (car clique) wrld)
(and (eq defun-mode :program)
(uniform-defun-modes defun-mode (cdr clique) wrld)))
(t (and (eq defun-mode :logic)
(uniform-defun-modes defun-mode (cdr clique) wrld)))))
(defun chk-acceptable-verify-termination (lst ctx wrld state)
; We check that lst is acceptable input for verify-termination. To be
; acceptable, lst must be of the form ((fn . dcls) ...) where each fn is the
; name of a function, all of which are in the same clique and are in :program
; mode, not non-executable, where each dcls above is a plausible-dclsp. We
; cause an error or return (value nil).
(cond
((and (consp lst)
(consp (car lst))
(symbolp (caar lst)))
(cond
((not (function-symbolp (caar lst) wrld))
(er soft ctx
"The symbol ~x0 is not a function symbol in the current ACL2 world."
(caar lst)))
((and (not (programp (caar lst) wrld))
; If (caar lst) was introduced by encapsulate, then recover-defs-lst below will
; cause an implementation error. So we short-circuit our checks here,
; especially given since the uniform-defun-modes assertion below suggests that
; all functions should then be in :logic mode. Eventually, we will generate
; the empty list of definitions and treat the verify-termination as redundant,
; except: as a courtesy to the user, we may cause an error here if the function
; could not have been upgraded from :program mode.
(getpropc (caar lst) 'constrainedp nil wrld))
(er soft ctx
"The :LOGIC mode function symbol ~x0 was originally introduced ~
introduced not with DEFUN, but ~#1~[as a constrained ~
function~/with DEFCHOOSE~]. So VERIFY-TERMINATION does not make ~
sense for this function symbol."
(caar lst)
(cond ((getpropc (caar lst) 'defchoose-axiom nil wrld)
1)
(t 0))))
((getpropc (caar lst) 'non-executablep nil wrld)
(er soft ctx
"The :PROGRAM mode function symbol ~x0 is declared non-executable, ~
so ~x1 is not legal for this symbol. Such functions are intended ~
only for hacking with defattach; see :DOC defproxy."
(caar lst)
'verify-termination
'defun))
(t
(let ((clique (get-clique (caar lst) wrld)))
(assert$
; We maintain the invariant that all functions in a mutual-recursion clique
; have the same defun-mode. This assertion check is not complete; for all we
; know, lst involves two mutual-recursion nests, and only the one for (caar
; lst) has uniform defun-modes. But we include this simple assertion to
; provide an extra bit of checking.
(uniform-defun-modes (fdefun-mode (caar lst) wrld)
clique
wrld)
(chk-acceptable-verify-termination1 lst clique (caar lst) ctx wrld
state))))))
((atom lst)
(er soft ctx
"Verify-termination requires at least one argument."))
(t (er soft ctx
"The first argument supplied to verify-termination, ~x0, is not of ~
the form (fn dcl ...)."
(car lst)))))
(defun verify-termination1 (lst state)
(let* ((lst (cond ((and (consp lst)
(symbolp (car lst)))
(list lst))
(t lst)))
(ctx
(cond ((null lst) "(VERIFY-TERMINATION)")
((and (consp lst)
(consp (car lst)))
(cond
((null (cdr lst))
(cond
((symbolp (caar lst))
(cond
((null (cdr (car lst)))
(msg "( VERIFY-TERMINATION ~x0)" (caar lst)))
(t (msg "( VERIFY-TERMINATION ~x0 ...)" (caar lst)))))
((null (cdr (car lst)))
(msg "( VERIFY-TERMINATION (~x0))" (caar lst)))
(t (msg "( VERIFY-TERMINATION (~x0 ...))" (caar lst)))))
((null (cdr (car lst)))
(msg "( VERIFY-TERMINATION (~x0) ...)" (caar lst)))
(t (msg "( VERIFY-TERMINATION (~x0 ...) ...)" (caar lst)))))
(t (cons 'VERIFY-TERMINATION lst))))
(wrld (w state)))
(er-progn
(chk-acceptable-verify-termination lst ctx wrld state)
(let ((defs
; At one time we returned nil here if the chk-acceptable-verify-termination
; returned a value of :redundant. However, it was then possible for
; verify-termination to be redundant when that was undesirable. For an
; example, see community book books/system/tests/verify-termination/top.lisp.
(recover-defs-lst (caar lst) wrld)))
(value (make-verify-termination-defs-lst
defs
lst wrld))))))
(defun verify-termination-boot-strap-chk1 (lst n wrld state)
; Lst is a list of cdrs of defun forms; thus, the car of each element of lst is
; a symbol. Moreover, each such symbol is a known function symbol of wrld.
(cond ((endp lst) (value nil))
((<= (getpropc (caar lst) 'absolute-event-number
'(:error "Implementation error: Missing absolute event ~
number. Please contact the ACL2 ~
implementors.")
wrld)
n)
(verify-termination-boot-strap-chk1 (cdr lst) n wrld state))
(t
; To see why we cause the following error, consider the following example from
; :DOC verify-termination, modified to use verify-termination-boot-strap.
; (encapsulate
; ()
; (defun foo (x y)
; (declare (xargs :mode :program
; :guard (and (natp x) (natp y))))
; (if (or (zp x) (zp y))
; (list x y)
; (foo (1+ x) (1- y))))
; (local (defun foo (x y)
; (declare (xargs :measure (acl2-count y)
; :guard (and (natp x) (natp y))))
; (if (or (zp x) (zp y))
; (list x y)
; (foo (1+ x) (1- y)))))
; (verify-termination-boot-strap foo))
; If that were permitted in axioms.lisp, then one could start ACL2 and
; immediately prove the following.
; (defthm bad-lemma
; (zp x)
; :hints (("Goal" :induct (foo x 1)))
; :rule-classes nil)
(er soft 'verify-termination-boot-strap-chk1
"Implementation error: Attempted verify-termination-boot-strap ~
for function symbol ~x0 in the same encapsulate where ~x0 is ~
introduced. Please contact the ACL2 implementors, who should ~
read the comment in the defun of ~
verify-termination-boot-strap-chk1."
(caar lst)))))
(defun pre-encapsulate-absolute-event-number (wrld)
(cond ((endp wrld)
(er hard 'pre-encapsulate-absolute-event-number
"Implementation error: Empty world. Please contact the ACL2 ~
implementors."))
((and (eq (caar wrld) 'embedded-event-lst)
(eq (cadar wrld) 'global-value)
(eq (cdr (cddar wrld)) nil))
(max-absolute-event-number wrld))
(t (pre-encapsulate-absolute-event-number (cdr wrld)))))
(defun verify-termination-boot-strap-chk (lst state)
(let ((wrld (w state)))
(cond ((in-encapsulatep (global-val 'embedded-event-lst wrld)
nil)
(verify-termination-boot-strap-chk1
lst
(pre-encapsulate-absolute-event-number wrld)
wrld
state))
(t (value nil)))))
(defun verify-termination-boot-strap-fn1 (lst state event-form)
(let ((event-form (or event-form
(cons 'VERIFY-TERMINATION lst))))
(er-let*
((defs-lst (verify-termination1 lst state))
(ignore (verify-termination-boot-strap-chk defs-lst state)))
(defuns-fn
defs-lst
state
event-form
#+:non-standard-analysis
nil))))
(defun verify-termination-boot-strap-fn (lst state event-form)
(cond
((global-val 'boot-strap-flg (w state))
(let* ((skipp (eq (car lst) ':skip-proofs))
(lst (if skipp (cdr lst) lst)))
(cond
(skipp
(state-global-let*
((ld-skip-proofsp 'initialize-acl2))
(verify-termination-boot-strap-fn1 lst state event-form)))
(t
(when-logic
; It is convenient to use when-logic so that we skip verify-termination during
; pass1 of the boot-strap in axioms.lisp.
"VERIFY-TERMINATION"
(verify-termination-boot-strap-fn1 lst state event-form))))))
(t
; We do not allow users to use 'verify-termination-boot-strap. Why? See the
; comment in redundant-or-reclassifying-defunp about "verify-termination is
; now just a macro for make-event", and see the discussion about make-event at
; the end of :doc verify-termination.
(er soft 'verify-termination-boot-strap
"~x0 may only be used while ACL2 is being built. Use ~x1 instead."
'verify-termination-boot-strap
'verify-termination))))
(defmacro when-logic3 (str x)
(list 'if
'(eq (default-defun-mode-from-state state)
:program)
(list 'er-progn
(list 'skip-when-logic (list 'quote str) 'state)
(list 'value ''(value-triple nil)))
x))
(defun verify-termination-fn (lst state)
(when-logic3
; We originally used when-logic here so that we would skip verify-termination during
; pass1 of the boot-strap in axioms.lisp. Now we use
; verify-termination-boot-strap for that purpose, but we continue the same
; convention, since by now users might rely on it.
; We could always return a defuns form, but the user may find it more pleasing
; to see a defun when there is a single definition, so we return a defun form
; in that case.
"VERIFY-TERMINATION"
(er-let*
((verify-termination-defs-lst (verify-termination1 lst state)))
(value (cond ((null verify-termination-defs-lst)
'(value-triple :redundant))
((null (cdr verify-termination-defs-lst))
(cons 'defun (car verify-termination-defs-lst)))
(t
(cons 'defuns verify-termination-defs-lst)))))))
; When we defined instantiablep we included the comment that a certain
; invariant holds between it and the axioms. The functions here are
; not used in the system but can be used to check that invariant.
; They were not defined earlier because they use event tuples.
(defun fns-used-in-axioms (lst wrld ans)
; Intended for use only by check-out-instantiablep.
(cond ((null lst) ans)
((and (eq (caar lst) 'event-landmark)
(eq (cadar lst) 'global-value)
(eq (access-event-tuple-type (cddar lst)) 'defaxiom))
; In this case, (car lst) is a tuple of the form
; (event-landmark global-value . tuple)
; where tuple is a defaxiom of some name, namex, and we are interested
; in all the function symbols occurring in the formula named namex.
(fns-used-in-axioms (cdr lst)
wrld
(all-ffn-symbs (formula
(access-event-tuple-namex
(cddar lst))
nil
wrld)
ans)))
(t (fns-used-in-axioms (cdr lst) wrld ans))))
(defun check-out-instantiablep1 (fns wrld)
; Intended for use only by check-out-instantiablep.
(cond ((null fns) nil)
((instantiablep (car fns) wrld)
(cons (car fns) (check-out-instantiablep1 (cdr fns) wrld)))
(t (check-out-instantiablep1 (cdr fns) wrld))))
(defun check-out-instantiablep (wrld)
; See the comment in instantiablep.
(let ((bad (check-out-instantiablep1 (fns-used-in-axioms wrld wrld nil)
wrld)))
(cond
((null bad) "Everything checks")
(t (er hard 'check-out-instantiablep
"The following functions are instantiable and shouldn't be:~%~x0"
bad)))))
; This is an odd place for the following event. But it needs to be in a file
; that is processed in the boot-strap before interface-raw.lisp, so a more
; natural place -- apply-prim.lisp -- won't work.
(defconst *avoid-oneify-fns*
; Each of these functions has a *1* function (defined in oneify-cltl-code) that
; simply calls it directly, in particular without any guard checks or any
; labels functions. Those missing labels functions normally, when
; guard-checking is off, lead to calls of primitives that check for the live
; state. Because of the direct calls without labels functions, these functions
; can cause a raw Lisp error when passing other than a live state as an actual
; that should be a state (as per the stobjs-in). So, calling these indirectly
; via apply$ has that problem as well; hence we put these all into
; *blacklisted-apply$-fns*, in addition to giving them special treatment in
; oneify-cltl-code. Without putting these into *blacklisted-apply$-fns* we
; were able to get a raw Lisp error as follows.
; (include-book "projects/apply/top" :dir :system)
; (defbadge set-body-fn)
; (apply$ 'set-body-fn (list 3 4 5 6)) ; raw Lisp error
'(thm-fn
make-event-fn
certify-book-fn
; Keep the following in sync with primitive-event-macros.
defun-fn
;; #+:non-standard-analysis
;; defun-std ; defun-fn
defuns-fn ; mutual-recursion
;; defuns ; calls defuns-fn, above
defthm-fn
;; #+:non-standard-analysis
;; defthm-std ; calls defthm-fn, above
defaxiom-fn
defconst-fn
defstobj-fn defabsstobj-fn
defpkg-fn
deflabel-fn
deftheory-fn
defchoose-fn
verify-guards-fn
defmacro-fn
in-theory-fn
in-arithmetic-theory-fn
regenerate-tau-database-fn
push-untouchable-fn
remove-untouchable-fn
reset-prehistory-fn
set-body-fn
table-fn
progn-fn
encapsulate-fn
include-book-fn
change-include-book-dir
comp-fn
verify-termination-fn
verify-termination-boot-strap-fn
;; add-match-free-override ; should be fast enough
; Theory-invariant is included in *macros-for-nonexpansion-in-raw-lisp*. The
; remaining members of primitive-event-macros, after theory-invariant, are
; handled well enough already since we included table-fn above.
))
|