qpdf-manual.xml 334 KB
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
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE book [
<!ENTITY ldquo "&#x201C;">
<!ENTITY rdquo "&#x201D;">
<!ENTITY mdash "&#x2014;">
<!ENTITY ndash "&#x2013;">
<!ENTITY nbsp "&#xA0;">
<!ENTITY swversion "9.1.1">
<!ENTITY lastreleased "January 26, 2020">
]>
<book>
 <bookinfo>
  <title>QPDF Manual</title>
  <subtitle>For QPDF Version &swversion;, &lastreleased;</subtitle>
  <author>
   <firstname>Jay</firstname><surname>Berkenbilt</surname>
  </author>
  <copyright>
   <year>2005&ndash;2020</year>
   <holder>Jay Berkenbilt</holder>
  </copyright>
 </bookinfo>
 <preface id="acknowledgments">
  <title>General Information</title>
  <para>
   QPDF is a program that does structural, content-preserving
   transformations on PDF files.  QPDF's website is located at <ulink
   url="http://qpdf.sourceforge.net/">http://qpdf.sourceforge.net/</ulink>.
   QPDF's source code is hosted on github at <ulink
   url="https://github.com/qpdf/qpdf">https://github.com/qpdf/qpdf</ulink>.
  </para>
  <para>
   QPDF is licensed under <ulink
   url="http://www.apache.org/licenses/LICENSE-2.0">the Apache
   License, Version 2.0</ulink> (the "License"). Unless required by
   applicable law or agreed to in writing, software distributed under
   the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES
   OR CONDITIONS OF ANY KIND, either express or implied. See the
   License for the specific language governing permissions and
   limitations under the License.
  </para>
  <para>
   Versions of qpdf prior to version 7 were released under the terms
   of <ulink url="https://opensource.org/licenses/Artistic-2.0">the
   Artistic License, version 2.0</ulink>. At your option, you may
   continue to consider qpdf to be licensed under those terms. The
   Apache License 2.0 permits everything that the Artistic License 2.0
   permits but is slightly less restrictive. Allowing the Artistic
   License to continue being used is primary to help people who may
   have to get specific approval to use qpdf in their products.
  </para>
  <para>
   QPDF is intentionally released with a permissive license. However,
   if there is some reason that the licensing terms don't work for
   your requirements, please feel free to contact the copyright holder
   to make other arrangements.
  </para>
  <para>
   QPDF was originally created in 2001 and modified periodically
   between 2001 and 2005 during my employment at <ulink
   url="http://www.apexcovantage.com">Apex CoVantage</ulink>.  Upon my
   departure from Apex, the company graciously allowed me to take
   ownership of the software and continue maintaining as an open
   source project, a decision for which I am very grateful.  I have
   made considerable enhancements to it since that time.  I feel
   fortunate to have worked for people who would make such a decision.
   This work would not have been possible without their support.
  </para>
 </preface>
 <chapter id="ref.overview">
  <title>What is QPDF?</title>
  <para>
   QPDF is a program that does structural, content-preserving
   transformations on PDF files.  It could have been called something
   like <emphasis>pdf-to-pdf</emphasis>.  It also provides many useful
   capabilities to developers of PDF-producing software or for people
   who just want to look at the innards of a PDF file to learn more
   about how they work.
  </para>
  <para>
   With QPDF, it is possible to copy objects from one PDF file into
   another and to manipulate the list of pages in a PDF file.  This
   makes it possible to merge and split PDF files.  The QPDF library
   also makes it possible for you to create PDF files from scratch.
   In this mode, you are responsible for supplying all the contents of
   the file, while the QPDF library takes care off all the syntactical
   representation of the objects, creation of cross references tables
   and, if you use them, object streams, encryption, linearization,
   and other syntactic details.  You are still responsible for
   generating PDF content on your own.
  </para>
  <para>
   QPDF has been designed with very few external dependencies, and it
   is intentionally very lightweight.  QPDF is
   <emphasis>not</emphasis> a PDF content creation library, a PDF
   viewer, or a program capable of converting PDF into other formats.
   In particular, QPDF knows nothing about the semantics of PDF
   content streams.  If you are looking for something that can do
   that, you should look elsewhere.  However, once you have a valid
   PDF file, QPDF can be used to transform that file in ways perhaps
   your original PDF creation can't handle.  For example, many
   programs generate simple PDF files but can't password-protect them,
   web-optimize them, or perform other transformations of that type.
  </para>
 </chapter>
 <chapter id="ref.installing">
  <title>Building and Installing QPDF</title>
  <para>
   This chapter describes how to build and install qpdf.  Please see
   also the <filename>README.md</filename> and
   <filename>INSTALL</filename> files in the source distribution.
  </para>
  <sect1 id="ref.prerequisites">
   <title>System Requirements</title>
   <para>
    The qpdf package has few external dependencies. In order to build
    qpdf, the following packages are required:
    <itemizedlist>
     <listitem>
      <para>
       A C++ compiler that supports C++-11.
      </para>
     </listitem>
     <listitem>
      <para>
       zlib: <ulink url="http://www.zlib.net/">http://www.zlib.net/</ulink>
      </para>
     </listitem>
     <listitem>
      <para>
       jpeg: <ulink
       url="http://www.ijg.org/files/">http://www.ijg.org/files/</ulink>
       or <ulink
       url="https://libjpeg-turbo.org/">https://libjpeg-turbo.org/</ulink>
      </para>
     </listitem>
     <listitem>
      <para>
       <emphasis>Recommended but not required:</emphasis> gnutls:
       <ulink url="https://www.gnutls.org/">https://www.gnutls.org/</ulink>
       to be able to use the gnutls crypto provider
      </para>
     </listitem>
     <listitem>
      <para>
       gnu make 3.81 or newer: <ulink url="http://www.gnu.org/software/make">http://www.gnu.org/software/make</ulink>
      </para>
     </listitem>
     <listitem>
      <para>
       perl version 5.8 or newer:
       <ulink url="http://www.perl.org/">http://www.perl.org/</ulink>;
       required for running the test suite. Starting with qpdf version
       9.1.1, perl is no longer required at runtime.
      </para>
     </listitem>
     <listitem>
      <para>
       GNU diffutils (any version): <ulink
       url="http://www.gnu.org/software/diffutils/">http://www.gnu.org/software/diffutils/</ulink>
       is required to run the test suite.  Note that this is the
       version of diff present on virtually all GNU/Linux systems.
       This is required because the test suite uses <command>diff
       -u</command>.
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Part of qpdf's test suite does comparisons of the contents PDF
    files by converting them images and comparing the images.  The
    image comparison tests are disabled by default.  Those tests are
    not required for determining correctness of a qpdf build if you
    have not modified the code since the test suite also contains
    expected output files that are compared literally.  The image
    comparison tests provide an extra check to make sure that any
    content transformations don't break the rendering of pages.
    Transformations that affect the content streams themselves are off
    by default and are only provided to help developers look into the
    contents of PDF files.  If you are making deep changes to the
    library that cause changes in the contents of the files that qpdf
    generates, then you should enable the image comparison tests.
    Enable them by running <command>configure</command> with the
    <option>--enable-test-compare-images</option> flag.  If you enable
    this, the following additional requirements are required by the
    test suite.  Note that in no case are these items required to use
    qpdf.
    <itemizedlist>
     <listitem>
      <para>
       libtiff: <ulink url="http://www.remotesensing.org/libtiff/">http://www.remotesensing.org/libtiff/</ulink>
      </para>
     </listitem>
     <listitem>
      <para>
       GhostScript version 8.60 or newer: <ulink
       url="http://www.ghostscript.com">http://www.ghostscript.com</ulink>
      </para>
     </listitem>
    </itemizedlist>
    If you do not enable this, then you do not need to have tiff and
    ghostscript.
   </para>
   <para>
    Pre-built documentation is distributed with qpdf, so you should
    generally not need to rebuild the documentation.  In order to
    build the documentation from its docbook sources, you need the
    docbook XML style sheets (<ulink
    url="http://downloads.sourceforge.net/docbook/">http://downloads.sourceforge.net/docbook/</ulink>).
    To build the PDF version of the documentation, you need Apache fop
    (<ulink
    url="http://xml.apache.org/fop/">http://xml.apache.org/fop/</ulink>)
    version 0.94 or higher.
   </para>
  </sect1>
  <sect1 id="ref.building">
   <title>Build Instructions</title>
   <para>
    Building qpdf on UNIX is generally just a matter of running

    <programlisting>./configure
make
</programlisting>
    You can also run <command>make check</command> to run the test
    suite and <command>make install</command> to install.  Please run
    <command>./configure --help</command> for options on what can be
    configured.  You can also set the value of
    <varname>DESTDIR</varname> during installation to install to a
    temporary location, as is common with many open source packages.
    Please see also the <filename>README.md</filename> and
    <filename>INSTALL</filename> files in the source distribution.
   </para>
   <para>
    Building on Windows is a little bit more complicated.  For
    details, please see <filename>README-windows.md</filename> in the
    source distribution.  You can also download a binary distribution
    for Windows.  There is a port of qpdf to Visual C++ version 6 in
    the <filename>contrib</filename> area generously contributed by
    Jian Ma.  This is also discussed in more detail in
    <filename>README-windows.md</filename>.
   </para>
   <para>
    There are some other things you can do with the build.  Although
    qpdf uses <application>autoconf</application>, it does not use
    <application>automake</application> but instead uses a
    hand-crafted non-recursive Makefile that requires gnu make.  If
    you're really interested, please read the comments in the
    top-level <filename>Makefile</filename>.
   </para>
  </sect1>
  <sect1 id="ref.crypto">
   <title>Crypto Providers</title>
   <para>
    Starting with qpdf 9.1.0, the qpdf library can be built with
    multiple implementations of providers of cryptographic functions,
    which we refer to as &ldquo;crypto providers.&rdquo; At the time
    of writing, a crypto implementation must provide MD5 and SHA2
    (256, 384, and 512-bit) hashes and RC4 and AES256 with and without
    CBC encryption. In the future, if digital signature is added to
    qpdf, there may be additional requirements beyond this.
   </para>
   <para>
    Starting with qpdf version 9.1.0, the available implementations
    are <literal>native</literal> and <literal>gnutls</literal>.
    Additional implementations may be added if needed. It is also
    possible for a developer to provide their own implementation
    without modifying the qpdf library.
   </para>
   <sect2 id="ref.crypto.build">
    <title>Build Support For Crypto Providers</title>
    <para>
     When building with qpdf's build system, crypto providers can be
     enabled at build time using various
     <command>./configure</command> options. The default behavior is
     for <command>./configure</command> to discover which crypto
     providers can be supported based on available external libraries,
     to build all available crypto providers, and to use an external
     provider as the default over the native one. This behavior can be
     changed with the following flags to
     <command>./configure</command>:
     <itemizedlist>
      <listitem>
       <para>
        <option>--enable-crypto-<replaceable>x</replaceable></option>
        (where <replaceable>x</replaceable> is a supported crypto
        provider): enable the <replaceable>x</replaceable> crypto
        provider, requiring any external dependencies it needs
       </para>
      </listitem>
      <listitem>
       <para>
        <option>--disable-crypto-<replaceable>x</replaceable></option>:
        disable the <replaceable>x</replaceable> provider, and do not
        link against its dependencies even if they are available
       </para>
      </listitem>
      <listitem>
       <para>
        <option>--with-default-crypto=<replaceable>x</replaceable></option>:
        make <replaceable>x</replaceable> the default provider even if
        a higher priority one is available
       </para>
      </listitem>
      <listitem>
       <para>
        <option>--disable-implicit-crypto</option>: only build crypto
        providers that are explicitly requested with an
        <option>--enable-crypto-<replaceable>x</replaceable></option>
        option
       </para>
      </listitem>
     </itemizedlist>
    </para>
    <para>
     For example, if you want to guarantee that the gnutls crypto
     provider is used and that the native provider is not built, you
     could run <command>./configure --enable-crypto-gnutls
     --disable-implicit-crypto</command>.
    </para>
    <para>
     If you build qpdf using your own build system, in order for qpdf
     to work at all, you need to enable at least one crypto provider.
     The file <filename>libqpdf/qpdf/qpdf-config.h.in</filename>
     provides macros <literal>DEFAULT_CRYPTO</literal>, whose value
     must be a string naming the default crypto provider, and various
     symbols starting with <literal>USE_CRYPTO_</literal>, at least
     one of which has to be enabled. Additionally, you must compile
     the source files that implement a crypto provider. To get a list
     of those files, look at <filename>libqpdf/build.mk</filename>. If
     you want to omit a particular crypto provider, as long as its
     <literal>USE_CRYPTO_</literal> symbol is undefined, you can
     completely ignore the source files that belong to a particular
     crypto provider. Additionally, crypto providers may have their
     own external dependencies that can be omitted if the crypto
     provider is not used. For example, if you are building qpdf
     yourself and are using an environment that does not support
     gnutls, you can ensure that <literal>USE_CRYPTO_NATIVE</literal>
     is defined, <literal>USE_CRYPTO_GNUTLS</literal> is not defined,
     and <literal>DEFAULT_CRYPTO</literal> is defined to
     <literal>"native"</literal>. Then you must include the source
     files used in the native implementation, some of which were added
     or renamed from earlier versions, to your build, and you can
     ignore <filename>QPDFCrypto_gnutls.cc</filename>. Always consult
     <filename>libqpdf/build.mk</filename> to get the list of source
     files you need to build.
    </para>
   </sect2>
   <sect2 id="ref.crypto.runtime">
    <title>Runtime Crypto Provider Selection</title>
    <para>
     You can use the <option>--show-crypto</option> option to
     <command>qpdf</command> to get a list of available crypto
     providers. The default provider is always listed first, and the
     rest are listed in lexical order. Each crypto provider is listed
     on a line by itself with no other text, enabling the output of
     this command to be used easily in scripts.
    </para>
    <para>
     You can override which crypto provider is used by setting the
     <literal>QPDF_CRYPTO_PROVIDER</literal> environment variable.
     There are few reasons to ever do this, but you might want to do
     it if you were explicitly trying to compare behavior of two
     different crypto providers while testing performance or
     reproducing a bug. It could also be useful for people who are
     implementing their own crypto providers.
    </para>
   </sect2>
   <sect2 id="ref.crypto.develop">
    <title>Crypto Provider Information for Developers</title>
    <para>
     If you are writing code that uses libqpdf and you want to force a
     certain crypto provider to be used, you can call the method
     <function>QPDFCryptoProvider::setDefaultProvider</function>. The
     argument is the name of a built-in or developer-supplied
     provider. To add your own crypto provider, you have to create a
     class derived from <classname>QPDFCryptoImpl</classname> and
     register it with <classname>QPDFCryptoProvider</classname>. For
     additional information, see comments in
     <filename>include/qpdf/QPDFCryptoImpl.hh</filename>.
    </para>
   </sect2>
   <sect2 id="ref.crypto.design">
    <title>Crypto Provider Design Notes</title>
    <para>
     This section describes a few bits of rationale for why the crypto
     provider interface was set up the way it was. You don't need to
     know any of this information, but it's provided for the record
     and in case it's interesting.
    </para>
    <para>
     As a general rule, I want to avoid as much as possible including
     large blocks of code that are conditionally compiled such that,
     in most builds, some code is never built. This is dangerous
     because it makes it very easy for invalid code to creep in
     unnoticed. As such, I want it to be possible to build qpdf with
     all available crypto providers, and this is the way I build qpdf
     for local development. At the same time, if a particular packager
     feels that it is a security liability for qpdf to use crypto
     functionality from other than a library that gets considerable
     scrutiny for this specific purpose (such as gnutls, openssl, or
     nettle), then I want to give that packager the ability to
     completely disable qpdf's native implementation. Or if someone
     wants to avoid adding a dependency on one of the external crypto
     providers, I don't want the availability of the provider to
     impose additional external dependencies within that environment.
     Both of these are situations that I know to be true for some
     users of qpdf.
    </para>
    <para>
     I want registration and selection of crypto providers to be
     thread-safe, and I want it to work deterministically for a
     developer to provide their own crypto provider and be able to set
     it up as the default. This was the primary motivation behind
     requiring C++-11 as doing so enabled me to exploit the guaranteed
     thread safety of local block static initialization. The
     <classname>QPDFCryptoProvider</classname> class uses a singleton
     pattern with thread-safe initialization to create the singleton
     instance of <classname>QPDFCryptoProvider</classname> and exposes
     only static methods in its public interface. In this way, if a
     developer wants to call any
     <classname>QPDFCryptoProvider</classname> methods, the library
     guarantees the <classname>QPDFCryptoProvider</classname> is fully
     initialized and all built-in crypto providers are registered.
     Making <classname>QPDFCryptoProvider</classname> actually know
     about all the built-in providers may seem a bit sad at first, but
     this choice makes it extremely clear exactly what the
     initialization behavior is. There's no question about provider
     implementations automatically registering themselves in a
     nondeterministic order. It also means that implementations do not
     need to know anything about the provider interface, which makes
     them easier to test in isolation. Another advantage of this
     approach is that a developer who wants to develop their own
     crypto provider can do so in complete isolation from the qpdf
     library and, with just two calls, can make qpdf use their
     provider in their application. If they decided to contribute
     their code, plugging it into the qpdf library would require a
     very small change to qpdf's source code.
    </para>
    <para>
     The decision to make the crypto provider selectable at runtime
     was one I struggled with a little, but I decided to do it for
     various reasons. Allowing an end user to switch crypto providers
     easily could be very useful for reproducing a potential bug. If a
     user reports a bug that some cryptographic thing is broken, I can
     easily ask that person to try with the
     <literal>QPDF_CRYPTO_PROVIDER</literal> variable set to different
     values. The same could apply in the event of a performance
     problem. This also makes it easier for qpdf's own test suite to
     exercise code with different providers without having to make
     every program that links with qpdf aware of the possibility of
     multiple providers. In qpdf's continuous integration environment,
     the entire test suite is run for each supported crypto provider.
     This is made simple by being able to select the provider using an
     environment variable.
    </para>
    <para>
     Finally, making crypto providers selectable in this way establish
     a pattern that I may follow again in the future for stream filter
     providers. One could imagine a future enhancement where someone
     could provide their own implementations for basic filters like
     <literal>/FlateDecode</literal> or for other filters that qpdf
     doesn't support. Implementing the registration functions and
     internal storage of registered providers was also easier using
     C++-11's functional interfaces, which was another reason to
     require C++-11 at this time.
    </para>
   </sect2>
  </sect1>
  <sect1 id="ref.packaging">
   <title>Notes for Packagers</title>
   <para>
    If you are packaging qpdf for an operating system distribution,
    here are some things you may want to keep in mind:
    <itemizedlist>
     <listitem>
      <para>
       Starting in qpdf version 9.1.1, qpdf no longer has a runtime
       dependency on perl. This is because fix-qdf was rewritten in
       C++. However, qpdf still has a build-time dependency on perl.
      </para>
     </listitem>
     <listitem>
      <para>
       Make sure you are getting the intended behavior with regard to
       crypto providers. Read <xref linkend="ref.crypto.build"/> for
       details.
      </para>
     </listitem>
     <listitem>
      <para>
       Passing <option>--enable-show-failed-test-output</option> to
       <command>./configure</command> will cause any failed test
       output to be written to the console. This can be very useful
       for seeing test failures generated by autobuilders where you
       can't access qtest.log after the fact.
      </para>
     </listitem>
     <listitem>
      <para>
       If qpdf's build environment detects the presence of autoconf
       and related tools, it will check to ensure that automatically
       generated files are up-to-date with recorded checksums and fail
       if it detects a discrepancy. This feature is intended to
       prevent you from accidentally forgetting to regenerate
       automatic files after modifying their sources. If your
       packaging environment automatically refreshes automatic files,
       it can cause this check to fail. Suppress qpdf's checks by
       passing <option>--disable-check-autofiles</option> to
       <command>/.configure</command>. This is safe since qpdf's
       <command>autogen.sh</command> just runs autotools in the normal
       way.
      </para>
     </listitem>
     <listitem>
      <para>
       QPDF's <command>make install</command> does not install
       completion files by default, but as a packager, it's good if
       you install them wherever your distribution expects such files
       to go. You can find completion files to install in the
       <filename>completions</filename> directory.
      </para>
     </listitem>
     <listitem>
      <para>
       Packagers are encouraged to install the source files from the
       <filename>examples</filename> directory along with qpdf
       development packages.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.using">
  <title>Running QPDF</title>
  <para>
   This chapter describes how to run the qpdf program from the command
   line.
  </para>
  <sect1 id="ref.invocation">
   <title>Basic Invocation</title>
   <para>
    When running qpdf, the basic invocation is as follows:

    <programlisting><command>qpdf</command><option> [ <replaceable>options</replaceable> ] <replaceable>infilename</replaceable> [ <replaceable>outfilename</replaceable> ]</option>
</programlisting>
    This converts PDF file <option>infilename</option> to PDF file
    <option>outfilename</option>.  The output file is functionally
    identical to the input file but may have been structurally
    reorganized.  Also, orphaned objects will be removed from the
    file.  Many transformations are available as controlled by the
    options below.  In place of <option>infilename</option>, the
    parameter <option>--empty</option> may be specified.  This causes
    qpdf to use a dummy input file that contains zero pages.  The only
    normal use case for using <option>--empty</option> would be if you
    were going to add pages from another source, as discussed in <xref
    linkend="ref.page-selection"/>.
   </para>
   <para>
    If <option>@filename</option> appears anywhere in the
    command-line, it will be read line by line, and each line will be
    treated as a command-line argument. The <option>@-</option> option
    allows arguments to be read from standard input. This allows qpdf
    to be invoked with an arbitrary number of arbitrarily long
    arguments. It is also very useful for avoiding having to pass
    passwords on the command line.
   </para>
   <para>
    <option>outfilename</option> does not have to be seekable, even
    when generating linearized files.  Specifying
    &ldquo;<option>-</option>&rdquo; as <option>outfilename</option>
    means to write to standard output. If you want to overwrite the
    input file with the output, use the option
    <option>--replace-input</option> and omit the output file name.
    You can't specify the same file as both the input and the output.
    If you do this, qpdf will tell you about the
    <option>--replace-input</option> option.
   </para>
   <para>
    Most options require an output file, but some testing or
    inspection commands do not.  These are specifically noted.
   </para>
  </sect1>
  <sect1 id="ref.shell-completion">
   <title>Shell Completion</title>
   <para>
    Starting in qpdf version 8.3.0, qpdf provides its own completion
    support for zsh and bash. You can enable bash completion with
    <command>eval $(qpdf --completion-bash)</command> and zsh
    completion with <command>eval $(qpdf --completion-zsh)</command>.
    If <command>qpdf</command> is not in your path, you should invoke
    it above with an absolute path. If you invoke it with a relative
    path, it will warn you, and the completion won't work if you're in
    a different directory.
   </para>
   <para>
    qpdf will use <literal>argv[0]</literal> to figure out where its
    executable is. This may produce unwanted results in some cases,
    especially if you are trying to use completion with copy of qpdf
    that is built from source. You can specify a full path to the qpdf
    you want to use for completion in the
    <literal>QPDF_EXECUTABLE</literal> environment variable.
   </para>
  </sect1>
  <sect1 id="ref.basic-options">
   <title>Basic Options</title>
   <para>
    The following options are the most common ones and perform
    commonly needed transformations.
    <variablelist>
     <varlistentry>
      <term><option>--help</option></term>
      <listitem>
       <para>
        Display command-line invocation help.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--version</option></term>
      <listitem>
       <para>
        Display the current version of qpdf.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--copyright</option></term>
      <listitem>
       <para>
        Show detailed copyright information.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-crypto</option></term>
      <listitem>
       <para>
        Show a list of available crypto providers, each on a line by
        itself. The default provider is always listed first. See <xref
        linkend="ref.crypto"/> for more information about crypto
        providers.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--completion-bash</option></term>
      <listitem>
       <para>
        Output a completion command you can eval to enable shell
        completion from bash.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--completion-zsh</option></term>
      <listitem>
       <para>
        Output a completion command you can eval to enable shell
        completion from zsh.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--password=password</option></term>
      <listitem>
       <para>
        Specifies a password for accessing encrypted files. Note that
        you can use <option>@filename</option> or <option>@-</option>
        as described above to put the password in a file or pass it
        via standard input so you can avoid specifying it on the
        command line.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--is-encrypted</option></term>
      <listitem>
       <para>
        Silently exit with status 0 if the file is encrypted or status
        2 if the file is not encrypted. This is useful for shell
        scripts. Other options are ignored if this is given. This
        option is mutually exclusive with
        <option>--requires-password</option>. Both this option and
        <option>--requires-password</option> exit with status 2 for
        non-encrypted files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--requires-password</option></term>
      <listitem>
       <para>
        Silently exit with status 0 if a password (other than as
        supplied) is required. Exit with status 2 if the file is not
        encrypted. Exit with status 3 if the file is encrypted but
        requires no password or the correct password has been
        supplied. This is useful for shell scripts. Note that any
        supplied password is used when opening the file. When used
        with a <option>--password</option> option, this option can be
        used to check the correctness of the password. In that case,
        an exit status of 3 means the file works with the supplied
        password. This option is mutually exclusive with
        <option>--is-encrypted</option>. Both this option and
        <option>--is-encrypted</option> exit with status 2 for
        non-encrypted files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--verbose</option></term>
      <listitem>
       <para>
        Increase verbosity of output. For now, this just prints some
        indication of any file that it creates.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--progress</option></term>
      <listitem>
       <para>
        Indicate progress while writing files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--no-warn</option></term>
      <listitem>
       <para>
        Suppress writing of warnings to stderr. If warnings were
        detected and suppressed, <command>qpdf</command> will still
        exit with exit code 3.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--linearize</option></term>
      <listitem>
       <para>
        Causes generation of a linearized (web-optimized) output file.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--replace-input</option></term>
      <listitem>
       <para>
        If specified, the output file name should be omitted. This
        option tells qpdf to replace the input file with the output.
        It does this by writing to
        <filename><replaceable>infilename</replaceable>.~qpdf-temp#</filename>
        and, when done, overwriting the input file with the temporary
        file. If there were any warnings, the original input is saved
        as
        <filename><replaceable>infilename</replaceable>.~qpdf-orig</filename>.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--copy-encryption=file</option></term>
      <listitem>
       <para>
        Encrypt the file using the same encryption parameters,
        including user and owner password, as the specified file.  Use
        <option>--encrypt-file-password</option> to specify a password
        if one is needed to open this file.  Note that copying the
        encryption parameters from a file also copies the first half
        of <literal>/ID</literal> from the file since this is part of
        the encryption parameters.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--encrypt-file-password=password</option></term>
      <listitem>
       <para>
        If the file specified with <option>--copy-encryption</option>
        requires a password, specify the password using this option.
        Note that only one of the user or owner password is required.
        Both passwords will be preserved since QPDF does not
        distinguish between the two passwords.  It is possible to
        preserve encryption parameters, including the owner password,
        from a file even if you don't know the file's owner password.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--encrypt options --</option></term>
      <listitem>
       <para>
        Causes generation an encrypted output file.  Please see <xref
        linkend="ref.encryption-options"/> for details on how to
        specify encryption parameters.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--decrypt</option></term>
      <listitem>
       <para>
        Removes any encryption on the file.  A password must be
        supplied if the file is password protected.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--password-is-hex-key</option></term>
      <listitem>
       <para>
        Overrides the usual computation/retrieval of the PDF file's
        encryption key from user/owner password with an explicit
        specification of the encryption key. When this option is
        specified, the argument to the <option>--password</option>
        option is interpreted as a hexadecimal-encoded key value. This
        only applies to the password used to open the main input file.
        It does not apply to other files opened by
        <option>--pages</option> or other options or to files being
        written.
       </para>
       <para>
        Most users will never have a need for this option, and no
        standard viewers support this mode of operation, but it can be
        useful for forensic or investigatory purposes. For example, if
        a PDF file is encrypted with an unknown password, a
        brute-force attack using the key directly is sometimes more
        efficient than one using the password. Also, if a file is
        heavily damaged, it may be possible to derive the encryption
        key and recover parts of the file using it directly. To expose
        the encryption key used by an encrypted file that you can open
        normally, use the <option>--show-encryption-key</option>
        option.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--suppress-password-recovery</option></term>
      <listitem>
       <para>
        Ordinarily, qpdf attempts to automatically compensate for
        passwords specified in the wrong character encoding. This
        option suppresses that behavior. Under normal conditions,
        there are no reasons to use this option. See <xref
        linkend="ref.unicode-passwords"/> for a discussion
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--password-mode=<replaceable>mode</replaceable></option></term>
      <listitem>
       <para>
        This option can be used to fine-tune how qpdf interprets
        Unicode (non-ASCII) password strings passed on the command
        line. With the exception of the <option>hex-bytes</option>
        mode, these only apply to passwords provided when encrypting
        files. The <option>hex-bytes</option> mode also applies to
        passwords specified for reading files. For additional
        discussion of the supported password modes and when you might
        want to use them, see <xref linkend="ref.unicode-passwords"/>.
        The following modes are supported:
        <itemizedlist>
         <listitem>
          <para>
           <option>auto</option>: Automatically determine whether the
           specified password is a properly encoded Unicode (UTF-8)
           string, and transcode it as required by the PDF spec based
           on the type encryption being applied. On Windows starting
           with version 8.4.0, and on almost all other modern
           platforms, incoming passwords will be properly encoded in
           UTF-8, so this is almost always what you want.
          </para>
         </listitem>
         <listitem>
          <para>
           <option>unicode</option>: Tells qpdf that the incoming
           password is UTF-8, overriding whatever its automatic
           detection determines. The only difference between this mode
           and <option>auto</option> is that qpdf will fail with an
           error message if the password is not valid UTF-8 instead of
           falling back to <option>bytes</option> mode with a warning.
          </para>
         </listitem>
         <listitem>
          <para>
           <option>bytes</option>: Interpret the password as a literal
           byte string. For non-Windows platforms, this is what
           versions of qpdf prior to 8.4.0 did. For Windows platforms,
           there is no way to specify strings of binary data on the
           command line directly, but you can use the
           <option>@filename</option> option to do it, in which case
           this option forces qpdf to respect the string of bytes as
           provided. This option will allow you to encrypt PDF files
           with passwords that will not be usable by other readers.
          </para>
         </listitem>
         <listitem>
          <para>
           <option>hex-bytes</option>: Interpret the password as a
           hex-encoded string. This provides a way to pass binary data
           as a password on all platforms including Windows. As with
           <option>bytes</option>, this option may allow creation of
           files that can't be opened by other readers. This mode
           affects qpdf's interpretation of passwords specified for
           decrypting files as well as for encrypting them. It makes
           it possible to specify strings that are encoded in some
           manner other than the system's default encoding.
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--rotate=[+|-]angle[:page-range]</option></term>
      <listitem>
       <para>
        Apply rotation to specified pages. The
        <option>page-range</option> portion of the option value has
        the same format as page ranges in <xref
        linkend="ref.page-selection"/>. If the page range is omitted,
        the rotation is applied to all pages. The
        <option>angle</option> portion of the parameter may be either
        90, 180, or 270. If preceded by <option>+</option> or
        <option>-</option>, the angle is added to or subtracted from
        the specified pages' original rotations. Otherwise the pages'
        rotations are set to the exact value. For example, the command
        <command>qpdf in.pdf out.pdf --rotate=+90:2,4,6
        --rotate=180:7-8</command> would rotate pages 2, 4, and 6 90
        degrees clockwise from their original rotation and force the
        rotation of pages 7 through 9 to 180 degrees regardless of
        their original rotation, and the command <command>qpdf in.pdf
        out.pdf --rotate=180</command> would rotate all pages by 180
        degrees.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--keep-files-open=<replaceable>[yn]</replaceable></option></term>
      <listitem>
       <para>
        This option controls whether qpdf keeps individual files open
        while merging. Prior to version 8.1.0, qpdf always kept all
        files open, but this meant that the number of files that could
        be merged was limited by the operating system's open file
        limit. Version 8.1.0 opened files as they were referenced and
        closed them after each read, but this caused a major
        performance impact. Version 8.2.0 optimized the performance
        but did so in a way that, for local file systems, there was a
        small but unavoidable performance hit, but for networked file
        systems, the performance impact could be very high. Starting
        with version 8.2.1, the default behavior is that files are
        kept open if no more than 200 files are specified, but that
        the behavior can be explicitly overridden with the
        <option>--keep-files-open</option> flag. If you are merging
        more than 200 files but less than the operating system's max
        open files limit, you may want to use
        <option>--keep-files-open=y</option>, especially if working
        over a networked file system. If you are using a local file
        system where the overhead is low and you might sometimes merge
        more than the OS limit's number of files from a script and are
        not worried about a few seconds additional processing time,
        you may want to specify <option>--keep-files-open=n</option>.
        The threshold for switching may be changed from the default
        200 with the <option>--keep-files-open-threshold</option>
        option.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--keep-files-open-threshold=<replaceable>count</replaceable></option></term>
      <listitem>
       <para>
        If specified, overrides the default value of 200 used as the
        threshold for qpdf deciding whether or not to keep files open.
        See <option>--keep-files-open</option> for details.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--pages options --</option></term>
      <listitem>
       <para>
        Select specific pages from one or more input files.  See <xref
        linkend="ref.page-selection"/> for details on how to do page
        selection (splitting and merging).
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--collate</option></term>
      <listitem>
       <para>
        When specified, collate rather than concatenate pages from
        files specified with <option>--pages</option>.  See <xref
        linkend="ref.page-selection"/> for additional details.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--split-pages=[n]</option></term>
      <listitem>
       <para>
        Write each group of <option>n</option> pages to a separate
        output file. If <option>n</option> is not specified, create
        single pages. Output file names are generated as follows:
        <itemizedlist>
         <listitem>
          <para>
           If the string <literal>%d</literal> appears in the output
           file name, it is replaced with a range of zero-padded page
           numbers starting from 1.
          </para>
         </listitem>
         <listitem>
          <para>
           Otherwise, if the output file name ends in
           <filename>.pdf</filename> (case insensitive), a zero-padded
           page range, preceded by a dash, is inserted before the file
           extension.
          </para>
         </listitem>
         <listitem>
          <para>
           Otherwise, the file name is appended with a zero-padded
           page range preceded by a dash.
          </para>
         </listitem>
        </itemizedlist>
       </para>
       <para>
        Page ranges are a single number in the case of single-page
        groups or two numbers separated by a dash otherwise.
        For example, if <filename>infile.pdf</filename> has 12 pages
        <itemizedlist>
         <listitem>
          <para>
           <command>qpdf --split-pages infile.pdf %d-out</command>
           would generate files <filename>01-out</filename> through
           <filename>12-out</filename>
          </para>
         </listitem>
         <listitem>
          <para>
           <command>qpdf --split-pages=2 infile.pdf
           outfile.pdf</command> would generate files
           <filename>outfile-01-02.pdf</filename> through
           <filename>outfile-11-12.pdf</filename>
          </para>
         </listitem>
         <listitem>
          <para>
           <command>qpdf --split-pages infile.pdf
           something.else</command> would generate files
           <filename>something.else-01</filename> through
           <filename>something.else-12</filename>
          </para>
         </listitem>
        </itemizedlist>
       </para>
       <para>
        Note that outlines, threads, and other global features of the
        original PDF file are not preserved. For each page of output,
        this option creates an empty PDF and copies a single page from
        the output into it. If you require the global data, you will
        have to run <command>qpdf</command> with the
        <option>--pages</option> option once for each file. Using
        <option>--split-pages</option> is much faster if you don't
        require the global data.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--overlay options --</option></term>
      <listitem>
       <para>
        Overlay pages from another file onto the output pages. See
        <xref linkend="ref.overlay-underlay"/> for details on
        overlay/underlay.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--underlay options --</option></term>
      <listitem>
       <para>
        Overlay pages from another file onto the output pages. See
        <xref linkend="ref.overlay-underlay"/> for details on
        overlay/underlay.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
   </para>
   <para>
    Password-protected files may be opened by specifying a password.
    By default, qpdf will preserve any encryption data associated with
    a file.  If <option>--decrypt</option> is specified, qpdf will
    attempt to remove any encryption information.  If
    <option>--encrypt</option> is specified, qpdf will replace the
    document's encryption parameters with whatever is specified.
   </para>
   <para>
    Note that qpdf does not obey encryption restrictions already
    imposed on the file.  Doing so would be meaningless since qpdf can
    be used to remove encryption from the file entirely.  This
    functionality is not intended to be used for bypassing copyright
    restrictions or other restrictions placed on files by their
    producers.
   </para>
   <para>
    Prior to 8.4.0, in the case of passwords that contain characters
    that fall outside of 7-bit US-ASCII, qpdf left the burden of
    supplying properly encoded encryption and decryption passwords to
    the user. Starting in qpdf 8.4.0, qpdf does this automatically in
    most cases. For an in-depth discussion, please see <xref
    linkend="ref.unicode-passwords"/>. Previous versions of this
    manual described workarounds using the <command>iconv</command>
    command. Such workarounds are no longer required or recommended
    with qpdf 8.4.0. However, for backward compatibility, qpdf
    attempts to detect those workarounds and do the right thing in
    most cases.
   </para>
  </sect1>
  <sect1 id="ref.encryption-options">
   <title>Encryption Options</title>
   <para>
    To change the encryption parameters of a file, use the --encrypt
    flag.  The syntax is

    <programlisting><option>--encrypt <replaceable>user-password</replaceable> <replaceable>owner-password</replaceable> <replaceable>key-length</replaceable> [ <replaceable>restrictions</replaceable> ] --</option>
</programlisting>
    Note that &ldquo;<option>--</option>&rdquo; terminates parsing of
    encryption flags and must be present even if no restrictions are
    present.
   </para>
   <para>
    Either or both of the user password and the owner password may be
    empty strings.
   </para>
   <para>
    The value for
    <option><replaceable>key-length</replaceable></option> may be 40,
    128, or 256.  The restriction flags are dependent upon key length.
    When no additional restrictions are given, the default is to be
    fully permissive.
   </para>
   <para>
    If <option><replaceable>key-length</replaceable></option> is 40,
    the following restriction options are available:
    <variablelist>
     <varlistentry>
      <term><option>--print=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow printing.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--modify=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow document modification.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--extract=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow text/image extraction.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--annotate=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow comments and form fill-in
        and signing.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
    If <option><replaceable>key-length</replaceable></option> is 128,
    the following restriction options are available:
    <variablelist>
     <varlistentry>
      <term><option>--accessibility=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow accessibility to visually
        impaired. The qpdf library disregards this field when AES is
        used or when 256-bit encryption is used. You should really
        never disable accessibility, but qpdf lets you do it in case
        you need to configure a file this way for testing purposes.
        The PDF spec says that conforming readers should disregard
        this permission and always allow accessibility.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--extract=[yn]</option></term>
      <listitem>
       <para>
        Determines whether or not to allow text/graphic extraction.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--assemble=[yn]</option></term>
      <listitem>
       <para>
        Determines whether document assembly (rotation and reordering
        of pages) is allowed.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--annotate=[yn]</option></term>
      <listitem>
       <para>
        Determines whether modifying annotations is allowed. This
        includes adding comments and filling in form fields. Also
        allows editing of form fields if
        <option>--modify-other=y</option> is given.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--form=[yn]</option></term>
      <listitem>
       <para>
        Determines whether filling form fields is allowed.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--modify-other=[yn]</option></term>
      <listitem>
       <para>
        Allow all document editing except those controlled separately
        by the <option>--assemble</option>,
        <option>--annotate</option>, and <option>--form</option>
        options.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--print=<replaceable>print-opt</replaceable></option></term>
      <listitem>
       <para>
        Controls printing access.
        <option><replaceable>print-opt</replaceable></option> may be
        one of the following:
        <itemizedlist>
         <listitem>
          <para>
           <option>full</option>: allow full printing
          </para>
         </listitem>
         <listitem>
          <para>
           <option>low</option>: allow low-resolution printing only
          </para>
         </listitem>
         <listitem>
          <para>
           <option>none</option>: disallow printing
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--modify=<replaceable>modify-opt</replaceable></option></term>
      <listitem>
       <para>
        Controls modify access. This way of controlling modify access
        has less granularity than new options added in qpdf 8.4.
        <option><replaceable>modify-opt</replaceable></option> may be
        one of the following:
        <itemizedlist>
         <listitem>
          <para>
           <option>all</option>: allow full document modification
          </para>
         </listitem>
         <listitem>
          <para>
           <option>annotate</option>: allow comment authoring, form
           operations, and document assembly
          </para>
         </listitem>
         <listitem>
          <para>
           <option>form</option>: allow form field fill-in and signing
           and document assembly
          </para>
         </listitem>
         <listitem>
          <para>
           <option>assembly</option>: allow document assembly only
          </para>
         </listitem>
         <listitem>
          <para>
           <option>none</option>: allow no modifications
          </para>
         </listitem>
        </itemizedlist>
        Using the <option>--modify</option> option does not allow you
        to create certain combinations of permissions such as allowing
        form filling but not allowing document assembly. Starting with
        qpdf 8.4, you can either just use the other options to control
        fields individually, or you can use something like
        <option>--modify=form --assembly=n</option> to fine tune.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--cleartext-metadata</option></term>
      <listitem>
       <para>
        If specified, any metadata stream in the document will be left
        unencrypted even if the rest of the document is encrypted.
        This also forces the PDF version to be at least 1.5.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--use-aes=[yn]</option></term>
      <listitem>
       <para>
        If <option>--use-aes=y</option> is specified, AES encryption
        will be used instead of RC4 encryption.  This forces the PDF
        version to be at least 1.6.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--force-V4</option></term>
      <listitem>
       <para>
        Use of this option forces the <literal>/V</literal> and
        <literal>/R</literal> parameters in the document's encryption
        dictionary to be set to the value <literal>4</literal>.  As
        qpdf will automatically do this when required, there is no
        reason to ever use this option.  It exists primarily for use
        in testing qpdf itself.  This option also forces the PDF
        version to be at least 1.5.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
    If <option><replaceable>key-length</replaceable></option> is 256,
    the minimum PDF version is 1.7 with extension level 8, and the
    AES-based encryption format used is the PDF 2.0 encryption method
    supported by Acrobat X.  the same options are available as with
    128 bits with the following exceptions:
    <variablelist>
     <varlistentry>
      <term><option>--use-aes</option></term>
      <listitem>
       <para>
        This option is not available with 256-bit keys.  AES is always
        used with 256-bit encryption keys.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--force-V4</option></term>
      <listitem>
       <para>
        This option is not available with 256 keys.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--force-R5</option></term>
      <listitem>
       <para>
        If specified, qpdf sets the minimum version to 1.7 at
        extension level 3 and writes the deprecated encryption format
        used by Acrobat version IX.  This option should not be used in
        practice to generate PDF files that will be in general use,
        but it can be useful to generate files if you are trying to
        test proper support in another application for PDF files
        encrypted in this way.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
    The default for each permission option is to be fully permissive.
   </para>
  </sect1>
  <sect1 id="ref.page-selection">
   <title>Page Selection Options</title>
   <para>
    Starting with qpdf 3.0, it is possible to split and merge PDF
    files by selecting pages from one or more input files.  Whatever
    file is given as the primary input file is used as the starting
    point, but its pages are replaced with pages as specified.

    <programlisting><option>--pages <replaceable>input-file</replaceable> [ <replaceable>--password=password</replaceable> ] [ <replaceable>page-range</replaceable> ] [ ... ] --</option>
</programlisting>
    Multiple input files may be specified.  Each one is given as the
    name of the input file, an optional password (if required to open
    the file), and the range of pages.  Note that
    &ldquo;<option>--</option>&rdquo; terminates parsing of page
    selection flags.
   </para>
   <para>
    Starting with qpf 8.4, the special input file name
    &ldquo;<filename>.</filename>&rdquo; can be used shortcut for the
    primary input filename.
   </para>
   <para>
    For each file that pages should be taken from, specify the file, a
    password needed to open the file (if any), and a page range.  The
    password needs to be given only once per file.  If any of the
    input files are the same as the primary input file or the file
    used to copy encryption parameters (if specified), you do not need
    to repeat the password here.  The same file can be repeated
    multiple times.  If a file that is repeated has a password, the
    password only has to be given the first time.  All non-page data
    (info, outlines, page numbers, etc.) are taken from the primary
    input file.  To discard these, use <option>--empty</option> as the
    primary input.
   </para>
   <para>
    Starting with qpdf 5.0.0, it is possible to omit the page range.
    If qpdf sees a value in the place where it expects a page range
    and that value is not a valid range but is a valid file name, qpdf
    will implicitly use the range <literal>1-z</literal>, meaning that
    it will include all pages in the file.  This makes it possible to
    easily combine all pages in a set of files with a command like
    <command>qpdf --empty out.pdf --pages *.pdf --</command>.
   </para>
   <para>
    The page range is a set of numbers separated by commas, ranges of
    numbers separated dashes, or combinations of those. The character
    &ldquo;z&rdquo; represents the last page. A number preceded by an
    &ldquo;r&rdquo; indicates to count from the end, so
    <literal>r3-r1</literal> would be the last three pages of the
    document. Pages can appear in any order. Ranges can appear with a
    high number followed by a low number, which causes the pages to
    appear in reverse. Numbers may be repeated in a page range. A page
    range may be optionally appended with <literal>:even</literal> or
    <literal>:odd</literal> to indicate only the even or odd pages in
    the given range. Note that even and odd refer to the positions
    within the specified, range, not whether the original number is
    even or odd.
   </para>
   <para>
    Example page ranges:
    <itemizedlist>
     <listitem>
      <para>
       <literal>1,3,5-9,15-12</literal>: pages 1, 3, 5, 6, 7, 8,
       9, 15, 14, 13, and 12 in that order.
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>z-1</literal>: all pages in the document in reverse
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>r3-r1</literal>: the last three pages of the document
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>r1-r3</literal>: the last three pages of the document
       in reverse order
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>1-20:even</literal>: even pages from 2 to 20
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>5,7-9,12:odd</literal>: pages 5, 8, and, 12, which are
       the pages in odd positions from among the original range, which
       represents pages 5, 7, 8, 9, and 12.
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Starting in qpdf version 8.3, you can specify the
    <option>--collate</option> option. Note that this option is
    specified outside of <option>--pages&nbsp;...&nbsp;--</option>.
    When <option>--collate</option> is specified, it changes the
    meaning of <option>--pages</option> so that the specified files,
    as modified by page ranges, are collated rather than concatenated.
    For example, if you add the files <filename>odd.pdf</filename> and
    <filename>even.pdf</filename> containing odd and even pages of a
    document respectively, you could run <command>qpdf --collate
    odd.pdf --pages odd.pdf even.pdf -- all.pdf</command> to collate
    the pages. This would pick page 1 from odd, page 1 from even, page
    2 from odd, page 2 from even, etc. until all pages have been
    included. Any number of files and page ranges can be specified. If
    any file has fewer pages, that file is just skipped when its pages
    have all been included. For example, if you ran <command>qpdf
    --collate --empty --pages a.pdf 1-5 b.pdf 6-4 c.pdf r1 --
    out.pdf</command>, you would get the following pages in this
    order:
    <itemizedlist>
     <listitem><para>a.pdf page 1</para></listitem>
     <listitem><para>b.pdf page 6</para></listitem>
     <listitem><para>c.pdf last page</para></listitem>
     <listitem><para>a.pdf page 2</para></listitem>
     <listitem><para>b.pdf page 5</para></listitem>
     <listitem><para>a.pdf page 3</para></listitem>
     <listitem><para>b.pdf page 4</para></listitem>
     <listitem><para>a.pdf page 4</para></listitem>
     <listitem><para>a.pdf page 5</para></listitem>
    </itemizedlist>
   </para>
   <para>
    Starting in qpdf version 8.3, when you split and merge files, any
    page labels (page numbers) are preserved in the final file. It is
    expected that more document features will be preserved by
    splitting and merging. In the mean time, semantics of splitting
    and merging vary across features. For example, the document's
    outlines (bookmarks) point to actual page objects, so if you
    select some pages and not others, bookmarks that point to pages
    that are in the output file will work, and remaining bookmarks
    will not work. A future version of <command>qpdf</command> may do
    a better job at handling these issues. (Note that the qpdf library
    already contains all of the APIs required in order to implement
    this in your own application if you need it.) In the mean time,
    you can always use <option>--empty</option> as the primary input
    file to avoid copying all of that from the first file. For
    example, to take pages 1 through 5 from a
    <filename>infile.pdf</filename> while preserving all metadata
    associated with that file, you could use

    <programlisting><command>qpdf</command> <option>infile.pdf --pages . 1-5 -- outfile.pdf</option>
</programlisting>
    If you wanted pages 1 through 5 from
    <filename>infile.pdf</filename> but you wanted the rest of the
    metadata to be dropped, you could instead run

    <programlisting><command>qpdf</command> <option>--empty --pages infile.pdf 1-5 -- outfile.pdf</option>
</programlisting>
    If you wanted to take pages 1&ndash;5 from
    <filename>file1.pdf</filename> and pages 11&ndash;15 from
    <filename>file2.pdf</filename> in reverse, you would run

    <programlisting><command>qpdf</command> <option>file1.pdf --pages file1.pdf 1-5 . 15-11 -- outfile.pdf</option>
</programlisting>
    If, for some reason, you wanted to take the first page of an
    encrypted file called <filename>encrypted.pdf</filename> with
    password <literal>pass</literal> and repeat it twice in an output
    file, and if you wanted to drop document-level metadata but
    preserve encryption, you would use

    <programlisting><command>qpdf</command> <option>--empty --copy-encryption=encrypted.pdf --encryption-file-password=pass
--pages encrypted.pdf --password=pass 1 ./encrypted.pdf --password=pass 1 --
outfile.pdf</option>
</programlisting>
    Note that we had to specify the password all three times because
    giving a password as <option>--encryption-file-password</option>
    doesn't count for page selection, and as far as qpdf is concerned,
    <filename>encrypted.pdf</filename> and
    <filename>./encrypted.pdf</filename> are separated files.  These
    are all corner cases that most users should hopefully never have
    to be bothered with.
   </para>
   <para>
    Prior to version 8.4, it was not possible to specify the same page
    from the same file directly more than once, and the workaround of
    specifying the same file in more than one way was required.
    Version 8.4 removes this limitation.
   </para>
  </sect1>
  <sect1 id="ref.overlay-underlay">
   <title>Overlay and Underlay Options</title>
   <para>
    Starting with qpdf 8.4, it is possible to overlay or underlay
    pages from other files onto the output generated by qpdf. Specify
    overlay or underlay as follows:

    <programlisting>{ <option>--overlay</option> | <option>--underlay</option> } <replaceable>file</replaceable> [ <option>options</option> ] <option>--</option>
</programlisting>
    Overlay and underlay options are processed late, so they can be
    combined with other like merging and will apply to the final
    output. The <option>--overlay</option> and
    <option>--underlay</option> options work the same way, except
    underlay pages are drawn underneath the page to which they are
    applied, possibly obscured by the original page, and overlay files
    are drawn on top of the page to which they are applied, possibly
    obscuring the page. You can combine overlay and underlay.
   </para>
   <para>
    The default behavior of overlay and underlay is that pages are
    taken from the overlay/underlay file in sequence and applied to
    corresponding pages in the output until there are no more output
    pages. If the overlay or underlay file runs out of pages,
    remaining output pages are left alone. This behavior can be
    modified by options, which are provided between the
    <option>--overlay</option> or <option>--underlay</option> flag and
    the <option>--</option> option. The following options are
    supported:
    <itemizedlist>
     <listitem>
      <para>
       <option>--password=password</option>: supply a password if the
       overlay/underlay file is encrypted.
      </para>
     </listitem>
     <listitem>
      <para>
       <option>--to=page-range</option>: a range of pages in the same
       form at described in <xref linkend="ref.page-selection"/>
       indicates which pages in the output should have the
       overlay/underlay applied. If not specified, overlay/underlay
       are applied to all pages.
      </para>
     </listitem>
     <listitem>
      <para>
       <option>--from=[page-range]</option>: a range of pages that
       specifies which pages in the overlay/underlay file will be used
       for overlay or underlay. If not specified, all pages will be
       used. This can be explicitly specified to be empty if
       <option>--repeat</option> is used.
      </para>
     </listitem>
     <listitem>
      <para>
       <option>--repeat=page-range</option>: an optional range of
       pages that specifies which pages in the overlay/underlay file
       will be repeated after the &ldquo;from&rdquo; pages are used
       up. If you want to repeat a range of pages starting at the
       beginning, you can explicitly use <option>--from=</option>.
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Here are some examples.
    <itemizedlist>
     <listitem>
      <para>
       <command>--overlay o.pdf --to=1-5 --from=1-3
       --repeat=4 --</command>: overlay the first three pages from file
       <filename>o.pdf</filename> onto the first three pages of the
       output, then overlay page 4 from <filename>o.pdf</filename>
       onto pages 4 and 5 of the output. Leave remaining output pages
       untouched.
      </para>
     </listitem>
     <listitem>
      <para>
       <command>--underlay footer.pdf --from= --repeat=1,2 --</command>:
       Underlay page 1 of <filename>footer.pdf</filename> on all odd
       output pages, and underlay page 2 of
       <filename>footer.pdf</filename> on all even output pages.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
  <sect1 id="ref.advanced-parsing">
   <title>Advanced Parsing Options</title>
   <para>
    These options control aspects of how qpdf reads PDF files. Mostly
    these are of use to people who are working with damaged files.
    There is little reason to use these options unless you are trying
    to solve specific problems. The following options are available:
    <variablelist>
     <varlistentry>
      <term><option>--suppress-recovery</option></term>
      <listitem>
       <para>
        Prevents qpdf from attempting to recover damaged files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--ignore-xref-streams</option></term>
      <listitem>
       <para>
        Tells qpdf to ignore any cross-reference streams.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
   </para>
   <para>
    Ordinarily, qpdf will attempt to recover from certain types of
    errors in PDF files.  These include errors in the cross-reference
    table, certain types of object numbering errors, and certain types
    of stream length errors.  Sometimes, qpdf may think it has
    recovered but may not have actually recovered, so care should be
    taken when using this option as some data loss is possible.  The
    <option>--suppress-recovery</option> option will prevent qpdf from
    attempting recovery.  In this case, it will fail on the first
    error that it encounters.
   </para>
   <para>
    Ordinarily, qpdf reads cross-reference streams when they are
    present in a PDF file.  If <option>--ignore-xref-streams</option>
    is specified, qpdf will ignore any cross-reference streams for
    hybrid PDF files.  The purpose of hybrid files is to make some
    content available to viewers that are not aware of cross-reference
    streams.  It is almost never desirable to ignore them.  The only
    time when you might want to use this feature is if you are testing
    creation of hybrid PDF files and wish to see how a PDF consumer
    that doesn't understand object and cross-reference streams would
    interpret such a file.
   </para>
  </sect1>
  <sect1 id="ref.advanced-transformation">
   <title>Advanced Transformation Options</title>
   <para>
    These transformation options control fine points of how qpdf
    creates the output file.  Mostly these are of use only to people
    who are very familiar with the PDF file format or who are PDF
    developers.  The following options are available:
    <variablelist>
     <varlistentry>
      <term><option>--compress-streams=<replaceable>[yn]</replaceable></option></term>
      <listitem>
       <para>
        By default, or with <option>--compress-streams=y</option>,
        qpdf will compress any stream with no other filters applied to
        it with the <literal>/FlateDecode</literal> filter when it
        writes it. To suppress this behavior and preserve uncompressed
        streams as uncompressed, use
        <option>--compress-streams=n</option>.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--decode-level=<replaceable>option</replaceable></option></term>
      <listitem>
       <para>
        Controls which streams qpdf tries to decode. The default is
        <option>generalized</option>. The following options are
        available:
        <itemizedlist>
         <listitem>
          <para>
           <option>none</option>: do not attempt to decode any streams
          </para>
         </listitem>
         <listitem>
          <para>
           <option>generalized</option>: decode streams filtered with
           supported generalized filters:
           <literal>/LZWDecode</literal>,
           <literal>/FlateDecode</literal>,
           <literal>/ASCII85Decode</literal>, and
           <literal>/ASCIIHexDecode</literal>. We define generalized
           filters as those to be used for general-purpose compression
           or encoding, as opposed to filters specifically designed
           for image data. Note that, by default, streams already
           compressed with <literal>/FlateDecode</literal> are not
           uncompressed and recompressed unless you also specify
           <option>--recompress-flate</option>.
          </para>
         </listitem>
         <listitem>
          <para>
           <option>specialized</option>: in addition to generalized,
           decode streams with supported non-lossy specialized
           filters; currently this is just
           <literal>/RunLengthDecode</literal>
          </para>
         </listitem>
         <listitem>
          <para>
           <option>all</option>: in addition to generalized and
           specialized, decode streams with supported lossy filters;
           currently this is just <literal>/DCTDecode</literal> (JPEG)
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--stream-data=<replaceable>option</replaceable></option></term>
      <listitem>
       <para>
        Controls transformation of stream data. This option predates
        the <option>--compress-streams</option> and
        <option>--decode-level</option> options. Those options can be
        used to achieve the same affect with more control. The value
        of <option><replaceable>option</replaceable></option> may be
        one of the following:
        <itemizedlist>
         <listitem>
          <para>
           <option>compress</option>: recompress stream data when
           possible (default); equivalent to
           <option>--compress-streams=y</option>
           <option>--decode-level=generalized</option>. Does not
           recompress streams already compressed with
           <literal>/FlateDecode</literal> unless
           <option>--recompress-flate</option> is also specified.
          </para>
         </listitem>
         <listitem>
          <para>
           <option>preserve</option>: leave all stream data as is;
           equivalent to <option>--compress-streams=n</option>
           <option>--decode-level=none</option>
          </para>
         </listitem>
         <listitem>
          <para>
           <option>uncompress</option>: uncompress stream data
           compressed with generalized filters when possible;
           equivalent to <option>--compress-streams=n</option>
           <option>--decode-level=generalized</option>
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--recompress-flate</option></term>
      <listitem>
       <para>
        By default, streams already compressed with
        <literal>/FlateDecode</literal> are left alone rather than
        being uncompressed and recompressed. This option causes qpdf
        to uncompress and recompress the streams. There is a
        significant performance cost to using this option, but you
        probably want to use it if you specify
        <option>--compression-level</option>.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--compression-level=<replaceable>level</replaceable></option></term>
      <listitem>
       <para>
        When writing new streams that are compressed with
        <literal>/FlateDecode</literal>, use the specified compression
        level. The value of <option>level</option> should be a number
        from 1 to 9 and is passed directly to zlib, which implements
        deflate compression. Note that qpdf doesn't uncompress and
        recompress streams by default. To have this option apply to
        already compressed streams, you should also specify
        <option>--recompress-flate</option>. If your goal is to shrink
        the size of PDF files, you should also use
        <option>--object-streams=generate</option>.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--normalize-content=[yn]</option></term>
      <listitem>
       <para>
        Enables or disables normalization of content streams. Content
        normalization is enabled by default in QDF mode. Please see
        <xref linkend="ref.qdf"/> for additional discussion of QDF
        mode.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--object-streams=<replaceable>mode</replaceable></option></term>
      <listitem>
       <para>
        Controls handling of object streams.  The value of
        <option><replaceable>mode</replaceable></option> may be one of
        the following:
        <itemizedlist>
         <listitem>
          <para>
           <option>preserve</option>: preserve original object streams
           (default)
          </para>
         </listitem>
         <listitem>
          <para>
           <option>disable</option>: don't write any object streams
          </para>
         </listitem>
         <listitem>
          <para>
           <option>generate</option>: use object streams wherever
           possible
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--preserve-unreferenced</option></term>
      <listitem>
       <para>
        Tells qpdf to preserve objects that are not referenced when
        writing the file. Ordinarily any object that is not referenced
        in a traversal of the document from the trailer dictionary
        will be discarded. This may be useful in working with some
        damaged files or inspecting files with known unreferenced
        objects.
       </para>
       <para>
        This flag is ignored for linearized files and has the effect
        of causing objects in the new file to be written in order by
        object ID from the original file. This does not mean that
        object numbers will be the same since qpdf may create stream
        lengths as direct or indirect differently from the original
        file, and the original file may have gaps in its numbering.
       </para>
       <para>
        See also <option>--preserve-unreferenced-resources</option>,
        which does something completely different.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--preserve-unreferenced-resources</option></term>
      <listitem>
       <para>
        Starting with qpdf 8.1, when splitting pages, qpdf ordinarily
        attempts to remove images and fonts that are not used by a
        page even if they are referenced in the page's resources
        dictionary. This option suppresses that behavior. There are
        few reasons to use this option. One reason to use this is if
        you suspect that qpdf is removing resources it shouldn't be
        removing. If you encounter that case, please report it as a
        bug. Another reason is that the new behavior can be much
        slower for files that include a very large number of images or
        other XObjects on a page. In that case, using this option will
        return qpdf to the old behavior and speed.
       </para>
       <para>
        See also <option>--preserve-unreferenced</option>, which does
        something completely different.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--newline-before-endstream</option></term>
      <listitem>
       <para>
        Tells qpdf to insert a newline before the
        <literal>endstream</literal> keyword, not counted in the
        length, after any stream content even if the last character of
        the stream was a newline. This may result in two newlines in
        some cases. This is a requirement of PDF/A. While qpdf doesn't
        specifically know how to generate PDF/A-compliant PDFs, this
        at least prevents it from removing compliance on already
        compliant files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--linearize-pass1=<replaceable>file</replaceable></option></term>
      <listitem>
       <para>
        Write the first pass of linearization to the named file. The
        resulting file is not a valid PDF file. This option is useful
        only for debugging <classname>QPDFWriter</classname>'s
        linearization code. When qpdf linearizes files, it writes the
        file in two passes, using the first pass to calculate sizes
        and offsets that are required for hint tables and the
        linearization dictionary. Ordinarily, the first pass is
        discarded. This option enables it to be captured.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--coalesce-contents</option></term>
      <listitem>
       <para>
        When a page's contents are split across multiple streams, this
        option causes qpdf to combine them into a single stream. Use
        of this option is never necessary for ordinary usage, but it
        can help when working with some files in some cases. For
        example, some PDF writers split page contents into small
        streams at arbitrary points that may fall in the middle of
        lexical tokens within the content, and some PDF readers may
        get confused on such files. If you use qpdf to coalesce the
        content streams, such readers may be able to work with the
        file more easily. This can also be combined with QDF mode or
        content normalization to make it easier to look at all of a
        page's contents at once.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--flatten-annotations=<replaceable>option</replaceable></option></term>
      <listitem>
       <para>
        This option collapses annotations into the pages' contents
        with special handling for form fields. Ordinarily, an
        annotation is rendered separately and on top of the page.
        Combining annotations into the page's contents effectively
        freezes the placement of the annotations, making them look
        right after various page transformations. The library
        functionality backing this option was added for the benefit of
        programs that want to create <emphasis>n-up</emphasis> page
        layouts and other similar things that don't work well with
        annotations. The <replaceable>option</replaceable> parameter
        may be any of the following:
        <itemizedlist>
         <listitem>
          <para>
           <option>all</option>: include all annotations that are not
           marked invisible or hidden
          </para>
         </listitem>
         <listitem>
          <para>
           <option>print</option>: only include annotations that
           indicate that they should appear when the page is printed
          </para>
         </listitem>
         <listitem>
          <para>
           <option>screen</option>: omit annotations that indicate
           they should not appear on the screen
          </para>
         </listitem>
        </itemizedlist>
       </para>
       <para>
        Note that form fields are special because the annotations that
        are used to render filled-in form fields may become out of
        date from the fields' values if the form is filled in by a
        program that doesn't know how to update the appearances. If
        qpdf detects this case, its default behavior is not to flatten
        those annotations because doing so would cause the value of
        the form field to be lost. This gives you a chance to go back
        and resave the form with a program that knows how to generate
        appearances. QPDF itself can generate appearances with some
        limitations. See the <option>--generate-appearances</option>
        option below.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--generate-appearances</option></term>
      <listitem>
       <para>
        If a file contains interactive form fields and indicates that
        the appearances are out of date with the values of the form,
        this flag will regenerate appearances, subject to a few
        limitations. Note that there is not usually a reason to do
        this, but it can be necessary before using the
        <option>--flatten-annotations</option> option. Most of these
        are not a problem with well-behaved PDF files. The limitations
        are as follows:
        <itemizedlist>
         <listitem>
          <para>
           Radio button and checkbox appearances use the pre-set
           values in the PDF file. QPDF just makes sure that the
           correct appearance is displayed based on the value of the
           field. This is fine for PDF files that create their forms
           properly. Some PDF writers save appearances for fields when
           they change, which could cause some controls to have
           inconsistent appearances.
          </para>
         </listitem>
        </itemizedlist>
        <itemizedlist>
         <listitem>
          <para>
           For text fields and list boxes, any characters that fall
           outside of US-ASCII or, if detected, &ldquo;Windows
           ANSI&rdquo; or &ldquo;Mac Roman&rdquo; encoding, will be
           replaced by the <literal>?</literal> character.
          </para>
         </listitem>
        </itemizedlist>
        <itemizedlist>
         <listitem>
          <para>
           Quadding is ignored. Quadding is used to specify whether
           the contents of a field should be left, center, or right
           aligned with the field.
          </para>
         </listitem>
        </itemizedlist>
        <itemizedlist>
         <listitem>
          <para>
           Rich text, multi-line, and other more elaborate formatting
           directives are ignored.
          </para>
         </listitem>
        </itemizedlist>
        <itemizedlist>
         <listitem>
          <para>
           There is no support for multi-select fields or signature
           fields.
          </para>
         </listitem>
        </itemizedlist>
        If qpdf doesn't do a good enough job with your form, use an
        external application to save your filled-in form before
        processing it with qpdf.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--optimize-images</option></term>
      <listitem>
       <para>
        This flag causes qpdf to recompress all images that are not
        compressed with DCT (JPEG) using DCT compression as long as
        doing so decreases the size in bytes of the image data and the
        image does not fall below minimum specified dimensions. Useful
        information is provided when used in combination with
        <option>--verbose</option>. See also the
        <option>--oi-min-width</option>,
        <option>--oi-min-height</option>, and
        <option>--oi-min-area</option> options. By default, starting
        in qpdf 8.4, inline images are converted to regular images
        and optimized as well. Use
        <option>--keep-inline-images</option> to prevent inline images
        from being included.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--oi-min-width=<replaceable>width</replaceable></option></term>
      <listitem>
       <para>
        Avoid optimizing images whose width is below the specified
        amount. If omitted, the default is 128 pixels. Use 0 for no
        minimum.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--oi-min-height=<replaceable>height</replaceable></option></term>
      <listitem>
       <para>
        Avoid optimizing images whose height is below the specified
        amount. If omitted, the default is 128 pixels. Use 0 for no
        minimum.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--oi-min-area=<replaceable>area-in-pixels</replaceable></option></term>
      <listitem>
       <para>
        Avoid optimizing images whose pixel count
        (width&nbsp;×&nbsp;height) is below the specified amount. If
        omitted, the default is 16,384 pixels. Use 0 for no minimum.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--externalize-inline-images</option></term>
      <listitem>
       <para>
        Convert inline images to regular images. By default, images
        whose data is at least 1,024 bytes are converted when this
        option is selected. Use <option>--ii-min-bytes</option> to
        change the size threshold. This option is implicitly selected
        when <option>--optimize-images</option> is selected. Use
        <option>--keep-inline-images</option> to exclude inline images
        from image optimization.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--ii-min-bytes=<replaceable>bytes</replaceable></option></term>
      <listitem>
       <para>
        Avoid converting inline images whose size is below the
        specified minimum size to regular images. If omitted, the
        default is 1,024 bytes. Use 0 for no minimum.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--keep-inline-images</option></term>
      <listitem>
       <para>
        Prevent inline images from being included in image
        optimization. This option has no affect when
        <option>--optimize-images</option> is not specified.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--remove-page-labels</option></term>
      <listitem>
       <para>
        Remove page labels from the output file.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--qdf</option></term>
      <listitem>
       <para>
        Turns on QDF mode.  For additional information on QDF, please
        see <xref linkend="ref.qdf"/>. Note that
        <option>--linearize</option> disables QDF mode.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--min-version=<replaceable>version</replaceable></option></term>
      <listitem>
       <para>
        Forces the PDF version of the output file to be at least
        <replaceable>version</replaceable>.  In other words, if the
        input file has a lower version than the specified version, the
        specified version will be used.  If the input file has a
        higher version, the input file's original version will be
        used.  It is seldom necessary to use this option since qpdf
        will automatically increase the version as needed when adding
        features that require newer PDF readers.
       </para>
       <para>
        The version number may be expressed in the form
        <replaceable>major.minor.extension-level</replaceable>, in
        which case the version is interpreted as
        <replaceable>major.minor</replaceable> at extension level
        <replaceable>extension-level</replaceable>.  For example,
        version <literal>1.7.8</literal> represents version 1.7 at
        extension level 8.  Note that minimal syntax checking is done
        on the command line.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--force-version=<replaceable>version</replaceable></option></term>
      <listitem>
       <para>
        This option forces the PDF version to be the exact version
        specified <emphasis>even when the file may have content that
        is not supported in that version</emphasis>.  The version
        number is interpreted in the same way as with
        <option>--min-version</option> so that extension levels can be
        set.  In some cases, forcing the output file's PDF version to
        be lower than that of the input file will cause qpdf to
        disable certain features of the document.  Specifically,
        256-bit keys are disabled if the version is less than 1.7 with
        extension level 8 (except R5 is disabled if less than 1.7 with
        extension level 3), AES encryption is disabled if the version
        is less than 1.6, cleartext metadata and object streams are
        disabled if less than 1.5, 128-bit encryption keys are
        disabled if less than 1.4, and all encryption is disabled if
        less than 1.3.  Even with these precautions, qpdf won't be
        able to do things like eliminate use of newer image
        compression schemes, transparency groups, or other features
        that may have been added in more recent versions of PDF.
       </para>
       <para>
        As a general rule, with the exception of big structural things
        like the use of object streams or AES encryption, PDF viewers
        are supposed to ignore features in files that they don't
        support from newer versions.  This means that forcing the
        version to a lower version may make it possible to open your
        PDF file with an older version, though bear in mind that some
        of the original document's functionality may be lost.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
   </para>
   <para>
    By default, when a stream is encoded using non-lossy filters that
    qpdf understands and is not already compressed using a good
    compression scheme, qpdf will uncompress and recompress streams.
    Assuming proper filter implements, this is safe and generally
    results in smaller files.  This behavior may also be explicitly
    requested with <option>--stream-data=compress</option>.
   </para>
   <para>
    When <option>--normalize-content=y</option> is specified, qpdf
    will attempt to normalize whitespace and newlines in page content
    streams.  This is generally safe but could, in some cases, cause
    damage to the content streams.  This option is intended for people
    who wish to study PDF content streams or to debug PDF content.
    You should not use this for &ldquo;production&rdquo; PDF files.
   </para>
   <para>
    This paragraph discusses edge cases of content normalization that
    are not of concern to most users and are not relevant when content
    normalization is not enabled. When normalizing content, if qpdf
    runs into any lexical errors, it will print a warning indicating
    that content may be damaged. The only situation in which qpdf is
    known to cause damage during content normalization is when a
    page's contents are split across multiple streams and streams are
    split in the middle of a lexical token such as a string, name, or
    inline image. There may be some pathological cases in which qpdf
    could damage content without noticing this, such as if the partial
    tokens at the end of one stream and the beginning of the next
    stream are both valid, but usually qpdf will be able to detect
    this case. For slightly increased safety, you can specify
    <option>--coalesce-contents</option> in addition to
    <option>--normalize-content</option> or <option>--qdf</option>.
    This will cause qpdf to combine all the content streams into one,
    thus recombining any split tokens. However doing this will prevent
    you from being able to see the original layout of the content
    streams. If you must inspect the original content streams in an
    uncompressed format, you can always run with <option>--qdf
    --normalize-content=n</option> for a QDF file without content
    normalization, or alternatively
    <option>--stream-data=uncompress</option> for a regular non-QDF
    mode file with uncompressed streams. These will both uncompress
    all the streams but will not attempt to normalize content. Please
    note that if you are using content normalization or QDF mode for
    the purpose of manually inspecting files, you don't have to care
    about this.
   </para>
   <para>
    Object streams, also known as compressed objects, were introduced
    into the PDF specification at version 1.5, corresponding to
    Acrobat 6.  Some older PDF viewers may not support files with
    object streams.  qpdf can be used to transform files with object
    streams to files without object streams or vice versa.  As
    mentioned above, there are three object stream modes:
    <option>preserve</option>, <option>disable</option>, and
    <option>generate</option>.
   </para>
   <para>
    In <option>preserve</option> mode, the relationship to objects and
    the streams that contain them is preserved from the original file.
    In <option>disable</option> mode, all objects are written as
    regular, uncompressed objects.  The resulting file should be
    readable by older PDF viewers.  (Of course, the content of the
    files may include features not supported by older viewers, but at
    least the structure will be supported.)  In
    <option>generate</option> mode, qpdf will create its own object
    streams.  This will usually result in more compact PDF files,
    though they may not be readable by older viewers.  In this mode,
    qpdf will also make sure the PDF version number in the header is
    at least 1.5.
   </para>
   <para>
    The <option>--qdf</option> flag turns on QDF mode, which changes
    some of the defaults described above.  Specifically, in QDF mode,
    by default, stream data is uncompressed, content streams are
    normalized, and encryption is removed.  These defaults can still
    be overridden by specifying the appropriate options as described
    above.  Additionally, in QDF mode, stream lengths are stored as
    indirect objects, objects are laid out in a less efficient but
    more readable fashion, and the documents are interspersed with
    comments that make it easier for the user to find things and also
    make it possible for <command>fix-qdf</command> to work properly.
    QDF mode is intended for people, mostly developers, who wish to
    inspect or modify PDF files in a text editor.  For details, please
    see <xref linkend="ref.qdf"/>.
   </para>
  </sect1>
  <sect1 id="ref.testing-options">
   <title>Testing, Inspection, and Debugging Options</title>
   <para>
    These options can be useful for digging into PDF files or for use
    in automated test suites for software that uses the qpdf library.
    When any of the options in this section are specified, no output
    file should be given.  The following options are available:
    <variablelist>
     <varlistentry>
      <term><option>--deterministic-id</option></term>
      <listitem>
       <para>
        Causes generation of a deterministic value for /ID. This
        prevents use of timestamp and output file name information in
        the /ID generation. Instead, at some slight additional runtime
        cost, the /ID field is generated to include a digest of the
        significant parts of the content of the output PDF file. This
        means that a given qpdf operation should generate the same /ID
        each time it is run, which can be useful when caching results
        or for generation of some test data. Use of this flag is not
        compatible with creation of encrypted files.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--static-id</option></term>
      <listitem>
       <para>
        Causes generation of a fixed value for /ID. This is intended
        for testing only. Never use it for production files. If you
        are trying to get the same /ID each time for a given file and
        you are not generating encrypted files, consider using the
        <option>--deterministic-id</option> option.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--static-aes-iv</option></term>
      <listitem>
       <para>
        Causes use of a static initialization vector for AES-CBC.
        This is intended for testing only so that output files can be
        reproducible.  Never use it for production files.  This option
        in particular is not secure since it significantly weakens the
        encryption.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--no-original-object-ids</option></term>
      <listitem>
       <para>
        Suppresses inclusion of original object ID comments in QDF
        files.  This can be useful when generating QDF files for test
        purposes, particularly when comparing them to determine
        whether two PDF files have identical content.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-encryption</option></term>
      <listitem>
       <para>
        Shows document encryption parameters.  Also shows the
        document's user password if the owner password is given.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-encryption-key</option></term>
      <listitem>
       <para>
        When encryption information is being displayed, as when
        <option>--check</option> or <option>--show-encryption</option>
        is given, display the computed or retrieved encryption key as
        a hexadecimal string. This value is not ordinarily useful to
        users, but it can be used as the argument to
        <option>--password</option> if the
        <option>--password-is-hex-key</option> is specified. Note
        that, when PDF files are encrypted, passwords and other
        metadata are used only to compute an encryption key, and the
        encryption key is what is actually used for encryption. This
        enables retrieval of that key.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--check-linearization</option></term>
      <listitem>
       <para>
        Checks file integrity and linearization status.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-linearization</option></term>
      <listitem>
       <para>
        Checks and displays all data in the linearization hint tables.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-xref</option></term>
      <listitem>
       <para>
        Shows the contents of the cross-reference table in a
        human-readable form.  This is especially useful for files with
        cross-reference streams which are stored in a binary format.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-object=trailer|obj[,gen]</option></term>
      <listitem>
       <para>
        Show the contents of the given object.  This is especially
        useful for inspecting objects that are inside of object
        streams (also known as &ldquo;compressed objects&rdquo;).
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--raw-stream-data</option></term>
      <listitem>
       <para>
        When used along with the <option>--show-object</option>
        option, if the object is a stream, shows the raw stream data
        instead of object's contents.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--filtered-stream-data</option></term>
      <listitem>
       <para>
        When used along with the <option>--show-object</option>
        option, if the object is a stream, shows the filtered stream
        data instead of object's contents.  If the stream is filtered
        using filters that qpdf does not support, an error will be
        issued.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-npages</option></term>
      <listitem>
       <para>
        Prints the number of pages in the input file on a line by
        itself.  Since the number of pages appears by itself on a
        line, this option can be useful for scripting if you need to
        know the number of pages in a file.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--show-pages</option></term>
      <listitem>
       <para>
        Shows the object and generation number for each page
        dictionary object and for each content stream associated with
        the page.  Having this information makes it more convenient to
        inspect objects from a particular page.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--with-images</option></term>
      <listitem>
       <para>
        When used along with <option>--show-pages</option>, also shows
        the object and generation numbers for the image objects on
        each page.  (At present, information about images in shared
        resource dictionaries are not output by this command.  This is
        discussed in a comment in the source code.)
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--json</option></term>
      <listitem>
       <para>
        Generate a JSON representation of the file. This is described
        in depth in <xref linkend="ref.json"/>
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--json-help</option></term>
      <listitem>
       <para>
        Describe the format of the JSON output.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--json-key=key</option></term>
      <listitem>
       <para>
        This option is repeatable. If specified, only top-level keys
        specified will be included in the JSON output. If not
        specified, all keys will be shown.
       </para>
      </listitem>
      </varlistentry>
     <varlistentry>
      <term><option>--json-object=trailer|obj[,gen]</option></term>
      <listitem>
       <para>
        This option is repeatable. If specified, only specified
        objects will be shown in the
        &ldquo;<literal>objects</literal>&rdquo; key of the JSON
        output. If absent, all objects will be shown.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term><option>--check</option></term>
      <listitem>
       <para>
        Checks file structure and well as encryption, linearization,
        and encoding of stream data.  A file for which
        <option>--check</option> reports no errors may still have
        errors in stream data content but should otherwise be
        structurally sound.  If <option>--check</option> any errors,
        qpdf will exit with a status of 2.  There are some recoverable
        conditions that <option>--check</option> detects.  These are
        issued as warnings instead of errors.  If qpdf finds no errors
        but finds warnings, it will exit with a status of 3 (as of
        version&nbsp;2.0.4). When <option>--check</option> is combined
        with other options, checks are always performed before any
        other options are processed. For erroneous files,
        <option>--check</option> will cause qpdf to attempt to
        recover, after which other options are effectively operating
        on the recovered file. Combining <option>--check</option> with
        other options in this way can be useful for manually
        recovering severely damaged files.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
   </para>
   <para>
    The <option>--raw-stream-data</option> and
    <option>--filtered-stream-data</option> options are ignored unless
    <option>--show-object</option> is given.  Either of these options
    will cause the stream data to be written to standard output.  In
    order to avoid commingling of stream data with other output, it is
    recommend that these objects not be combined with other
    test/inspection options.
   </para>
   <para>
    If <option>--filtered-stream-data</option> is given and
    <option>--normalize-content=y</option> is also given, qpdf will
    attempt to normalize the stream data as if it is a page content
    stream.  This attempt will be made even if it is not a page
    content stream, in which case it will produce unusable results.
   </para>
  </sect1>
  <sect1 id="ref.unicode-passwords">
   <title>Unicode Passwords</title>
   <para>
    At the library API level, all methods that perform encryption and
    decryption interpret passwords as strings of bytes. It is up to
    the caller to ensure that they are appropriately encoded. Starting
    with qpdf version 8.4.0, qpdf will attempt to make this easier for
    you when interact with qpdf via its command line interface. The
    PDF specification requires passwords used to encrypt files with
    40-bit or 128-bit encryption to be encoded with PDF Doc encoding.
    This encoding is a single-byte encoding that supports ISO-Latin-1
    and a handful of other commonly used characters. It has a large
    overlap with Windows ANSI but is not exactly the same. There is
    generally not a way to provide PDF Doc encoded strings on the
    command line. As such, qpdf versions prior to 8.4.0 would often
    create PDF files that couldn't be opened with other software when
    given a password with non-ASCII characters to encrypt a file with
    40-bit or 128-bit encryption. Starting with qpdf 8.4.0, qpdf
    recognizes the encoding of the parameter and transcodes it as
    needed. The rest of this section provides the details about
    exactly how qpdf behaves. Most users will not need to know this
    information, but it might be useful if you have been working
    around qpdf's old behavior or if you are using qpdf to generate
    encrypted files for testing other PDF software.
   </para>
   <para>
    A note about Windows: when qpdf builds, it attempts to determine
    what it has to do to use <function>wmain</function> instead of
    <function>main</function> on Windows. The
    <function>wmain</function> function is an alternative entry point
    that receives all arguments as UTF-16-encoded strings. When qpdf
    starts up this way, it converts all the strings to UTF-8 encoding
    and then invokes the regular main. This means that, as far as qpdf
    is concerned, it receives its command-line arguments with UTF-8
    encoding, just as it would in any modern Linux or UNIX
    environment.
   </para>
   <para>
    If a file is being encrypted with 40-bit or 128-bit encryption and
    the supplied password is not a valid UTF-8 string, qpdf will fall
    back to the behavior of interpreting the password as a string of
    bytes. If you have old scripts that encrypt files by passing the
    output of <command>iconv</command> to qpdf, you no longer need to
    do that, but if you do, qpdf should still work. The only exception
    would be for the extremely unlikely case of a password that is
    encoded with a single-byte encoding but also happens to be valid
    UTF-8. Such a password would contain strings of even numbers of
    characters that alternate between accented letters and symbols. In
    the extremely unlikely event that you are intentionally using such
    passwords and qpdf is thwarting you by interpreting them as UTF-8,
    you can use <option>--password-mode=bytes</option> to suppress
    qpdf's automatic behavior.
   </para>
   <para>
    The <option>--password-mode</option> option, as described earlier
    in this chapter, can be used to change qpdf's interpretation of
    supplied passwords. There are very few reasons to use this option.
    One would be the unlikely case described in the previous paragraph
    in which the supplied password happens to be valid UTF-8 but isn't
    supposed to be UTF-8. Your best bet would be just to provide the
    password as a valid UTF-8 string, but you could also use
    <option>--password-mode=bytes</option>. Another reason to use
    <option>--password-mode=bytes</option> would be to intentionally
    generate PDF files encrypted with passwords that are not properly
    encoded. The qpdf test suite does this to generate invalid files
    for the purpose of testing its password recovery capability. If
    you were trying to create intentionally incorrect files for a
    similar purposes, the <option>bytes</option> password mode can
    enable you to do this.
   </para>
   <para>
    When qpdf attempts to decrypt a file with a password that contains
    non-ASCII characters, it will generate a list of alternative
    passwords by attempting to interpret the password as each of a
    handful of different coding systems and then transcode them to the
    required format. This helps to compensate for the supplied
    password being given in the wrong coding system, such as would
    happen if you used the <command>iconv</command> workaround that
    was previously needed. It also generates passwords by doing the
    reverse operation: translating from correct in incorrect encoding
    of the password. This would enable qpdf to decrypt files using
    passwords that were improperly encoded by whatever software
    encrypted the files, including older versions of qpdf invoked
    without properly encoded passwords. The combination of these two
    recovery methods should make qpdf transparently open most
    encrypted files with the password supplied correctly but in the
    wrong coding system. There are no real downsides to this behavior,
    but if you don't want qpdf to do this, you can use the
    <option>--suppress-password-recovery</option> option. One reason
    to do that is to ensure that you know the exact password that was
    used to encrypt the file.
   </para>
   <para>
    With these changes, qpdf now generates compliant passwords in most
    cases. There are still some exceptions. In particular, the PDF
    specification directs compliant writers to normalize Unicode
    passwords and to perform certain transformations on passwords with
    bidirectional text. Implementing this functionality requires using
    a real Unicode library like ICU. If a client application that uses
    qpdf wants to do this, the qpdf library will accept the resulting
    passwords, but qpdf will not perform these transformations itself.
    It is possible that this will be addressed in a future version of
    qpdf. The <classname>QPDFWriter</classname> methods that enable
    encryption on the output file accept passwords as strings of
    bytes.
   </para>
   <para>
    Please note that the <option>--password-is-hex-key</option> option
    is unrelated to all this. This flag bypasses the normal process of
    going from password to encryption string entirely, allowing the
    raw encryption key to be specified directly. This is useful for
    forensic purposes or for brute-force recovery of files with
    unknown passwords.
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.qdf">
  <title>QDF Mode</title>
  <para>
   In QDF mode, qpdf creates PDF files in what we call <firstterm>QDF
   form</firstterm>.  A PDF file in QDF form, sometimes called a QDF
   file, is a completely valid PDF file that has
   <literal>%QDF-1.0</literal> as its third line (after the pdf header
   and binary characters) and has certain other characteristics.  The
   purpose of QDF form is to make it possible to edit PDF files, with
   some restrictions, in an ordinary text editor.  This can be very
   useful for experimenting with different PDF constructs or for
   making one-off edits to PDF files (though there are other reasons
   why this may not always work). Note that QDF mode does not support
   linearized files. If you enable linearization, QDF mode is
   automatically disabled.
  </para>
  <para>
   It is ordinarily very difficult to edit PDF files in a text editor
   for two reasons: most meaningful data in PDF files is compressed,
   and PDF files are full of offset and length information that makes
   it hard to add or remove data.  A QDF file is organized in a manner
   such that, if edits are kept within certain constraints, the
   <command>fix-qdf</command> program, distributed with qpdf, is able
   to restore edited files to a correct state.  The
   <command>fix-qdf</command> program takes no command-line
   arguments.  It reads a possibly edited QDF file from standard input
   and writes a repaired file to standard output.
  </para>
  <para>
   The following attributes characterize a QDF file:
   <itemizedlist>
    <listitem>
     <para>
      All objects appear in numerical order in the PDF file, including
      when objects appear in object streams.
     </para>
    </listitem>
    <listitem>
     <para>
      Objects are printed in an easy-to-read format, and all line
      endings are normalized to UNIX line endings.
     </para>
    </listitem>
    <listitem>
     <para>
      Unless specifically overridden, streams appear uncompressed
      (when qpdf supports the filters and they are compressed with a
      non-lossy compression scheme), and most content streams are
      normalized (line endings are converted to just a UNIX-style
      linefeeds).
     </para>
    </listitem>
    <listitem>
     <para>
      All streams lengths are represented as indirect objects, and the
      stream length object is always the next object after the stream.
      If the stream data does not end with a newline, an extra newline
      is inserted, and a special comment appears after the stream
      indicating that this has been done.
     </para>
    </listitem>
    <listitem>
     <para>
      If the PDF file contains object streams, if object stream
      <emphasis>n</emphasis> contains <emphasis>k</emphasis> objects,
      those objects are numbered from <emphasis>n+1</emphasis> through
      <emphasis>n+k</emphasis>, and the object number/offset pairs
      appear on a separate line for each object.  Additionally, each
      object in the object stream is preceded by a comment indicating
      its object number and index.  This makes it very easy to find
      objects in object streams.
     </para>
    </listitem>
    <listitem>
     <para>
      All beginnings of objects, <literal>stream</literal> tokens,
      <literal>endstream</literal> tokens, and
      <literal>endobj</literal> tokens appear on lines by themselves.
      A blank line follows every <literal>endobj</literal> token.
     </para>
    </listitem>
    <listitem>
     <para>
      If there is a cross-reference stream, it is unfiltered.
     </para>
    </listitem>
    <listitem>
     <para>
      Page dictionaries and page content streams are marked with
      special comments that make them easy to find.
     </para>
    </listitem>
    <listitem>
     <para>
      Comments precede each object indicating the object number of the
      corresponding object in the original file.
     </para>
    </listitem>
   </itemizedlist>
  </para>
  <para>
   When editing a QDF file, any edits can be made as long as the above
   constraints are maintained.  This means that you can freely edit a
   page's content without worrying about messing up the QDF file.  It
   is also possible to add new objects so long as those objects are
   added after the last object in the file or subsequent objects are
   renumbered.  If a QDF file has object streams in it, you can always
   add the new objects before the xref stream and then change the
   number of the xref stream, since nothing generally ever references
   it by number.
  </para>
  <para>
   It is not generally practical to remove objects from QDF files
   without messing up object numbering, but if you remove all
   references to an object, you can run qpdf on the file (after
   running <command>fix-qdf</command>), and qpdf will omit the
   now-orphaned object.
  </para>
  <para>
   When <command>fix-qdf</command> is run, it goes through the file
   and recomputes the following parts of the file:
   <itemizedlist>
    <listitem>
     <para>
      the <literal>/N</literal>, <literal>/W</literal>, and
      <literal>/First</literal> keys of all object stream dictionaries
     </para>
    </listitem>
    <listitem>
     <para>
      the pairs of numbers representing object numbers and offsets of
      objects in object streams
     </para>
    </listitem>
    <listitem>
     <para>
      all stream lengths
     </para>
    </listitem>
    <listitem>
     <para>
      the cross-reference table or cross-reference stream
     </para>
    </listitem>
    <listitem>
     <para>
      the offset to the cross-reference table or cross-reference
      stream following the <literal>startxref</literal> token
     </para>
    </listitem>
   </itemizedlist>
  </para>
 </chapter>
 <chapter id="ref.using-library">
  <title>Using the QPDF Library</title>
  <sect1 id="ref.using.from-cxx">
   <title>Using QPDF from C++</title>
   <para>
    The source tree for the qpdf package has an
    <filename>examples</filename> directory that contains a few
    example programs.  The <filename>qpdf/qpdf.cc</filename> source
    file also serves as a useful example since it exercises almost all
    of the qpdf library's public interface.  The best source of
    documentation on the library itself is reading comments in
    <filename>include/qpdf/QPDF.hh</filename>,
    <filename>include/qpdf/QPDFWriter.hh</filename>, and
    <filename>include/qpdf/QPDFObjectHandle.hh</filename>.
   </para>
   <para>
    All header files are installed in the <filename>include/qpdf</filename> directory.  It
    is recommend that you use <literal>#include
    &lt;qpdf/QPDF.hh&gt;</literal> rather than adding
    <filename>include/qpdf</filename> to your include path.
   </para>
   <para>
    When linking against the qpdf static library, you may also need to
    specify <literal>-lz -ljpeg</literal> on your link command. If
    your system understands how to read libtool
    <filename>.la</filename> files, this may not be necessary.
   </para>
   <para>
    The qpdf library is safe to use in a multithreaded program, but no
    individual <type>QPDF</type> object instance (including
    <type>QPDF</type>, <type>QPDFObjectHandle</type>, or
    <type>QPDFWriter</type>) can be used in more than one thread at a
    time.  Multiple threads may simultaneously work with different
    instances of these and all other QPDF objects.
   </para>
  </sect1>
  <sect1 id="ref.using.other-languages">
   <title>Using QPDF from other languages</title>
   <para>
    The qpdf library is implemented in C++, which makes it hard to use
    directly in other languages. There are a few things that can help.
   </para>
   <variablelist>
    <varlistentry>
     <term>&ldquo;C&rdquo;</term>
     <listitem>
      <para>
       The qpdf library includes a &ldquo;C&rdquo; language interface
       that provides a subset of the overall capabilities. The header
       file <filename>qpdf/qpdf-c.h</filename> includes information
       about its use. As long as you use a C++ linker, you can link C
       programs with qpdf and use the C API. For languages that can
       directly load methods from a shared library, the C API can also
       be useful. People have reported success using the C API from
       other languages on Windows by directly calling functions in the
       DLL.
      </para>
     </listitem>
    </varlistentry>
    <varlistentry>
     <term>Python</term>
     <listitem>
      <para>
       A Python module called <ulink
       url="https://pypi.org/project/pikepdf/">pikepdf</ulink>
       provides a clean and highly functional set of Python bindings
       to the qpdf library. Using pikepdf, you can work with PDF files
       in a natural way and combine qpdf's capabilities with other
       functionality provided by Python's rich standard library and
       available modules.
      </para>
     </listitem>
    </varlistentry>
    <varlistentry>
     <term>Other Languages</term>
     <listitem>
      <para>
       Starting with version 8.3.0, the <command>qpdf</command>
       command-line tool can produce a JSON representation of the PDF
       file's non-content data. This can facilitate interacting
       programmatically with PDF files through qpdf's command line
       interface. For more information, please see <xref
       linkend="ref.json"/>.
      </para>
     </listitem>
    </varlistentry>
   </variablelist>
  </sect1>
  <sect1 id="ref.unicode-files">
   <title>A Note About Unicode File Names</title>
   <para>
    When strings are passed to qpdf library routines either as
    <literal>char*</literal> or as <literal>std::string</literal>,
    they are treated as byte arrays except where otherwise noted. When
    Unicode is desired, qpdf wants UTF-8 unless otherwise noted in
    comments in header files. In modern UNIX/Linux environments, this
    generally does the right thing. In Windows, it's a bit more
    complicated. Starting in qpdf 8.4.0, passwords that contain
    Unicode characters are handled much better, and starting in qpdf
    8.4.1, the library attempts to properly handle Unicode characters
    in filenames. In particular, in Windows, if a UTF-8 encoded string
    is used as a filename in either <classname>QPDF</classname> or
    <classname>QPDFWriter</classname>, it is internally converted to
    <literal>wchar_t*</literal>, and Unicode-aware Windows APIs are
    used. As such, qpdf will generally operate properly on files with
    non-ASCII characters in their names as long as the filenames are
    UTF-8 encoded for passing into the qpdf library API, but there are
    still some rough edges, such as the encoding of the filenames in
    error messages our CLI output messages. Patches or bug reports are
    welcome for any continuing issues with Unicode file names in
    Windows.
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.json">
  <title>QPDF JSON</title>
  <sect1 id="ref.json-overview">
   <title>Overview</title>
   <para>
    Beginning with qpdf version 8.3.0, the <command>qpdf</command>
    command-line program can produce a JSON representation of the
    non-content data in a PDF file. It includes a dump in JSON format
    of all objects in the PDF file excluding the content of streams.
    This JSON representation makes it very easy to look in detail at
    the structure of a given PDF file, and it also provides a great way
    to work with PDF files programmatically from the command-line in
    languages that can't call or link with the qpdf library directly.
    Note that stream data can be extracted from PDF files using other
    qpdf command-line options.
   </para>
  </sect1>
  <sect1 id="ref.json-guarantees">
   <title>JSON Guarantees</title>
   <para>
    The qpdf JSON representation includes a JSON serialization of the
    raw objects in the PDF file as well as some computed information in
    a more easily extracted format. QPDF provides some guarantees about
    its JSON format. These guarantees are designed to simplify the
    experience of a developer working with the JSON format.
    <variablelist>
     <varlistentry>
      <term>Compatibility</term>
      <listitem>
       <para>
        The top-level JSON object output is a dictionary. The JSON
        output contains various nested dictionaries and arrays. With
        the exception of dictionaries that are populated by the fields
        of objects from the file, all instances of a dictionary are
        guaranteed to have exactly the same keys. Future versions of
        qpdf are free to add additional keys but not to remove keys or
        change the type of object that a key points to. The qpdf
        program validates this guarantee, and in the unlikely event
        that a bug in qpdf should cause it to generate data that
        doesn't conform to this rule, it will ask you to file a bug
        report.
       </para>
       <para>
        The top-level JSON structure contains a
        &ldquo;<literal>version</literal>&rdquo; key whose value is
        simple integer. The value of the <literal>version</literal> key
        will be incremented if a non-compatible change is made. A
        non-compatible change would be any change that involves removal
        of a key, a change to the format of data pointed to by a key,
        or a semantic change that requires a different interpretation
        of a previously existing key. A strong effort will be made to
        avoid breaking compatibility.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term>Documentation</term>
      <listitem>
       <para>
        The <command>qpdf</command> command can be invoked with the
        <option>--json-help</option> option. This will output a JSON
        structure that has the same structure as the JSON output that
        qpdf generates, except that each field in the help output is a
        description of the corresponding field in the JSON output. The
        specific guarantees are as follows:
        <itemizedlist>
         <listitem>
          <para>
           A dictionary in the help output means that the corresponding
           location in the actual JSON output is also a dictionary with
           exactly the same keys; that is, no keys present in help are
           absent in the real output, and no keys will be present in
           the real output that are not in help.
          </para>
         </listitem>
         <listitem>
          <para>
           A string in the help output is a description of the item
           that appears in the corresponding location of the actual
           output. The corresponding output can have any format.
          </para>
         </listitem>
         <listitem>
          <para>
           An array in the help output always contains a single
           element. It indicates that the corresponding location in the
           actual output is also an array, and that each element of the
           array has whatever format is implied by the single element
           of the help output's array.
          </para>
         </listitem>
        </itemizedlist>
        For example, the help output indicates includes a
        &ldquo;<literal>pagelabels</literal>&rdquo; key whose value is
        an array of one element. That element is a dictionary with keys
        &ldquo;<literal>index</literal>&rdquo; and
        &ldquo;<literal>label</literal>&rdquo;. In addition to
        describing the meaning of those keys, this tells you that the
        actual JSON output will contain a <literal>pagelabels</literal>
        array, each of whose elements is a dictionary that contains an
        <literal>index</literal> key, a <literal>label</literal> key,
        and no other keys.
       </para>
      </listitem>
     </varlistentry>
     <varlistentry>
      <term>Directness and Simplicity</term>
      <listitem>
       <para>
        The JSON output contains the value of every object in the file,
        but it also contains some processed data. This is analogous to
        how qpdf's library interface works. The processed data is
        similar to the helper functions in that it allows you to look
        at certain aspects of the PDF file without having to understand
        all the nuances of the PDF specification, while the raw objects
        allow you to mine the PDF for anything that the higher-level
        interfaces are lacking.
       </para>
      </listitem>
     </varlistentry>
    </variablelist>
   </para>
  </sect1>
  <sect1 id="json.limitations">
   <title>Limitations of JSON Representation</title>
   <para>
    There are a few limitations to be aware of with the JSON structure:
    <itemizedlist>
     <listitem>
      <para>
       Strings, names, and indirect object references in the original
       PDF file are all converted to strings in the JSON
       representation. In the case of a &ldquo;normal&rdquo; PDF file,
       you can tell the difference because a name starts with a slash
       (<literal>/</literal>), and an indirect object reference looks
       like <literal>n n R</literal>, but if there were to be a string
       that looked like a name or indirect object reference, there
       would be no way to tell this from the JSON output. Note that
       there are certain cases where you know for sure what something
       is, such as knowing that dictionary keys in objects are always
       names and that certain things in the higher-level computed data
       are known to contain indirect object references.
      </para>
     </listitem>
     <listitem>
      <para>
       The JSON format doesn't support binary data very well. Mostly
       the details are not important, but they are presented here for
       information. When qpdf outputs a string in the JSON
       representation, it converts the string to UTF-8, assuming usual
       PDF string semantics. Specifically, if the original string is
       UTF-16, it is converted to UTF-8. Otherwise, it is assumed to
       have PDF doc encoding, and is converted to UTF-8 with that
       assumption. This causes strange things to happen to binary
       strings. For example, if you had the binary string
       <literal>&lt;038051&gt;</literal>, this would be output to the
       JSON as <literal>\u0003•Q</literal> because
       <literal>03</literal> is not a printable character and
       <literal>80</literal> is the bullet character in PDF doc
       encoding and is mapped to the Unicode value
       <literal>2022</literal>. Since <literal>51</literal> is
       <literal>Q</literal>, it is output as is. If you wanted to
       convert back from here to a binary string, would have to
       recognize Unicode values whose code points are higher than
       <literal>0xFF</literal> and map those back to their
       corresponding PDF doc encoding characters. There is no way to
       tell the difference between a Unicode string that was originally
       encoded as UTF-16 or one that was converted from PDF doc
       encoding. In other words, it's best if you don't try to use the
       JSON format to extract binary strings from the PDF file, but if
       you really had to, it could be done. Note that qpdf's
       <option>--show-object</option> option does not have this
       limitation and will reveal the string as encoded in the original
       file.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
  <sect1 id="json.considerations">
   <title>JSON: Special Considerations</title>
   <para>
    For the most part, the built-in JSON help tells you everything you
    need to know about the JSON format, but there are a few
    non-obvious things to be aware of:
    <itemizedlist>
     <listitem>
      <para>
       While qpdf guarantees that keys present in the help will be
       present in the output, those fields may be null or empty if the
       information is not known or absent in the file. Also, if you
       specify <option>--json-keys</option>, the keys that are not
       listed will be excluded entirely except for those that
       <option>--json-help</option> says are always present.
      </para>
     </listitem>
     <listitem>
      <para>
       In a few places, there are keys with names containing
       <literal>pageposfrom1</literal>. The values of these keys are
       null or an integer. If an integer, they point to a page index
       within the file numbering from 1. Note that JSON indexes from
       0, and you would also use 0-based indexing using the API.
       However, 1-based indexing is easier in this case because the
       command-line syntax for specifying page ranges is 1-based. If
       you were going to write a program that looked through the JSON
       for information about specific pages and then use the
       command-line to extract those pages, 1-based indexing is
       easier. Besides, it's more convenient to subtract 1 from a
       program in a real programming language than it is to add 1 from
       shell code.
      </para>
     </listitem>
     <listitem>
      <para>
       The image information included in the <literal>page</literal>
       section of the JSON output includes the key
       &ldquo;<literal>filterable</literal>&rdquo;. Note that the
       value of this field may depend on the
       <option>--decode-level</option> that you invoke qpdf with. The
       JSON output includes a top-level key
       &ldquo;<literal>parameters</literal>&rdquo; that indicates the
       decode level used for computing whether a stream was
       filterable. For example, jpeg images will be shown as not
       filterable by default, but they will be shown as filterable if
       you run <command>qpdf --json --decode-level=all</command>.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.design">
  <title>Design and Library Notes</title>
  <sect1 id="ref.design.intro">
   <title>Introduction</title>
   <para>
    This section was written prior to the implementation of the qpdf
    package and was subsequently modified to reflect the
    implementation.  In some cases, for purposes of explanation, it
    may differ slightly from the actual implementation.  As always,
    the source code and test suite are authoritative.  Even if there
    are some errors, this document should serve as a road map to
    understanding how this code works.
   </para>
   <para>
    In general, one should adhere strictly to a specification when
    writing but be liberal in reading. This way, the product of our
    software will be accepted by the widest range of other programs,
    and we will accept the widest range of input files. This library
    attempts to conform to that philosophy whenever possible but also
    aims to provide strict checking for people who want to validate
    PDF files. If you don't want to see warnings and are trying to
    write something that is tolerant, you can call
    <literal>setSuppressWarnings(true)</literal>. If you want to fail
    on the first error, you can call
    <literal>setAttemptRecovery(false)</literal>. The default behavior
    is to generating warnings for recoverable problems. Note that
    recovery will not always produce the desired results even if it is
    able to get through the file. Unlike most other PDF files that
    produce generic warnings such as &ldquo;This file is
    damaged,&rdquo;, qpdf generally issues a detailed error message
    that would be most useful to a PDF developer. This is by design as
    there seems to be a shortage of PDF validation tools out there.
    This was, in fact, one of the major motivations behind the initial
    creation of qpdf.
   </para>
  </sect1>
  <sect1 id="ref.design-goals">
   <title>Design Goals</title>
   <para>
    The QPDF package includes support for reading and rewriting PDF
    files. It aims to hide from the user details involving object
    locations, modified (appended) PDF files, the
    directness/indirectness of objects, and stream filters including
    encryption. It does not aim to hide knowledge of the object
    hierarchy or content stream contents. Put another way, a user of
    the qpdf library is expected to have knowledge about how PDF files
    work, but is not expected to have to keep track of bookkeeping
    details such as file positions.
   </para>
   <para>
    A user of the library never has to care whether an object is
    direct or indirect, though it is possible to determine whether an
    object is direct or not if this information is needed. All access
    to objects deals with this transparently. All memory management
    details are also handled by the library.
   </para>
   <para>
    The <classname>PointerHolder</classname> object is used internally
    by the library to deal with memory management. This is basically a
    smart pointer object very similar in spirit to C++-11's
    <classname>std::shared_ptr</classname> object, but predating it by
    several years. This library also makes use of a technique for
    giving fine-grained access to methods in one class to other
    classes by using public subclasses with friends and only private
    members that in turn call private methods of the containing class.
    See <classname>QPDFObjectHandle::Factory</classname> as an
    example.
   </para>
   <para>
    The top-level qpdf class is <classname>QPDF</classname>.  A
    <classname>QPDF</classname> object represents a PDF file.  The
    library provides methods for both accessing and mutating PDF
    files.
   </para>
   <para>
    The primary class for interacting with PDF objects is
    <classname>QPDFObjectHandle</classname>. Instances of this class
    can be passed around by value, copied, stored in containers, etc.
    with very low overhead. Instances of
    <classname>QPDFObjectHandle</classname> created by reading from a
    file will always contain a reference back to the
    <classname>QPDF</classname> object from which they were created. A
    <classname>QPDFObjectHandle</classname> may be direct or indirect.
    If indirect, the <classname>QPDFObject</classname> the
    <classname>PointerHolder</classname> initially points to is a null
    pointer. In this case, the first attempt to access the underlying
    <classname>QPDFObject</classname> will result in the
    <classname>QPDFObject</classname> being resolved via a call to the
    referenced <classname>QPDF</classname> instance. This makes it
    essentially impossible to make coding errors in which certain
    things will work for some PDF files and not for others based on
    which objects are direct and which objects are indirect.
   </para>
   <para>
    Instances of <classname>QPDFObjectHandle</classname> can be
    directly created and modified using static factory methods in the
    <classname>QPDFObjectHandle</classname> class. There are factory
    methods for each type of object as well as a convenience method
    <function>QPDFObjectHandle::parse</function> that creates an
    object from a string representation of the object.  Existing
    instances of <classname>QPDFObjectHandle</classname> can also be
    modified in several ways.  See comments in
    <filename>QPDFObjectHandle.hh</filename> for details.
   </para>
   <para>
    An instance of <classname>QPDF</classname> is constructed by using
    the class's default constructor.  If desired, the
    <classname>QPDF</classname> object may be configured with various
    methods that change its default behavior.  Then the
    <function>QPDF::processFile()</function> method is passed the name
    of a PDF file, which permanently associates the file with that
    QPDF object.  A password may also be given for access to
    password-protected files.  QPDF does not enforce encryption
    parameters and will treat user and owner passwords equivalently.
    Either password may be used to access an encrypted file.
    <footnote>
     <para>
      As pointed out earlier, the intention is not for qpdf to be used
      to bypass security on files. but as any open source PDF consumer
      may be easily modified to bypass basic PDF document security,
      and qpdf offers may transformations that can do this as well,
      there seems to be little point in the added complexity of
      conditionally enforcing document security.
     </para>
    </footnote>
    <classname>QPDF</classname> will allow recovery of a user password
    given an owner password.  The input PDF file must be seekable.
    (Output files written by <classname>QPDFWriter</classname> need
    not be seekable, even when creating linearized files.)  During
    construction, <classname>QPDF</classname> validates the PDF file's
    header, and then reads the cross reference tables and trailer
    dictionaries.  The <classname>QPDF</classname> class keeps only
    the first trailer dictionary though it does read all of them so it
    can check the <literal>/Prev</literal> key.
    <classname>QPDF</classname> class users may request the root
    object and the trailer dictionary specifically.  The cross
    reference table is kept private.  Objects may then be requested by
    number of by walking the object tree.
   </para>
   <para>
    When a PDF file has a cross-reference stream instead of a
    cross-reference table and trailer, requesting the document's
    trailer dictionary returns the stream dictionary from the
    cross-reference stream instead.
   </para>
   <para>
    There are some convenience routines for very common operations
    such as walking the page tree and returning a vector of all page
    objects. For full details, please see the header files
    <filename>QPDF.hh</filename> and
    <filename>QPDFObjectHandle.hh</filename>. There are also some
    additional helper classes that provide higher level API functions
    for certain document constructions. These are discussed in <xref
    linkend="ref.helper-classes"/>.
   </para>
  </sect1>
  <sect1 id="ref.helper-classes">
   <title>Helper Classes</title>
   <para>
    QPDF version 8.1 introduced the concept of helper classes. Helper
    classes are intended to contain higher level APIs that allow
    developers to work with certain document constructs at an
    abstraction level above that of
    <classname>QPDFObjectHandle</classname> while staying true to
    qpdf's philosophy of not hiding document structure from the
    developer. As with qpdf in general, the goal is take away some of
    the more tedious bookkeeping aspects of working with PDF files,
    not to remove the need for the developer to understand how the PDF
    construction in question works. The driving factor behind the
    creation of helper classes was to allow the evolution of higher
    level interfaces in qpdf without polluting the interfaces of the
    main top-level classes <classname>QPDF</classname> and
    <classname>QPDFObjectHandle</classname>.
   </para>
   <para>
    There are two kinds of helper classes:
    <emphasis>document</emphasis> helpers and
    <emphasis>object</emphasis> helpers. Document helpers are
    constructed with a reference to a <classname>QPDF</classname>
    object and provide methods for working with structures that are at
    the document level. Object helpers are constructed with an
    instance of a <classname>QPDFObjectHandle</classname> and provide
    methods for working with specific types of objects.
   </para>
   <para>
    Examples of document helpers include
    <classname>QPDFPageDocumentHelper</classname>, which contains
    methods for operating on the document's page trees, such as
    enumerating all pages of a document and adding and removing pages;
    and <classname>QPDFAcroFormDocumentHelper</classname>, which
    contains document-level methods related to interactive forms, such
    as enumerating form fields and creating mappings between form
    fields and annotations.
   </para>
   <para>
    Examples of object helpers include
    <classname>QPDFPageObjectHelper</classname> for performing
    operations on pages such as page rotation and some operations on
    content streams, <classname>QPDFFormFieldObjectHelper</classname>
    for performing operations related to interactive form fields, and
    <classname>QPDFAnnotationObjectHelper</classname> for working with
    annotations.
   </para>
   <para>
    It is always possible to retrieve the underlying
    <classname>QPDF</classname> reference from a document helper and
    the underlying <classname>QPDFObjectHandle</classname> reference
    from an object helper. Helpers are designed to be helpers, not
    wrappers. The intention is that, in general, it is safe to freely
    intermix operations that use helpers with operations that use the
    underlying objects. Document and object helpers do not attempt to
    provide a complete interface for working with the things they are
    helping with, nor do they attempt to encapsulate underlying
    structures. They just provide a few methods to help with
    error-prone, repetitive, or complex tasks. In some cases, a helper
    object may cache some information that is expensive to gather. In
    such cases, the helper classes are implemented so that their own
    methods keep the cache consistent, and the header file will
    provide a method to invalidate the cache and a description of what
    kinds of operations would make the cache invalid. If in doubt, you
    can always discard a helper class and create a new one with the
    same underlying objects, which will ensure that you have discarded
    any stale information.
   </para>
   <para>
    By Convention, document helpers are called
    <classname>QPDFSomethingDocumentHelper</classname> and are derived
    from <classname>QPDFDocumentHelper</classname>, and object helpers
    are called <classname>QPDFSomethingObjectHelper</classname> and
    are derived from <classname>QPDFObjectHelper</classname>. For
    details on specific helpers, please see their header files. You
    can find them by looking at
    <filename>include/qpdf/QPDF*DocumentHelper.hh</filename> and
    <filename>include/qpdf/QPDF*ObjectHelper.hh</filename>.
   </para>
   <para>
    In order to avoid creation of circular dependencies, the following
    general guidelines are followed with helper classes:
    <itemizedlist>
     <listitem>
      <para>
       Core class interfaces do not know about helper classes. For
       example, no methods of <classname>QPDF</classname> or
       <classname>QPDFObjectHandle</classname> will include helper
       classes in their interfaces.
      </para>
     </listitem>
     <listitem>
      <para>
       Interfaces of object helpers will usually not use document
       helpers in their interfaces. This is because it is much more
       useful for document helpers to have methods that return object
       helpers. Most operations in PDF files start at the document
       level and go from there to the object level rather than the
       other way around. It can sometimes be useful to map back from
       object-level structures to document-level structures. If there
       is a desire to do this, it will generally be provided by a
       method in the document helper class.
      </para>
     </listitem>
     <listitem>
      <para>
       Most of the time, object helpers don't know about other object
       helpers. However, in some cases, one type of object may be a
       container for another type of object, in which case it may make
       sense for the outer object to know about the inner object. For
       example, there are methods in the
       <classname>QPDFPageObjectHelper</classname> that know
       <classname>QPDFAnnotationObjectHelper</classname> because
       references to annotations are contained in page dictionaries.
      </para>
     </listitem>
     <listitem>
      <para>
       Any helper or core library class may use helpers in their
       implementations.
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Prior to qpdf version 8.1, higher level interfaces were added as
    &ldquo;convenience functions&rdquo; in either
    <classname>QPDF</classname> or
    <classname>QPDFObjectHandle</classname>. For compatibility, older
    convenience functions for operating with pages will remain in
    those classes even as alternatives are provided in helper classes.
    Going forward, new higher level interfaces will be provided using
    helper classes.
   </para>
  </sect1>
  <sect1 id="ref.implementation-notes">
   <title>Implementation Notes</title>
   <para>
    This section contains a few notes about QPDF's internal
    implementation, particularly around what it does when it first
    processes a file. This section is a bit of a simplification of
    what it actually does, but it could serve as a starting point to
    someone trying to understand the implementation. There is nothing
    in this section that you need to know to use the qpdf library.
   </para>
   <para>
    <classname>QPDFObject</classname> is the basic PDF Object class.
    It is an abstract base class from which are derived classes for
    each type of PDF object.  Clients do not interact with Objects
    directly but instead interact with
    <classname>QPDFObjectHandle</classname>.
   </para>
   <para>
    When the <classname>QPDF</classname> class creates a new object,
    it dynamically allocates the appropriate type of
    <classname>QPDFObject</classname> and immediately hands the
    pointer to an instance of <classname>QPDFObjectHandle</classname>.
    The parser reads a token from the current file position. If the
    token is a not either a dictionary or array opener, an object is
    immediately constructed from the single token and the parser
    returns. Otherwise, the parser iterates in a special mode in which
    it accumulates objects until it finds a balancing closer. During
    this process, the &ldquo;<literal>R</literal>&rdquo; keyword is
    recognized and an indirect <classname>QPDFObjectHandle</classname>
    may be constructed.
   </para>
   <para>
    The <function>QPDF::resolve()</function> method, which is used to
    resolve an indirect object, may be invoked from the
    <classname>QPDFObjectHandle</classname> class.  It first checks a
    cache to see whether this object has already been read.  If not,
    it reads the object from the PDF file and caches it.  It the
    returns the resulting <classname>QPDFObjectHandle</classname>.
    The calling object handle then replaces its
    <classname>PointerHolder&lt;QDFObject&gt;</classname> with the one
    from the newly returned <classname>QPDFObjectHandle</classname>.
    In this way, only a single copy of any direct object need exist
    and clients can access objects transparently without knowing
    caring whether they are direct or indirect objects.  Additionally,
    no object is ever read from the file more than once.  That means
    that only the portions of the PDF file that are actually needed
    are ever read from the input file, thus allowing the qpdf package
    to take advantage of this important design goal of PDF files.
   </para>
   <para>
    If the requested object is inside of an object stream, the object
    stream itself is first read into memory.  Then the tokenizer reads
    objects from the memory stream based on the offset information
    stored in the stream.  Those individual objects are cached, after
    which the temporary buffer holding the object stream contents are
    discarded.  In this way, the first time an object in an object
    stream is requested, all objects in the stream are cached.
   </para>
   <para>
    The following example should clarify how
    <classname>QPDF</classname> processes a simple file.
    <itemizedlist>
     <listitem>
      <para>
       Client constructs <classname>QPDF</classname>
       <varname>pdf</varname> and calls
       <function>pdf.processFile("a.pdf");</function>.
      </para>
     </listitem>
     <listitem>
      <para>
       The <classname>QPDF</classname> class checks the beginning of
       <filename>a.pdf</filename> for a PDF header. It then reads the
       cross reference table mentioned at the end of the file,
       ensuring that it is looking before the last
       <literal>%%EOF</literal>. After getting to
       <literal>trailer</literal> keyword, it invokes the parser.
      </para>
     </listitem>
     <listitem>
      <para>
       The parser sees &ldquo;<literal>&lt;&lt;</literal>&rdquo;, so
       it calls itself recursively in dictionary creation mode.
      </para>
     </listitem>
     <listitem>
      <para>
       In dictionary creation mode, the parser keeps accumulating
       objects until it encounters
       &ldquo;<literal>&gt;&gt;</literal>&rdquo;.  Each object that is
       read is pushed onto a stack.  If
       &ldquo;<literal>R</literal>&rdquo; is read, the last two
       objects on the stack are inspected.  If they are integers, they
       are popped off the stack and their values are used to construct
       an indirect object handle which is then pushed onto the stack.
       When &ldquo;<literal>&gt;&gt;</literal>&rdquo; is finally read,
       the stack is converted into a
       <classname>QPDF_Dictionary</classname> which is placed in a
       <classname>QPDFObjectHandle</classname> and returned.
      </para>
     </listitem>
     <listitem>
      <para>
       The resulting dictionary is saved as the trailer dictionary.
      </para>
     </listitem>
     <listitem>
      <para>
       The <literal>/Prev</literal> key is searched.  If present,
       <classname>QPDF</classname> seeks to that point and repeats
       except that the new trailer dictionary is not saved.  If
       <literal>/Prev</literal> is not present, the initial parsing
       process is complete.
      </para>
      <para>
       If there is an encryption dictionary, the document's encryption
       parameters are initialized.
      </para>
     </listitem>
     <listitem>
      <para>
       The client requests root object.  The
       <classname>QPDF</classname> class gets the value of root key
       from trailer dictionary and returns it.  It is an unresolved
       indirect <classname>QPDFObjectHandle</classname>.
      </para>
     </listitem>
     <listitem>
      <para>
       The client requests the <literal>/Pages</literal> key from root
       <classname>QPDFObjectHandle</classname>.  The
       <classname>QPDFObjectHandle</classname> notices that it is
       indirect so it asks <classname>QPDF</classname> to resolve it.
       <classname>QPDF</classname> looks in the object cache for an
       object with the root dictionary's object ID and generation
       number.  Upon not seeing it, it checks the cross reference
       table, gets the offset, and reads the object present at that
       offset.  It stores the result in the object cache and returns
       the cached result.  The calling
       <classname>QPDFObjectHandle</classname> replaces its object
       pointer with the one from the resolved
       <classname>QPDFObjectHandle</classname>, verifies that it a
       valid dictionary object, and returns the (unresolved indirect)
       <classname>QPDFObject</classname> handle to the top of the
       Pages hierarchy.
      </para>
      <para>
       As the client continues to request objects, the same process is
       followed for each new requested object.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
  <sect1 id="ref.casting">
   <title>Casting Policy</title>
   <para>
    This section describes the casting policy followed by qpdf's
    implementation.  This is no concern to qpdf's end users and
    largely of no concern to people writing code that uses qpdf, but
    it could be of interest to people who are porting qpdf to a new
    platform or who are making modifications to the code.
   </para>
   <para>
    The C++ code in qpdf is free of old-style casts except where
    unavoidable (e.g. where the old-style cast is in a macro provided
    by a third-party header file).  When there is a need for a cast,
    it is handled, in order of preference, by rewriting the code to
    avoid the need for a cast, calling
    <function>const_cast</function>, calling
    <function>static_cast</function>, calling
    <function>reinterpret_cast</function>, or calling some combination
    of the above.  As a last resort, a compiler-specific
    <literal>#pragma</literal> may be used to suppress a warning that
    we don't want to fix.  Examples may include suppressing warnings
    about the use of old-style casts in code that is shared between C
    and C++ code.
   </para>
   <para>
    The <classname>QIntC</classname> namespace, provided by
    <filename>include/qpdf/QIntC.hh</filename>, implements safe
    functions for converting between integer types. These functions do
    range checking and throw a <type>std::range_error</type>, which is
    subclass of <type>std::runtime_error</type>, if conversion from one
    integer type to another results in loss of information. There are
    many cases in which we have to move between different integer
    types because of incompatible integer types used in interoperable
    interfaces. Some are unavoidable, such as moving between sizes and
    offsets, and others are there because of old code that is too in
    entrenched to be fixable without breaking source compatibility and
    causing pain for users. QPDF is compiled with extra warnings to
    detect conversions with potential data loss, and all such cases
    should be fixed by either using a function from
    <classname>QIntC</classname> or a
    <function>static_cast</function>.
   </para>
   <para>
    When the intention is just to switch the type because of
    exchanging data between incompatible interfaces, use
    <classname>QIntC</classname>. This is the usual case. However,
    there are some cases in which we are explicitly intending to use
    the exact same bit pattern with a different type. This is most
    common when switching between signed and unsigned characters. A
    lot of qpdf's code uses unsigned characters internally, but
    <type>std::string</type> and <type>char</type> are signed. Using
    <function>QIntC::to_char</function> would be wrong for converting
    from unsigned to signed characters because a negative
    <type>char</type> value and the corresponding <type>unsigned
    char</type> value greater than 127 <emphasis>mean the same
    thing</emphasis>. There are also cases in which we use
    <function>static_cast</function> when working with bit fields
    where we are not representing a numerical value but rather a bunch
    of bits packed together in some integer type. Also note that
    <type>size_t</type> and <type>long</type> both typically differ
    between 32-bit and 64-bit environments, so sometimes an explicit
    cast may not be needed to avoid warnings on one platform but may
    be needed on another. A conversion with
    <classname>QIntC</classname> should always be used when the types
    are different even if the underlying size is the same. QPDF's CI
    build builds on 32-bit and 64-bit platforms, and the test suite is
    very thorough, so it is hard to make any of the potential errors
    here without being caught in build or test.
   </para>
   <para>
    Non-const <type>unsigned char*</type> is used in the
    <type>Pipeline</type> interface. The pipeline interface has a
    <function>write</function> call that uses <type>unsigned
    char*</type> without a <type>const</type> qualifier. The main
    reason for this is to support pipelines that make calls to
    third-party libraries, such as zlib, that don't include
    <type>const</type> in their interfaces. Unfortunately, there are
    many places in the code where it is desirable to have <type>const
    char*</type> with pipelines. None of the pipeline implementations
    in qpdf currently modify the data passed to write, and doing so
    would be counter to the intent of <type>Pipeline</type>, but there
    is nothing in the code to prevent this from being done. There are
    places in the code where <function>const_cast</function> is used
    to remove the const-ness of pointers going into
    <type>Pipeline</type>s. This could theoretically be unsafe, but
    there is adequate testing to assert that it is safe and will
    remain safe in qpdf's code.
   </para>
  </sect1>
  <sect1 id="ref.encryption">
   <title>Encryption</title>
   <para>
    Encryption is supported transparently by qpdf.  When opening a PDF
    file, if an encryption dictionary exists, the
    <classname>QPDF</classname> object processes this dictionary using
    the password (if any) provided.  The primary decryption key is
    computed and cached.  No further access is made to the encryption
    dictionary after that time.  When an object is read from a file,
    the object ID and generation of the object in which it is
    contained is always known.  Using this information along with the
    stored encryption key, all stream and string objects are
    transparently decrypted.  Raw encrypted objects are never stored
    in memory.  This way, nothing in the library ever has to know or
    care whether it is reading an encrypted file.
   </para>
   <para>
    An interface is also provided for writing encrypted streams and
    strings given an encryption key.  This is used by
    <classname>QPDFWriter</classname> when it rewrites encrypted
    files.
   </para>
   <para>
    When copying encrypted files, unless otherwise directed, qpdf will
    preserve any encryption in force in the original file.  qpdf can
    do this with either the user or the owner password.  There is no
    difference in capability based on which password is used.  When 40
    or 128 bit encryption keys are used, the user password can be
    recovered with the owner password.  With 256 keys, the user and
    owner passwords are used independently to encrypt the actual
    encryption key, so while either can be used, the owner password
    can no longer be used to recover the user password.
   </para>
   <para>
    Starting with version 4.0.0, qpdf can read files that are not
    encrypted but that contain encrypted attachments, but it cannot
    write such files.  qpdf also requires the password to be specified
    in order to open the file, not just to extract attachments, since
    once the file is open, all decryption is handled transparently.
    When copying files like this while preserving encryption, qpdf
    will apply the file's encryption to everything in the file, not
    just to the attachments.  When decrypting the file, qpdf will
    decrypt the attachments.  In general, when copying PDF files with
    multiple encryption formats, qpdf will choose the newest format.
    The only exception to this is that clear-text metadata will be
    preserved as clear-text if it is that way in the original file.
   </para>
   <para>
    One point of confusion some people have about encrypted PDF files
    is that encryption is not the same as password protection.
    Password protected files are always encrypted, but it is also
    possible to create encrypted files that do not have passwords.
    Internally, such files use the empty string as a password, and
    most readers try the empty string first to see if it works and
    prompt for a password only if the empty string doesn't work.
    Normally such files have an empty user password and a non-empty
    owner password. In that way, if the file is opened by an ordinary
    reader without specification of password, the restrictions
    specified in the encryption dictionary can be enforced. Most users
    wouldn't even realize such a file was encrypted. Since qpdf always
    ignores the restrictions (except for the purpose of reporting what
    they are), qpdf doesn't care which password you use. QPDF will
    allow you to create PDF files with non-empty user passwords and
    empty owner passwords. Some readers will require a password when
    you open these files, and others will open the files without a
    password and not enforce restrictions. Having a non-empty user
    password and an empty owner password doesn't really make sense
    because it would mean that opening the file with the user password
    would be more restrictive than not supplying a password at all.
    QPDF also allows you to create PDF files with the same password as
    both the user and owner password. Some readers will not ever allow
    such files to be accessed without restrictions because they never
    try the password as the owner password if it works as the user
    password. Nonetheless, one of the powerful aspects of qpdf is that
    it allows you to finely specify the way encrypted files are
    created, even if the results are not useful to some readers. One
    use case for this would be for testing a PDF reader to ensure that
    it handles odd configurations of input files.
   </para>
  </sect1>
  <sect1 id="ref.random-numbers">
   <title>Random Number Generation</title>
   <para>
    QPDF generates random numbers to support generation of encrypted
    data.  Versions prior to 5.0.1 used <function>random</function> or
    <function>rand</function> from <filename>stdlib</filename> to
    generate random numbers.  Version 5.0.1, if available, used
    operating system-provided secure random number generation instead,
    enabling use of <filename>stdlib</filename> random number
    generation only if enabled by a compile-time option.  Starting in
    version 5.1.0, use of insecure random numbers was disabled unless
    enabled at compile time.  Starting in version 5.1.0, it is also
    possible for you to disable use of OS-provided secure random
    numbers.  This is especially useful on Windows if you want to
    avoid a dependency on Microsoft's cryptography API.  In this case,
    you must provide your own random data provider.  Regardless of how
    you compile qpdf, starting in version 5.1.0, it is possible for
    you to provide your own random data provider at runtime.  This
    would enable you to use some software-based secure pseudorandom
    number generator and to avoid use of whatever the operating system
    provides.  For details on how to do this, please refer to the
    top-level README.md file in the source distribution and to comments
    in <filename>QUtil.hh</filename>.
   </para>
  </sect1>
  <sect1 id="ref.adding-and-remove-pages">
   <title>Adding and Removing Pages</title>
   <para>
    While qpdf's API has supported adding and modifying objects for
    some time, version 3.0 introduces specific methods for adding and
    removing pages.  These are largely convenience routines that
    handle two tricky issues: pushing inheritable resources from the
    <literal>/Pages</literal> tree down to individual pages and
    manipulation of the <literal>/Pages</literal> tree itself.  For
    details, see <function>addPage</function> and surrounding methods
    in <filename>QPDF.hh</filename>.
   </para>
  </sect1>
  <sect1 id="ref.reserved-objects">
   <title>Reserving Object Numbers</title>
   <para>
    Version 3.0 of qpdf introduced the concept of reserved objects.
    These are seldom needed for ordinary operations, but there are
    cases in which you may want to add a series of indirect objects
    with references to each other to a <classname>QPDF</classname>
    object.  This causes a problem because you can't determine the
    object ID that a new indirect object will have until you add it to
    the <classname>QPDF</classname> object with
    <function>QPDF::makeIndirectObject</function>.  The only way to
    add two mutually referential objects to a
    <classname>QPDF</classname> object prior to version 3.0 would be
    to add the new objects first and then make them refer to each
    other after adding them.  Now it is possible to create a
    <firstterm>reserved object</firstterm> using
    <function>QPDFObjectHandle::newReserved</function>.  This is an
    indirect object that stays &ldquo;unresolved&rdquo; even if it is
    queried for its type.  So now, if you want to create a set of
    mutually referential objects, you can create reservations for each
    one of them and use those reservations to construct the
    references.  When finished, you can call
    <function>QPDF::replaceReserved</function> to replace the reserved
    objects with the real ones.  This functionality will never be
    needed by most applications, but it is used internally by QPDF
    when copying objects from other PDF files, as discussed in <xref
    linkend="ref.foreign-objects"/>.  For an example of how to use
    reserved objects, search for <function>newReserved</function> in
    <filename>test_driver.cc</filename> in qpdf's sources.
   </para>
  </sect1>
  <sect1 id="ref.foreign-objects">
   <title>Copying Objects From Other PDF Files</title>
   <para>
    Version 3.0 of qpdf introduced the ability to copy objects into a
    <classname>QPDF</classname> object from a different
    <classname>QPDF</classname> object, which we refer to as
    <firstterm>foreign objects</firstterm>. This allows arbitrary
    merging of PDF files. The &ldquo;from&rdquo;
    <classname>QPDF</classname> object must remain valid after the
    copy as discussed in the note below. The <command>qpdf</command>
    command-line tool provides limited support for basic page
    selection, including merging in pages from other files, but the
    library's API makes it possible to implement arbitrarily complex
    merging operations. The main method for copying foreign objects is
    <function>QPDF::copyForeignObject</function>. This takes an
    indirect object from another <classname>QPDF</classname> and
    copies it recursively into this object while preserving all object
    structure, including circular references. This means you can add a
    direct object that you create from scratch to a
    <classname>QPDF</classname> object with
    <function>QPDF::makeIndirectObject</function>, and you can add an
    indirect object from another file with
    <function>QPDF::copyForeignObject</function>. The fact that
    <function>QPDF::makeIndirectObject</function> does not
    automatically detect a foreign object and copy it is an explicit
    design decision. Copying a foreign object seems like a
    sufficiently significant thing to do that it should be done
    explicitly.
   </para>
   <para>
    The other way to copy foreign objects is by passing a page from
    one <classname>QPDF</classname> to another by calling
    <function>QPDF::addPage</function>.  In contrast to
    <function>QPDF::makeIndirectObject</function>, this method
    automatically distinguishes between indirect objects in the
    current file, foreign objects, and direct objects.
   </para>
   <para>
    Please note: when you copy objects from one
    <classname>QPDF</classname> to another, the source
    <classname>QPDF</classname> object must remain valid until you
    have finished with the destination object. This is because the
    original object is still used to retrieve any referenced stream
    data from the copied object.
   </para>
  </sect1>
  <sect1 id="ref.rewriting">
   <title>Writing PDF Files</title>
   <para>
    The qpdf library supports file writing of
    <classname>QPDF</classname> objects to PDF files through the
    <classname>QPDFWriter</classname> class.  The
    <classname>QPDFWriter</classname> class has two writing modes: one
    for non-linearized files, and one for linearized files.  See <xref
    linkend="ref.linearization"/> for a description of linearization
    is implemented.  This section describes how we write
    non-linearized files including the creation of QDF files (see
    <xref linkend="ref.qdf"/>.
   </para>
   <para>
    This outline was written prior to implementation and is not
    exactly accurate, but it provides a correct &ldquo;notional&rdquo;
    idea of how writing works.  Look at the code in
    <classname>QPDFWriter</classname> for exact details.
    <itemizedlist>
     <listitem>
      <para>
       Initialize state:
       <itemizedlist>
        <listitem>
         <para>
          next object number = 1
         </para>
        </listitem>
        <listitem>
         <para>
          object queue = empty
         </para>
        </listitem>
        <listitem>
         <para>
          renumber table: old object id/generation to new id/0 = empty
         </para>
        </listitem>
        <listitem>
         <para>
          xref table: new id -> offset = empty
         </para>
        </listitem>
       </itemizedlist>
      </para>
     </listitem>
     <listitem>
      <para>
       Create a QPDF object from a file.
      </para>
     </listitem>
     <listitem>
      <para>
       Write header for new PDF file.
      </para>
     </listitem>
     <listitem>
      <para>
       Request the trailer dictionary.
      </para>
     </listitem>
     <listitem>
      <para>
       For each value that is an indirect object, grab the next object
       number (via an operation that returns and increments the
       number).  Map object to new number in renumber table.  Push
       object onto queue.
      </para>
     </listitem>
     <listitem>
      <para>
       While there are more objects on the queue:
       <itemizedlist>
        <listitem>
         <para>
          Pop queue.
         </para>
        </listitem>
        <listitem>
         <para>
          Look up object's new number <emphasis>n</emphasis> in the
          renumbering table.
         </para>
        </listitem>
        <listitem>
         <para>
          Store current offset into xref table.
         </para>
        </listitem>
        <listitem>
         <para>
          Write <literal><replaceable>n</replaceable> 0 obj</literal>.
         </para>
        </listitem>
        <listitem>
         <para>
          If object is null, whether direct or indirect, write out
          null, thus eliminating unresolvable indirect object
          references.
         </para>
        </listitem>
        <listitem>
         <para>
          If the object is a stream stream, write stream contents,
          piped through any filters as required, to a memory buffer.
          Use this buffer to determine the stream length.
         </para>
        </listitem>
        <listitem>
         <para>
          If object is not a stream, array, or dictionary, write out
          its contents.
         </para>
        </listitem>
        <listitem>
         <para>
          If object is an array or dictionary (including stream),
          traverse its elements (for array) or values (for
          dictionaries), handling recursive dictionaries and arrays,
          looking for indirect objects.  When an indirect object is
          found, if it is not resolvable, ignore.  (This case is
          handled when writing it out.)  Otherwise, look it up in the
          renumbering table.  If not found, grab the next available
          object number, assign to the referenced object in the
          renumbering table, and push the referenced object onto the
          queue.  As a special case, when writing out a stream
          dictionary, replace length, filters, and decode parameters
          as required.
         </para>
         <para>
          Write out dictionary or array, replacing any unresolvable
          indirect object references with null (pdf spec says
          reference to non-existent object is legal and resolves to
          null) and any resolvable ones with references to the
          renumbered objects.
         </para>
        </listitem>
        <listitem>
         <para>
          If the object is a stream, write
          <literal>stream\n</literal>, the stream contents (from the
          memory buffer), and <literal>\nendstream\n</literal>.
         </para>
        </listitem>
        <listitem>
         <para>
          When done, write <literal>endobj</literal>.
         </para>
        </listitem>
       </itemizedlist>
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Once we have finished the queue, all referenced objects will have
    been written out and all deleted objects or unreferenced objects
    will have been skipped.  The new cross-reference table will
    contain an offset for every new object number from 1 up to the
    number of objects written.  This can be used to write out a new
    xref table.  Finally we can write out the trailer dictionary with
    appropriately computed /ID (see spec, 8.3, File Identifiers), the
    cross reference table offset, and <literal>%%EOF</literal>.
   </para>
  </sect1>
  <sect1 id="ref.filtered-streams">
   <title>Filtered Streams</title>
   <para>
    Support for streams is implemented through the
    <classname>Pipeline</classname> interface which was designed for
    this package.
   </para>
   <para>
    When reading streams, create a series of
    <classname>Pipeline</classname> objects.  The
    <classname>Pipeline</classname> abstract base requires
    implementation <function>write()</function> and
    <function>finish()</function> and provides an implementation of
    <function>getNext()</function>.  Each pipeline object, upon
    receiving data, does whatever it is going to do and then writes
    the data (possibly modified) to its successor.  Alternatively, a
    pipeline may be an end-of-the-line pipeline that does something
    like store its output to a file or a memory buffer ignoring a
    successor.  For additional details, look at
    <filename>Pipeline.hh</filename>.
   </para>
   <para>
    <classname>QPDF</classname> can read raw or filtered streams.
    When reading a filtered stream, the <classname>QPDF</classname>
    class creates a <classname>Pipeline</classname> object for one of
    each appropriate filter object and chains them together.  The last
    filter should write to whatever type of output is required.  The
    <classname>QPDF</classname> class has an interface to write raw or
    filtered stream contents to a given pipeline.
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.linearization">
  <title>Linearization</title>
  <para>
   This chapter describes how <classname>QPDF</classname> and
   <classname>QPDFWriter</classname> implement creation and processing
   of linearized PDFS.
  </para>
  <sect1 id="ref.linearization-strategy">
   <title>Basic Strategy for Linearization</title>
   <para>
    To avoid the incestuous problem of having the qpdf library
    validate its own linearized files, we have a special linearized
    file checking mode which can be invoked via <command>qpdf
    --check-linearization</command> (or <command>qpdf
    --check</command>).  This mode reads the linearization parameter
    dictionary and the hint streams and validates that object
    ordering, parameters, and hint stream contents are correct.  The
    validation code was first tested against linearized files created
    by external tools (Acrobat and pdlin) and then used to validate
    files created by <classname>QPDFWriter</classname> itself.
   </para>
  </sect1>
  <sect1 id="ref.linearized.preparation">
   <title>Preparing For Linearization</title>
   <para>
    Before creating a linearized PDF file from any other PDF file, the
    PDF file must be altered such that all page attributes are
    propagated down to the page level (and not inherited from parents
    in the <literal>/Pages</literal> tree).  We also have to know
    which objects refer to which other objects, being concerned with
    page boundaries and a few other cases.  We refer to this part of
    preparing the PDF file as <firstterm>optimization</firstterm>,
    discussed in <xref linkend="ref.optimization"/>.  Note the, in
    this context, the term <firstterm>optimization</firstterm> is a
    qpdf term, and the term <firstterm>linearization</firstterm> is a
    term from the PDF specification.  Do not be confused by the fact
    that many applications refer to linearization as optimization or
    web optimization.
   </para>
   <para>
    When creating linearized PDF files from optimized PDF files, there
    are really only a few issues that need to be dealt with:
    <itemizedlist>
     <listitem>
      <para>
       Creation of hints tables
      </para>
     </listitem>
     <listitem>
      <para>
       Placing objects in the correct order
      </para>
     </listitem>
     <listitem>
      <para>
       Filling in offsets and byte sizes
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
  <sect1 id="ref.optimization">
   <title>Optimization</title>
   <para>
    In order to perform various operations such as linearization and
    splitting files into pages, it is necessary to know which objects
    are referenced by which pages, page thumbnails, and root and
    trailer dictionary keys.  It is also necessary to ensure that all
    page-level attributes appear directly at the page level and are
    not inherited from parents in the pages tree.
   </para>
   <para>
    We refer to the process of enforcing these constraints as
    <firstterm>optimization</firstterm>.  As mentioned above, note
    that some applications refer to linearization as optimization.
    Although this optimization was initially motivated by the need to
    create linearized files, we are using these terms separately.
   </para>
   <para>
    PDF file optimization is implemented in the
    <filename>QPDF_optimization.cc</filename> source file.  That file
    is richly commented and serves as the primary reference for the
    optimization process.
   </para>
   <para>
    After optimization has been completed, the private member
    variables <varname>obj_user_to_objects</varname> and
    <varname>object_to_obj_users</varname> in
    <classname>QPDF</classname> have been populated.  Any object that
    has more than one value in the
    <varname>object_to_obj_users</varname> table is shared.  Any
    object that has exactly one value in the
    <varname>object_to_obj_users</varname> table is private.  To find
    all the private objects in a page or a trailer or root dictionary
    key, one merely has make this determination for each element in
    the <varname>obj_user_to_objects</varname> table for the given
    page or key.
   </para>
   <para>
    Note that pages and thumbnails have different object user types,
    so the above test on a page will not include objects referenced by
    the page's thumbnail dictionary and nothing else.
   </para>
  </sect1>
  <sect1 id="ref.linearization.writing">
   <title>Writing Linearized Files</title>
   <para>
    We will create files with only primary hint streams.  We will
    never write overflow hint streams.  (As of PDF version 1.4,
    Acrobat doesn't either, and they are never necessary.)  The hint
    streams contain offset information to objects that point to where
    they would be if the hint stream were not present.  This means
    that we have to calculate all object positions before we can
    generate and write the hint table.  This means that we have to
    generate the file in two passes.  To make this reliable,
    <classname>QPDFWriter</classname> in linearization mode invokes
    exactly the same code twice to write the file to a pipeline.
   </para>
   <para>
    In the first pass, the target pipeline is a count pipeline chained
    to a discard pipeline.  The count pipeline simply passes its data
    through to the next pipeline in the chain but can return the
    number of bytes passed through it at any intermediate point.  The
    discard pipeline is an end of line pipeline that just throws its
    data away.  The hint stream is not written and dummy values with
    adequate padding are stored in the first cross reference table,
    linearization parameter dictionary, and /Prev key of the first
    trailer dictionary.  All the offset, length, object renumbering
    information, and anything else we need for the second pass is
    stored.
   </para>
   <para>
    At the end of the first pass, this information is passed to the
    <classname>QPDF</classname> class which constructs a compressed
    hint stream in a memory buffer and returns it.
    <classname>QPDFWriter</classname> uses this information to write a
    complete hint stream object into a memory buffer.  At this point,
    the length of the hint stream is known.
   </para>
   <para>
    In the second pass, the end of the pipeline chain is a regular
    file instead of a discard pipeline, and we have known values for
    all the offsets and lengths that we didn't have in the first pass.
    We have to adjust offsets that appear after the start of the hint
    stream by the length of the hint stream, which is known.  Anything
    that is of variable length is padded, with the padding code
    surrounding any writing code that differs in the two passes.  This
    ensures that changes to the way things are represented never
    results in offsets that were gathered during the first pass
    becoming incorrect for the second pass.
   </para>
   <para>
    Using this strategy, we can write linearized files to a
    non-seekable output stream with only a single pass to disk or
    wherever the output is going.
   </para>
  </sect1>
  <sect1 id="ref.linearization-data">
   <title>Calculating Linearization Data</title>
   <para>
    Once a file is optimized, we have information about which objects
    access which other objects.  We can then process these tables to
    decide which part (as described in &ldquo;Linearized PDF Document
    Structure&rdquo; in the PDF specification) each object is
    contained within.  This tells us the exact order in which objects
    are written.  The <classname>QPDFWriter</classname> class asks for
    this information and enqueues objects for writing in the proper
    order.  It also turns on a check that causes an exception to be
    thrown if an object is encountered that has not already been
    queued.  (This could happen only if there were a bug in the
    traversal code used to calculate the linearization data.)
   </para>
  </sect1>
  <sect1 id="ref.linearization-issues">
   <title>Known Issues with Linearization</title>
   <para>
    There are a handful of known issues with this linearization code.
    These issues do not appear to impact the behavior of linearized
    files which still work as intended: it is possible for a web
    browser to begin to display them before they are fully
    downloaded.  In fact, it seems that various other programs that
    create linearized files have many of these same issues.  These
    items make reference to terminology used in the linearization
    appendix of the PDF specification.
    <itemizedlist>
     <listitem>
      <para>
       Thread Dictionary information keys appear in part 4 with the
       rest of Threads instead of in part 9.  Objects in part 9 are
       not grouped together functionally.
      </para>
     </listitem>
     <listitem>
      <para>
       We are not calculating numerators for shared object positions
       within content streams or interleaving them within content
       streams.
      </para>
     </listitem>
     <listitem>
      <para>
       We generate only page offset, shared object, and outline hint
       tables.  It would be relatively easy to add some additional
       tables.  We gather most of the information needed to create
       thumbnail hint tables.  There are comments in the code about
       this.
      </para>
     </listitem>
    </itemizedlist>
   </para>
  </sect1>
  <sect1 id="ref.linearization-debugging">
   <title>Debugging Note</title>
   <para>
    The <command>qpdf --show-linearization</command> command can show
    the complete contents of linearization hint streams.  To look at
    the raw data, you can extract the filtered contents of the
    linearization hint tables using <command>qpdf --show-object=n
    --filtered-stream-data</command>.  Then, to convert this into a
    bit stream (since linearization tables are bit streams written
    without regard to byte boundaries), you can pipe the resulting
    data through the following perl code:

    <programlisting>use bytes;
binmode STDIN;
undef $/;
my $a = &lt;STDIN&gt;;
my @ch = split(//, $a);
map { printf("%08b", ord($_)) } @ch;
print "\n";
</programlisting>
   </para>
  </sect1>
 </chapter>
 <chapter id="ref.object-and-xref-streams">
  <title>Object and Cross-Reference Streams</title>
  <para>
   This chapter provides information about the implementation of
   object stream and cross-reference stream support in qpdf.
  </para>
  <sect1 id="ref.object-streams">
   <title>Object Streams</title>
   <para>
    Object streams can contain any regular object except the
    following:
    <itemizedlist>
     <listitem>
      <para>
       stream objects
      </para>
     </listitem>
     <listitem>
      <para>
       objects with generation &gt; 0
      </para>
     </listitem>
     <listitem>
      <para>
       the encryption dictionary
      </para>
     </listitem>
     <listitem>
      <para>
       objects containing the /Length of another stream
      </para>
     </listitem>
    </itemizedlist>
    In addition, Adobe reader (at least as of version 8.0.0) appears
    to not be able to handle having the document catalog appear in an
    object stream if the file is encrypted, though this is not
    specifically disallowed by the specification.
   </para>
   <para>
    There are additional restrictions for linearized files.  See <xref
    linkend="ref.object-streams-linearization"/>for details.
   </para>
   <para>
    The PDF specification refers to objects in object streams as
    &ldquo;compressed objects&rdquo; regardless of whether the object
    stream is compressed.
   </para>
   <para>
    The generation number of every object in an object stream must be
    zero.  It is possible to delete and replace an object in an object
    stream with a regular object.
   </para>
   <para>
    The object stream dictionary has the following keys:
    <itemizedlist>
     <listitem>
      <para>
       <literal>/N</literal>: number of objects
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/First</literal>: byte offset of first object
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/Extends</literal>: indirect reference to stream that
       this extends
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    Stream collections are formed with <literal>/Extends</literal>.
    They must form a directed acyclic graph.  These can be used for
    semantic information and are not meaningful to the PDF document's
    syntactic structure.  Although qpdf preserves stream collections,
    it never generates them and doesn't make use of this information
    in any way.
   </para>
   <para>
    The specification recommends limiting the number of objects in
    object stream for efficiency in reading and decoding.  Acrobat 6
    uses no more than 100 objects per object stream for linearized
    files and no more 200 objects per stream for non-linearized files.
    <classname>QPDFWriter</classname>, in object stream generation
    mode, never puts more than 100 objects in an object stream.
   </para>
   <para>
    Object stream contents consists of <emphasis>N</emphasis> pairs of
    integers, each of which is the object number and the byte offset
    of the object relative to the first object in the stream, followed
    by the objects themselves, concatenated.
   </para>
  </sect1>
  <sect1 id="ref.xref-streams">
   <title>Cross-Reference Streams</title>
   <para>
    For non-hybrid files, the value following
    <literal>startxref</literal> is the byte offset to the xref stream
    rather than the word <literal>xref</literal>.
   </para>
   <para>
    For hybrid files (files containing both xref tables and
    cross-reference streams), the xref table's trailer dictionary
    contains the key <literal>/XRefStm</literal> whose value is the
    byte offset to a cross-reference stream that supplements the xref
    table.  A PDF 1.5-compliant application should read the xref table
    first.  Then it should replace any object that it has already seen
    with any defined in the xref stream.  Then it should follow any
    <literal>/Prev</literal> pointer in the original xref table's
    trailer dictionary.  The specification is not clear about what
    should be done, if anything, with a <literal>/Prev</literal>
    pointer in the xref stream referenced by an xref table.  The
    <classname>QPDF</classname> class ignores it, which is probably
    reasonable since, if this case were to appear for any sensible PDF
    file, the previous xref table would probably have a corresponding
    <literal>/XRefStm</literal> pointer of its own.  For example, if a
    hybrid file were appended, the appended section would have its own
    xref table and <literal>/XRefStm</literal>.  The appended xref
    table would point to the previous xref table which would point the
    <literal>/XRefStm</literal>, meaning that the new
    <literal>/XRefStm</literal> doesn't have to point to it.
   </para>
   <para>
    Since xref streams must be read very early, they may not be
    encrypted, and the may not contain indirect objects for keys
    required to read them, which are these:
    <itemizedlist>
     <listitem>
      <para>
       <literal>/Type</literal>: value <literal>/XRef</literal>
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/Size</literal>: value <emphasis>n+1</emphasis>: where
       <emphasis>n</emphasis> is highest object number (same as
       <literal>/Size</literal> in the trailer dictionary)
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/Index</literal> (optional): value
       <literal>[<replaceable>n count</replaceable> ...]</literal>
       used to determine which objects' information is stored in this
       stream.  The default is <literal>[0 /Size]</literal>.
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/Prev</literal>: value
       <replaceable>offset</replaceable>: byte offset of previous xref
       stream (same as <literal>/Prev</literal> in the trailer
       dictionary)
      </para>
     </listitem>
     <listitem>
      <para>
       <literal>/W [...]</literal>: sizes of each field in the xref
       table
      </para>
     </listitem>
    </itemizedlist>
   </para>
   <para>
    The other fields in the xref stream, which may be indirect if
    desired, are the union of those from the xref table's trailer
    dictionary.
   </para>
   <sect2 id="ref.xref-stream-data">
    <title>Cross-Reference Stream Data</title>
    <para>
     The stream data is binary and encoded in big-endian byte order.
     Entries are concatenated, and each entry has a length equal to
     the total of the entries in <literal>/W</literal> above.  Each
     entry consists of one or more fields, the first of which is the
     type of the field.  The number of bytes for each field is given
     by <literal>/W</literal> above.  A 0 in <literal>/W</literal>
     indicates that the field is omitted and has the default value.
     The default value for the field type is
     &ldquo;<literal>1</literal>&rdquo;.  All other default values are
     &ldquo;<literal>0</literal>&rdquo;.
    </para>
    <para>
     PDF 1.5 has three field types:
     <itemizedlist>
      <listitem>
       <para>
        0: for free objects.  Format: <literal>0 obj
        next-generation</literal>, same as the free table in a
        traditional cross-reference table
      </para>
     </listitem>
     <listitem>
      <para>
       1: regular non-compressed object.  Format: <literal>1 offset
       generation</literal>
      </para>
     </listitem>
     <listitem>
      <para>
       2: for objects in object streams.  Format: <literal>2
       object-stream-number index</literal>, the number of object
       stream containing the object and the index within the object
       stream of the object.
      </para>
     </listitem>
     </itemizedlist>
    </para>
    <para>
     It seems standard to have the first entry in the table be
     <literal>0 0 0</literal> instead of <literal>0 0 ffff</literal>
     if there are no deleted objects.
    </para>
   </sect2>
  </sect1>
  <sect1 id="ref.object-streams-linearization">
   <title>Implications for Linearized Files</title>
   <para>
    For linearized files, the linearization dictionary, document
    catalog, and page objects may not be contained in object streams.
   </para>
   <para>
    Objects stored within object streams are given the highest range
    of object numbers within the main and first-page cross-reference
    sections.
   </para>
   <para>
    It is okay to use cross-reference streams in place of regular xref
    tables.  There are on special considerations.
   </para>
   <para>
    Hint data refers to object streams themselves, not the objects in
    the streams.  Shared object references should also be made to the
    object streams.  There are no reference in any hint tables to the
    object numbers of compressed objects (objects within object
    streams).
   </para>
   <para>
    When numbering objects, all shared objects within both the first
    and second halves of the linearized files must be numbered
    consecutively after all normal uncompressed objects in that half.
   </para>
  </sect1>
  <sect1 id="ref.object-stream-implementation">
   <title>Implementation Notes</title>
   <para>
    There are three modes for writing object streams:
    <option>disable</option>, <option>preserve</option>, and
    <option>generate</option>.  In disable mode, we do not generate
    any object streams, and we also generate an xref table rather than
    xref streams.  This can be used to generate PDF files that are
    viewable with older readers.  In preserve mode, we write object
    streams such that written object streams contain the same objects
    and <literal>/Extends</literal> relationships as in the original
    file.  This is equal to disable if the file has no object streams.
    In generate, we create object streams ourselves by grouping
    objects that are allowed in object streams together in sets of no
    more than 100 objects.  We also ensure that the PDF version is at
    least 1.5 in generate mode, but we preserve the version header in
    the other modes.  The default is <option>preserve</option>.
   </para>
   <para>
    We do not support creation of hybrid files.  When we write files,
    even in preserve mode, we will lose any xref tables and merge any
    appended sections.
   </para>
  </sect1>
 </chapter>
 <appendix id="ref.release-notes">
  <title>Release Notes</title>
  <para>
   For a detailed list of changes, please see the file
   <filename>ChangeLog</filename> in the source distribution.
  </para>
  <variablelist>
   <varlistentry>
   <term>9.1.1: January 26, 2020</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Build/Packaging Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          The fix-qdf program was converted from perl to C++. As such,
          qpdf no longer has a runtime dependency on perl.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Library Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Added new helper routine
          <function>QUtil::call_main_from_wmain</function> which
          converts <type>wchar_t</type> arguments to UTF-8 encoded
          strings. This is useful for qpdf because library methods
          expect file names to be UTF-8 encoded, even on Windows
         </para>
        </listitem>
        <listitem>
         <para>
          Added new <function>QUtil::read_lines_from_file</function>
          methods that take <type>FILE*</type> arguments and that
          allow preservation of end-of-line characters. This also
          fixes a bug where
          <function>QUtil::read_lines_from_file</function> wouldn't
          work properly with Unicode filenames.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        CLI Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Added options <option>--is-encrypted</option> and
          <option>--requires-password</option> for testing whether a
          file is encrypted or requires a password other than the
          supplied (or empty) password. These communicate via exit
          status, making them useful for shell scripts. They also work
          on encrypted files with unknown passwords.
         </para>
        </listitem>
       </itemizedlist>
       <itemizedlist>
        <listitem>
         <para>
          Added <literal>encrypt</literal> key to JSON options. With
          the exception of the reconstructed user password for older
          encryption formats, this provides the same information as
          <option>--show-encryption</option> but in a consistent,
          parseable format. See output of <command>qpdf
          --json-help</command> for details.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          In QDF mode, be sure not to write more than one XRef stream
          to a file, even when
          <option>--preserve-unreferenced</option> is used.
          <command>fix-qdf</command> assumes that there is only one
          XRef stream, and that it appears at the end of the file.
         </para>
        </listitem>
        <listitem>
         <para>
          When externalizing inline images, properly handle images
          whose color space is a reference to an object in the page's
          resource dictionary.
         </para>
        </listitem>
        <listitem>
         <para>
          Windows-specific fix for acquiring crypt context with a new
          keyset.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
  </variablelist>
  <variablelist>
   <varlistentry>
    <term>9.1.0: November 17, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Build Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          A C++-11 compiler is now required to build qpdf.
         </para>
        </listitem>
        <listitem>
         <para>
          A new crypto provider that uses gnutls for crypto functions
          is now available and can be enabled at build time. See <xref
          linkend="ref.crypto"/> for more information about crypto
          providers and <xref linkend="ref.crypto.build"/> for
          specific information about the build.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Library Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Incorporate contribution from Masamichi Hosoda to properly
          handle signature dictionaries by not including them in
          object streams, formatting the <literal>Contents</literal>
          key has a hexadecimal string, and excluding the
          <literal>/Contents</literal> key from encryption and
          decryption.
         </para>
        </listitem>
        <listitem>
         <para>
          Incorporate contribution from Masamichi Hosoda to provide
          new API calls for getting file-level information about
          input and output files, enabling certain operations on
          the files at the file level rather than the object level.
          New methods include
          <function>QPDF::getXRefTable()</function>,
          <function>QPDFObjectHandle::getParsedOffset()</function>,
          <function>QPDFWriter::getRenumberedObjGen(QPDFObjGen)</function>,
          and <function>QPDFWriter::getWrittenXRefTable()</function>.
         </para>
        </listitem>
        <listitem>
         <para>
          Support build-time and runtime selectable crypto providers.
          This includes the addition of new classes
          <classname>QPDFCryptoProvider</classname> and
          <classname>QPDFCryptoImpl</classname> and the recognition
          of the <literal>QPDF_CRYPTO_PROVIDER</literal> environment
          variable. Crypto providers are described in depth in <xref
          linkend="ref.crypto"/>.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
        CLI Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Addition of the <option>--show-crypto</option> option in
          support of selectable crypto providers, as described in
          <xref linkend="ref.crypto"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          Allow <literal>:even</literal> or <literal>:odd</literal> to
          be appended to numeric ranges for specification of the even
          or odd pages from among the pages specified in the range.
         </para>
        </listitem>
        <listitem>
         <para>
          Fix shell wildcard expansion behavior (<literal>*</literal>
          and <literal>?</literal>) of the <command>qpdf.exe</command>
          as built my MSVC.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
  </variablelist>
  <variablelist>
   <varlistentry>
    <term>9.0.2: October 12, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug Fix
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Fix the name of the temporary file used by
          <option>--replace-input</option> so that it doesn't require
          path splitting and works with paths include directories.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>9.0.1: September 20, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug Fixes/Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Fix some build and test issues on big-endian systems and
          compilers with characters that are unsigned by default.
          The problems were in build and test only. There were no
          actual bugs in the qpdf library itself relating to
          endianness or unsigned characters.
         </para>
        </listitem>
        <listitem>
         <para>
          When a dictionary has a duplicated key, report this with a
          warning. The behavior of the library in this case is
          unchanged, but the error condition is no longer silently
          ignored.
         </para>
        </listitem>
        <listitem>
         <para>
          When a form field's display rectangle is erroneously
          specified with inverted coordinates, detect and correct this
          situation. This avoids some form fields from being flipped
          when flattening annotations on files with this condition.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>9.0.0: August 31, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Incompatible API (source-level) Changes (minor)
       </para>
       <itemizedlist>
        <listitem>
         <para>
          The method <function>QUtil::strcasecmp</function> has been
          renamed to <function>QUtil::str_compare_nocase</function>.
          This incompatible change is necessary to enable qpdf to
          build on platforms that define
          <function>strcasecmp</function> as a macro.
         </para>
        </listitem>
        <listitem>
         <para>
          The <function>QPDF::copyForeignObject</function> method had
          an overloaded version that took a boolean parameter that was
          not used. If you were using this version, just omit the
          extra parameter.
         </para>
        </listitem>
        <listitem>
         <para>
          There was a version
          <function>QPDFTokenizer::expectInlineImage</function> that
          took no arguments. This version has been removed since it
          caused the tokenizer to return incorrect inline images. A
          new version was added some time ago that produces correct
          output. This is a very low level method that doesn't make
          sense to call outside of qpdf's lexical engine. There are
          higher level methods for tokenizing content streams.
         </para>
        </listitem>
        <listitem>
         <para>
          Change
          <function>QPDFOutlineDocumentHelper::getTopLevelOutlines</function>
          and <function>QPDFOutlineObjectHelper::getKids</function> to
          return a <type>std::vector</type> instead of a
          <type>std::list</type> of
          <classname>QPDFOutlineObjectHelper</classname> objects.
         </para>
        </listitem>
        <listitem>
         <para>
          Remove method
          <function>QPDFTokenizer::allowPoundAnywhereInName</function>.
          This function would allow creation of name tokens whose
          value would change when unparsed, which is never the correct
          behavior.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        CLI Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          The <option>--replace-input</option> option may be given in
          place of an output file name. This causes qpdf to overwrite
          the input file with the output. See the description of
          <option>--replace-input</option> in <xref
          linkend="ref.basic-options"/> for more details.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--recompress-flate</option> instructs
          <command>qpdf</command> to recompress streams that are
          already compressed with <literal>/FlateDecode</literal>.
          Useful with <option>--compression-level</option>.
         </para>
        </listitem>
        <listitem>
         <para>
          The
          <option>--compression-level=<replaceable>level</replaceable></option>
          sets the zlib compression level used for any streams
          compressed by <literal>/FlateDecode</literal>. Most
          effective when combined with
          <option>--recompress-flate</option>.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Library Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          A new namespace <classname>QIntC</classname>, provided by
          <filename>qpdf/QIntC.hh</filename>, provides safe conversion
          methods between different integer types. These conversion
          methods do range checking to ensure that the cast can be
          performed with no loss of information. Every use of
          <function>static_cast</function> in the library was
          inspected to see if it could use one of these safe
          converters instead. See <xref linkend="ref.casting"/> for
          additional details.
         </para>
        </listitem>
        <listitem>
         <para>
          Method <function>QPDF::anyWarnings</function> tells whether
          there have been any warnings without clearing the list of
          warnings.
         </para>
        </listitem>
        <listitem>
         <para>
          Method <function>QPDF::closeInputSource</function> closes or
          otherwise releases the input source. This enables the input
          file to be deleted or renamed.
         </para>
        </listitem>
        <listitem>
         <para>
          New methods have been added to <classname>QUtil</classname>
          for converting back and forth between strings and unsigned
          integers: <function>uint_to_string</function>,
          <function>uint_to_string_base</function>,
          <function>string_to_uint</function>, and
          <function>string_to_ull</function>.
         </para>
        </listitem>
        <listitem>
         <para>
          New methods have been added to
          <classname>QPDFObjectHandle</classname> that return the
          value of <classname>Integer</classname> objects as
          <type>int</type> or <type>unsigned int</type> with range
          checking and sensible fallback values, and a new method was
          added to return an unsigned value. This makes it easier to
          write code that is safe from unintentional data loss.
          Functions: <function>getUIntValue</function>,
          <function>getIntVauleAsInt</function>,
          <function>getUIntValueAsUInt</function>.
         </para>
        </listitem>
        <listitem>
         <para>
          When parsing content streams with
          <classname>QPDFObjectHandle::ParserCallbacks</classname>, in
          place of the method
          <function>handleObject(QPDFObjectHandle)</function>, the
          developer may override
          <function>handleObject(QPDFObjectHandle, size_t offset,
          size_t length)</function>. If this method is defined, it
          will be invoked with the object along with its offset and
          length within the overall contents being parsed. Intervening
          spaces and comments are not included in offset and length.
          Additionally, a new method
          <function>contentSize(size_t)</function> may be implemented.
          If present, it will be called prior to the first call to
          <function>handleObject</function> with the total size in
          bytes of the combined contents.
         </para>
        </listitem>
        <listitem>
         <para>
          New methods <function>QPDF::userPasswordMatched</function>
          and <function>QPDF::ownerPasswordMatched</function> have
          been added to enable a caller to determine whether the
          supplied password was the user password, the owner password,
          or both. This information is also displayed by <command>qpdf
          --show-encryption</command> and <command>qpdf
          --check</command>.
         </para>
        </listitem>
        <listitem>
         <para>
          Static method
          <function>Pl_Flate::setCompressionLevel</function> can be
          called to set the zlib compression level globally used by
          all instances of Pl_Flate in deflate mode.
         </para>
        </listitem>
        <listitem>
         <para>
          The method
          <function>QPDFWriter::setRecompressFlate</function> can be
          called to tell <classname>QPDFWriter</classname> to
          uncompress and recompress streams already compressed with
          <literal>/FlateDecode</literal>.
         </para>
        </listitem>
        <listitem>
         <para>
          The underlying implementation of QPDF arrays has been
          enhanced to be much more memory efficient when dealing with
          arrays with lots of nulls. This enables qpdf to use
          drastically less memory for certain types of files.
         </para>
        </listitem>
        <listitem>
         <para>
          When traversing the pages tree, if nodes are encountered
          with invalid types, the types are fixed, and a warning is
          issued.
         </para>
        </listitem>
        <listitem>
         <para>
          A new helper method
          <function>QUtil::read_file_into_memory</function> was added.
         </para>
        </listitem>
        <listitem>
         <para>
          All conditions previously reported by
          <function>QPDF::checkLinearization()</function> as errors
          are now presented as warnings.
         </para>
        </listitem>
        <listitem>
         <para>
          Name tokens containing the <literal>#</literal> character
          not preceded by two hexadecimal digits, which is invalid in
          PDF 1.2 and above, are properly handled by the library: a
          warning is generated, and the name token is properly
          preserved, even if invalid, in the output. See
          <filename>ChangeLog</filename> for a more complete
          description of this change.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          A small handful of memory issues, assertion failures, and
          unhandled exceptions that could occur on badly mangled input
          files have been fixed. Most of these problems were found by
          Google's OSS-Fuzz project.
         </para>
        </listitem>
        <listitem>
         <para>
          When <command>qpdf --check</command> or <command>qpdf
          --check-linearization</command> encounters a file with
          linearization warnings but not errors, it now properly exits
          with exit code 3 instead of 2.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--completion-bash</option> and
          <option>--completion-zsh</option> options now work properly
          when qpdf is invoked as an AppImage.
         </para>
        </listitem>
        <listitem>
         <para>
          Calling
          <function>QPDFWriter::set*EncryptionParameters</function> on
          a <classname>QPDFWriter</classname> object whose output
          filename has not yet been set no longer produces a
          segmentation fault.
         </para>
        </listitem>
        <listitem>
         <para>
          When reading encrypted files, follow the spec more closely
          regarding encryption key length. This allows qpdf to open
          encrypted files in most cases when they have invalid or
          missing /Length keys in the encryption dictionary.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Build Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          On platforms that support it, qpdf now builds with
          <option>-fvisibility=hidden</option>. If you build qpdf with
          your own build system, this is now safe to use. This
          prevents methods that are not part of the public API from
          being exported by the shared library, and makes qpdf's ELF
          shared libraries (used on Linux, MacOS, and most other UNIX
          flavors) behave more like the Windows DLL. Since the DLL
          already behaves in much this way, it is unlikely that there
          are any methods that were accidentally not exported.
          However, with ELF shared libraries, typeinfo for some
          classes has to be explicitly exported. If there are problems
          in dynamically linked code catching exceptions or
          subclassing, this could be the reason. If you see this,
          please report a bug at <ulink
          url="https://github.com/qpdf/qpdf/issues/">pikepdf</ulink>.
         </para>
        </listitem>
        <listitem>
         <para>
          QPDF is now compiled with integer conversion and sign
          conversion warnings enabled. Numerous changes were made to
          the library to make this safe.
         </para>
        </listitem>
        <listitem>
         <para>
          QPDF's <command>make install</command> target explicitly
          specifies the mode to use when installing files instead of
          relying the user's umask. It was previously doing this for
          some files but not others.
         </para>
        </listitem>
        <listitem>
         <para>
          If <command>pkg-config</command> is available, use it to
          locate <filename>libjpeg</filename> and
          <filename>zlib</filename> dependencies, falling back on old
          behavior if unsuccessful.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Other Notes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          QPDF has been fully integrated into <ulink
          url="https://github.com/google/oss-fuzz">Google's OSS-Fuzz
          project</ulink>. This project exercises code with randomly
          mutated inputs and is great for discovering hidden security
          crashes and security issues. Several bugs found by oss-fuzz
          have already been fixed in qpdf.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.4.2: May 18, 2019</term>
    <listitem>
     <para>
      This release has just one change: correction of a buffer overrun
      in the Windows code used to open files. Windows users should
      take this update. There are no code changes that affect
      non-Windows releases.
     </para>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.4.1: April 27, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          When <command>qpdf --version</command> is run, it will
          detect if the qpdf CLI was built with a different version of
          qpdf than the library, which may indicate a problem with the
          installation.
         </para>
        </listitem>
        <listitem>
         <para>
	  New option <option>--remove-page-labels</option> will remove page
          labels before generating output. This used to happen if you
          ran <command>qpdf --empty --pages .. --</command>, but the
          behavior changed in qpdf 8.3.0. This option enables people
          who were relying on the old behavior to get it again.
         </para>
        </listitem>
        <listitem>
         <para>
          New option
          <option>--keep-files-open-threshold=<replaceable>count</replaceable></option>
          can be used to override number of files that qpdf will use
          to trigger the behavior of not keeping all files open when
          merging files. This may be necessary if your system allows
          fewer than the default value of 200 files to be open at the
          same time.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Handle Unicode characters in filenames on Windows. The
          changes to support Unicode on the CLI in Windows broke
          Unicode filenames for Windows.
         </para>
        </listitem>
        <listitem>
         <para>
          Slightly tighten logic that determines whether an object is
          a page. This should resolve problems in some rare files
          where some non-page objects were passing qpdf's test for
          whether something was a page, thus causing them to be
          erroneously lost during page splitting operations.
         </para>
        </listitem>
        <listitem>
         <para>
	  Revert change that included preservation of outlines
	  (bookmarks) in <option>--split-pages</option>. The way it
	  was implemented in 8.3.0 and 8.4.0 caused a very significant
	  degradation of performance for splitting certain files. A
	  future release of qpdf may re-introduce the behavior in a
	  more performant and also more correct fashion.
         </para>
        </listitem>
        <listitem>
         <para>
          In JSON mode, add missing leading 0 to decimal values
          between -1 and 1 even if not present in the input. The JSON
          specification requires the leading 0. The PDF specification
          does not.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.4.0: February 1, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Command-line Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          <emphasis>Non-compatible CLI change:</emphasis> The qpdf
          command-line tool interprets passwords given at the
          command-line differently from previous releases when the
          passwords contain non-ASCII characters. In some cases, the
          behavior differs from previous releases. For a discussion of
          the current behavior, please see <xref
          linkend="ref.unicode-passwords"/>. The incompatibilities are
          as follows:
          <itemizedlist>
           <listitem>
            <para>
             On Windows, qpdf now receives all command-line options as
             Unicode strings if it can figure out the appropriate
             compile/link options. This is enabled at least for MSVC
             and mingw builds. That means that if non-ASCII strings
             are passed to the qpdf CLI in Windows, qpdf will now
             correctly receive them. In the past, they would have
             either been encoded as Windows code page 1252 (also known
             as &ldquo;Windows ANSI&rdquo; or as something
             unintelligible. In almost all cases, qpdf is able to
             properly interpret Unicode arguments now, whereas in the
             past, it would almost never interpret them properly. The
             result is that non-ASCII passwords given to the qpdf CLI
             on Windows now have a much greater chance of creating PDF
             files that can be opened by a variety of readers. In the
             past, usually files encrypted from the Windows CLI using
             non-ASCII passwords would not be readable by most
             viewers. Note that the current version of qpdf is able to
             decrypt files that it previously created using the
             previously supplied password.
            </para>
           </listitem>
           <listitem>
            <para>
             The PDF specification requires passwords to be encoded as
             UTF-8 for 256-bit encryption and with PDF Doc encoding
             for 40-bit or 128-bit encryption. Older versions of qpdf
             left it up to the user to provide passwords with the
             correct encoding. The qpdf CLI now detects when a
             password is given with UTF-8 encoding and automatically
             transcodes it to what the PDF spec requires. While this
             is almost always the correct behavior, it is possible to
             override the behavior if there is some reason to do so.
             This is discussed in more depth in <xref
             linkend="ref.unicode-passwords"/>.
            </para>
           </listitem>
          </itemizedlist>
         </para>
        </listitem>
        <listitem>
         <para>
          New options <option>--externalize-inline-images</option>,
          <option>--ii-min-bytes</option>, and
          <option>--keep-inline-images</option> control qpdf's
          handling of inline images and possible conversion of them to
          regular images. By default,
          <option>--optimize-images</option> now also applies to
          inline images. These options are discussed in <xref
          linkend="ref.advanced-transformation"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          Add options <option>--overlay</option> and
          <option>--underlay</option> for overlaying or underlaying
          pages of other files onto output pages. See <xref
          linkend="ref.overlay-underlay"/> for details.
         </para>
        </listitem>
        <listitem>
         <para>
          When opening an encrypted file with a password, if the
          specified password doesn't work and the password contains
          any non-ASCII characters, qpdf will try a number of
          alternative passwords to try to compensate for possible
          character encoding errors. This behavior can be suppressed
          with the <option>--suppress-password-recovery</option>
          option. See <xref linkend="ref.unicode-passwords"/> for a
          full discussion.
         </para>
        </listitem>
        <listitem>
         <para>
          Add the <option>--password-mode</option> option to fine-tune
          how qpdf interprets password arguments, especially when they
          contain non-ASCII characters. See <xref
          linkend="ref.unicode-passwords"/> for more information.
         </para>
        </listitem>
        <listitem>
         <para>
          In the <option>--pages</option> option, it is now possible
          to copy the same page more than once from the same file
          without using the previous workaround of specifying two
          different paths to the same file.
         </para>
        </listitem>
        <listitem>
         <para>
          In the <option>--pages</option> option, allow use of
          &ldquo;.&rdquo; as a shortcut for the primary input file.
          That way, you can do <command>qpdf in.pdf --pages . 1-2 --
          out.pdf</command> instead of having to repeat
          <filename>in.pdf</filename> in the command.
         </para>
        </listitem>
        <listitem>
         <para>
          When encrypting with 128-bit and 256-bit encryption, new
          encryption options <option>--assemble</option>,
          <option>--annotate</option>, <option>--form</option>, and
          <option>--modify-other</option> allow more fine-grained
          granularity in configuring options. Before, the
          <option>--modify</option> option only configured certain
          predefined groups of permissions.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes and Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          <emphasis>Potential data-loss bug:</emphasis> Versions of
          qpdf between 8.1.0 and 8.3.0 had a bug that could cause page
          splitting and merging operations to drop some font or image
          resources if the PDF file's internal structure shared these
          resource lists across pages and if some but not all of the
          pages in the output did not reference all the fonts and
          images. Using the
          <option>--preserve-unreferenced-resources</option> option
          would work around the incorrect behavior. This bug was the
          result of a typo in the code and a deficiency in the test
          suite. The case that triggered the error was known, just not
          handled properly. This case is now exercised in qpdf's test
          suite and properly handled.
         </para>
        </listitem>
        <listitem>
         <para>
           When optimizing images, detect and refuse to optimize
           images that can't be converted to JPEG because of bit depth
           or color space.
         </para>
        </listitem>
        <listitem>
         <para>
          Linearization and page manipulation APIs now detect and
          recover from files that have duplicate Page objects in the
          pages tree.
         </para>
        </listitem>
        <listitem>
         <para>
          Using older option <option>--stream-data=compress</option>
          with object streams, object streams and xref streams were
          not compressed.
         </para>
        </listitem>
        <listitem>
         <para>
          When the tokenizer returns inline image tokens, delimiters
          following <literal>ID</literal> and <literal>EI</literal>
          operators are no longer excluded. This makes it possible to
          reliably extract the actual image data.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Library Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Add method
          <function>QPDFPageObjectHelper::externalizeInlineImages</function>
          to convert inline images to regular images.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method
          <function>QUtil::possible_repaired_encodings()</function> to
          generate a list of strings that represent other ways the
          given string could have been encoded. This is the method the
          QPDF CLI uses to generate the strings it tries when
          recovering incorrectly encoded Unicode passwords.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new versions of
          <function>QPDFWriter::setR{3,4,5,6}EncryptionParameters</function>
          that allow more granular setting of permissions bits. See
          <filename>QPDFWriter.hh</filename> for details.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new versions of the transcoders from UTF-8 to
          single-byte coding systems in <classname>QUtil</classname>
          that report success or failure rather than just substituting
          a specified unknown character.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method <function>QUtil::analyze_encoding()</function> to
          determine whether a string has high-bit characters and is
          appears to be UTF-16 or valid UTF-8 encoding.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new method
          <function>QPDFPageObjectHelper::shallowCopyPage()</function>
          to copy a new page that is a &ldquo;shallow copy&rdquo; of a
          page. The resulting object is an indirect object ready to be
          passed to
          <function>QPDFPageDocumentHelper::addPage()</function> for
          either the original <classname>QPDF</classname> object or a
          different one. This is what the <command>qpdf</command>
          command-line tool uses to copy the same page multiple times
          from the same file during splitting and merging operations.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method <function>QPDF::getUniqueId()</function>, which
          returns a unique identifier for the given QPDF object. The
          identifier will be unique across the life of the
          application. The returned value can be safely used as a map
          key.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method <function>QPDF::setImmediateCopyFrom</function>.
          This further enhances qpdf's ability to allow a
          <classname>QPDF</classname> object from which objects are
          being copied to go out of scope before the destination
          object is written. If you call this method on a
          <classname>QPDF</classname> instances, objects copied
          <emphasis>from</emphasis> this instance will be copied
          immediately instead of lazily. This option uses more memory
          but allows the source object to go out of scope before the
          destination object is written in all cases. See comments in
          <filename>QPDF.hh</filename> for details.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method
          <function>QPDFPageObjectHelper::getAttribute</function> for
          retrieving an attribute from the page dictionary taking
          inheritance into consideration, and optionally making a copy
          if your intention is to modify the attribute.
         </para>
        </listitem>
        <listitem>
         <para>
          Fix long-standing limitation of
          <function>QPDFPageObjectHelper::getPageImages</function> so
          that it now properly reports images from inherited resources
          dictionaries, eliminating the need to call
          <function>QPDFPageDocumentHelper::pushInheritedAttributesToPage</function>
          in this case.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method
          <function>QPDFObjectHandle::getUniqueResourceName</function>
          for finding an unused name in a resource dictionary.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method
          <function>QPDFPageObjectHelper::getFormXObjectForPage</function>
          for generating a form XObject equivalent to a page. The
          resulting object can be used in the same file or copied to
          another file with <function>copyForeignObject</function>.
          This can be useful for implementing underlay, overlay, n-up,
          thumbnails, or any other functionality requiring replication
          of pages in other contexts.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method
          <function>QPDFPageObjectHelper::placeFormXObject</function>
          for generating content stream text that places a given form
          XObject on a page, centered and fit within a specified
          rectangle. This method takes care of computing the proper
          transformation matrix and may optionally compensate for
          rotation or scaling of the destination page.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Build Improvements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Add new configure option
          <option>--enable-avoid-windows-handle</option>, which causes
          the preprocessor symbol
          <literal>AVOID_WINDOWS_HANDLE</literal> to be defined. When
          defined, qpdf will avoid referencing the Windows
          <classname>HANDLE</classname> type, which is disallowed with
          certain versions of the Windows SDK.
         </para>
        </listitem>
        <listitem>
         <para>
          For Windows builds, attempt to determine what options, if
          any, have to be passed to the compiler and linker to enable
          use of <function>wmain</function>. This causes the
          preprocessor symbol <literal>WINDOWS_WMAIN</literal> to be
          defined. If you do your own builds with other compilers, you
          can define this symbol to cause <function>wmain</function>
          to be used. This is needed to allow the Windows
          <command>qpdf</command> command to receive Unicode
          command-line options.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.3.0: January 7, 2019</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Command-line Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Shell completion: you can now use eval <command>$(qpdf
          --completion-bash)</command> and eval <command>$(qpdf
          --completion-zsh)</command> to enable shell completion for
          bash and zsh.
         </para>
        </listitem>
        <listitem>
         <para>
          Page numbers (also known as page labels) are now preserved
          when merging and splitting files with the
          <option>--pages</option> and <option>--split-pages</option>
          options.
         </para>
        </listitem>
        <listitem>
         <para>
          Bookmarks are partially preserved when splitting pages with
          the <option>--split-pages</option> option. Specifically, the
          outlines dictionary and some supporting metadata are copied
          into the split files. The result is that all bookmarks from
          the original file appear, those that point to pages that are
          preserved work, and those that point to pages that are not
          preserved don't do anything. This is an interim step toward
          proper support for bookmarks in splitting and merging
          operations.
         </para>
        </listitem>
        <listitem>
         <para>
          Page collation: add new option <option>--collate</option>.
          When specified, the semantics of <option>--pages</option>
          change from concatenation to collation. See <xref
          linkend="ref.page-selection"/> for examples and discussion.
         </para>
        </listitem>
        <listitem>
         <para>
          Generation of information in JSON format, primarily to
          facilitate use of qpdf from languages other than C++. Add
          new options <option>--json</option>,
          <option>--json-key</option>, and
          <option>--json-object</option> to generate a JSON
          representation of the PDF file. Run <command>qpdf
          --json-help</command> to get a description of the JSON
          format. For more information, see <xref linkend="ref.json"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--generate-appearances</option> flag will cause
          qpdf to generate appearances for form fields if the PDF file
          indicates that form field appearances are out of date. This
          can happen when PDF forms are filled in by a program that
          doesn't know how to regenerate the appearances of the
          filled-in fields.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--flatten-annotations</option> flag can be used
          to <emphasis>flatten</emphasis> annotations, including form
          fields. Ordinarily, annotations are drawn separately from
          the page. Flattening annotations is the process of combining
          their appearances into the page's contents. You might want
          to do this if you are going to rotate or combine pages using
          a tool that doesn't understand about annotations. You may
          also want to use <option>--generate-appearances</option>
          when using this flag since annotations for outdated form
          fields are not flattened as that would cause loss of
          information.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--optimize-images</option> flag tells qpdf to
          recompresses every image using DCT (JPEG) compression as
          long as the image is not already compressed with lossy
          compression and recompressing the image reduces its size.
          The additional options <option>--oi-min-width</option>,
          <option>--oi-min-height</option>, and
          <option>--oi-min-area</option> prevent recompression of
          images whose width, height, or pixel area
          (width&nbsp;&#xd7;&nbsp;height) are below a specified
          threshold.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--show-object</option> option can now be given
          as <option>--show-object=trailer</option> to show the
          trailer dictionary.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes and Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          QPDF now automatically detects and recovers from dangling
          references. If a PDF file contained an indirect reference to
          a non-existent object, which is valid, when adding a new
          object to the file, it was possible for the new object to
          take the object ID of the dangling reference, thereby
          causing the dangling reference to point to the new object.
          This case is now prevented.
         </para>
        </listitem>
        <listitem>
         <para>
          Fixes to form field setting code: strings are always written
          in UTF-16 format, and checkboxes and radio buttons are
          handled properly with respect to synchronization of values
          and appearance states.
         </para>
        </listitem>
        <listitem>
         <para>
          The <function>QPDF::checkLinearization()</function> no
          longer causes the program to crash when it detects problems
          with linearization data. Instead, it issues a normal warning
          or error.
         </para>
        </listitem>
        <listitem>
         <para>
          Ordinarily qpdf treats an argument of the form
          <option>@file</option> to mean that command-line options
          should be read from <filename>file</filename>. Now, if
          <filename>file</filename> does not exist but
          <filename>@file</filename> does, qpdf will treat
          <filename>@file</filename> as a regular option. This makes
          it possible to work more easily with PDF files whose names
          happen to start with the <literal>@</literal> character.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Library Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Remove the restriction in most cases that the source QPDF
          object used in a
          <function>QPDF::copyForeignObject</function> call has to
          stick around until the destination QPDF is written. The
          exceptional case is when the source stream gets is data
          using a QPDFObjectHandle::StreamDataProvider. For a more
          in-depth discussion, see comments around
          <function>copyForeignObject</function> in
          <filename>QPDF.hh</filename>.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new method
          <function>QPDFWriter::getFinalVersion()</function>, which
          returns the PDF version that will ultimately be written to
          the final file. See comments in
          <filename>QPDFWriter.hh</filename> for some restrictions on
          its use.
         </para>
        </listitem>
        <listitem>
         <para>
          Add several methods for transcoding strings to some of the
          character sets used in PDF files:
          <function>QUtil::utf8_to_ascii</function>,
          <function>QUtil::utf8_to_win_ansi</function>,
          <function>QUtil::utf8_to_mac_roman</function>, and
          <function>QUtil::utf8_to_utf16</function>. For the
          single-byte encodings that support only a limited character
          sets, these methods replace unsupported characters with a
          specified substitute.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new methods to
          <classname>QPDFAnnotationObjectHelper</classname> and
          <classname>QPDFFormFieldObjectHelper</classname> for
          querying flags and interpretation of different field types.
          Define constants in <filename>qpdf/Constants.h</filename> to
          help with interpretation of flag values.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new methods
          <function>QPDFAcroFormDocumentHelper::generateAppearancesIfNeeded</function>
          and
          <function>QPDFFormFieldObjectHelper::generateAppearance</function>
          for generating appearance streams. See discussion in
          <filename>QPDFFormFieldObjectHelper.hh</filename> for
          limitations.
         </para>
        </listitem>
        <listitem>
         <para>
          Add two new helper functions for dealing with resource
          dictionaries:
          <function>QPDFObjectHandle::getResourceNames()</function>
          returns a list of all second-level keys, which correspond to
          the names of resources, and
          <function>QPDFObjectHandle::mergeResources()</function>
          merges two resources dictionaries as long as they have
          non-conflicting keys. These methods are useful for certain
          types of objects that resolve resources from multiple places,
          such as form fields.
         </para>
        </listitem>
        <listitem>
         <para>
          Add methods
          <function>QPDFPageDocumentHelper::flattenAnnotations()</function>
          and
          <function>QPDFAnnotationObjectHelper::getPageContentForAppearance()</function>
          for handling low-level details of annotation flattening.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new helper classes:
          <classname>QPDFOutlineDocumentHelper</classname>,
          <classname>QPDFOutlineObjectHelper</classname>,
          <classname>QPDFPageLabelDocumentHelper</classname>,
          <classname>QPDFNameTreeObjectHelper</classname>, and
          <classname>QPDFNumberTreeObjectHelper</classname>.
         </para>
        </listitem>
        <listitem>
         <para>
          Add method <function>QPDFObjectHandle::getJSON()</function>
          that returns a JSON representation of the object. Call
          <function>serialize()</function> on the result to convert it
          to a string.
         </para>
        </listitem>
        <listitem>
         <para>
          Add a simple JSON serializer. This is not a complete or
          general-purpose JSON library. It allows assembly and
          serialization of JSON structures with some restrictions,
          which are described in the header file. This is the
          serializer used by qpdf's new JSON representation.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new <classname>QPDFObjectHandle::Matrix</classname>
          class along with a few convenience methods for dealing with
          six-element numerical arrays as matrices.
         </para>
        </listitem>
        <listitem>
         <para>
          Add new method
          <function>QPDFObjectHandle::wrapInArray</function>, which returns
          the object itself if it is an array, or an array containing
          the object otherwise. This is a common construct in PDF.
          This method prevents you from having to explicitly test
          whether something is a single element or an array.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Build Improvements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          It is no longer necessary to run
          <command>autogen.sh</command> to build from a pristine
          checkout. Automatically generated files are now committed so
          that it is possible to build on platforms without autoconf
          directly from a clean checkout of the repository. The
          <command>configure</command> script detects if the files are
          out of date when it also determines that the tools are
          present to regenerate them.
         </para>
        </listitem>
        <listitem>
         <para>
          Pull requests and the master branch are now built
          automatically in <ulink
          url="https://dev.azure.com/qpdf/qpdf/_build">Azure
          Pipelines</ulink>, which is free for open source projects.
          The build includes Linux, mac, Windows 32-bit and 64-bit
          with mingw and MSVC, and an AppImage build. Official qpdf
          releases are now built with Azure Pipelines.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Notes for Packagers
       </para>
       <itemizedlist>
        <listitem>
         <para>
          A new section has been added to the documentation with notes
          for packagers. Please see <xref linkend="ref.packaging"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          The qpdf detects out-of-date automatically generated files.
          If your packaging system automatically refreshes libtool or
          autoconf files, it could cause this check to fail. To avoid
          this problem, pass
          <option>--disable-check-autofiles</option> to
          <command>configure</command>.
         </para>
        </listitem>
        <listitem>
         <para>
          If you would like to have qpdf completion enabled
          automatically, you can install completion files in the
          distribution's default location. You can find sample
          completion files to install in the
          <filename>completions</filename> directory.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.2.1: August 18, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Command-line Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
	  Add
	  <option>--keep-files-open=<replaceable>[yn]</replaceable></option>
	  to override default determination of whether to keep files
	  open when merging. Please see the discussion of
	  <option>--keep-files-open</option> in <xref
	  linkend="ref.basic-options"/> for additional details.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.2.0: August 16, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Command-line Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
	  Add <option>--no-warn</option> option to suppress issuing
	  warning messages. If there are any conditions that would
	  have caused warnings to be issued, the exit status is still
	  3.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes and Optimizations
       </para>
       <itemizedlist>
        <listitem>
         <para>
	  Performance fix: optimize page merging operation to avoid
	  unnecessary open/close calls on files being merged. This
          solves a dramatic slow-down that was observed when merging
          certain types of files.
         </para>
        </listitem>
        <listitem>
         <para>
          Optimize how memory was used for the TIFF predictor,
          drastically improving performance and memory usage for files
          containing high-resolution images compressed with Flate
          using the TIFF predictor.
         </para>
        </listitem>
        <listitem>
         <para>
	  Bug fix: end of line characters were not properly handled
	  inside strings in some cases.
         </para>
        </listitem>
        <listitem>
         <para>
	  Bug fix: using <option>--progress</option> on very small
          files could cause an infinite loop.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        API enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
	  Add new class <classname>QPDFSystemError</classname>, derived
	  from <classname>std::runtime_error</classname>, which is now
	  thrown by <function>QUtil::throw_system_error</function>.
	  This enables the triggering <classname>errno</classname>
	  value to be retrieved.
         </para>
        </listitem>
        <listitem>
         <para>
	  Add <function>ClosedFileInputSource::stayOpen</function>
	  method, enabling a
	  <classname>ClosedFileInputSource</classname> to stay open
	  during manually indicated periods of high activity, thus
	  reducing the overhead of frequent open/close operations.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Build Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
	  For the mingw builds, change the name of the DLL import
	  library from <filename>libqpdf.a</filename> to
	  <filename>libqpdf.dll.a</filename> to more accurately
	  reflect that it is an import library rather than a static
	  library. This potentially clears the way for supporting a
	  static library in the future, though presently, the qpdf
	  Windows build only builds the DLL and executables.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.1.0: June 23, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Usability Improvements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          When splitting files, qpdf detects fonts and images that the
          document metadata claims are referenced from a page but are
          not actually referenced and omits them from the output file.
          This change can cause a significant reduction in the size of
          split PDF files for files created by some software packages.
          In some cases, it can also make page splitting slower. Prior
          versions of qpdf would believe the document metadata and
          sometimes include all the images from all the other pages
          even though the pages were no longer present. In the
          unlikely event that the old behavior should be desired, or
          if you have a case where page splitting is very slow, the
          old behavior (and speed) can be enabled by specifying
          <option>--preserve-unreferenced-resources</option>. For
          additional details, please see <xref
          linkend="ref.advanced-transformation"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          When merging multiple PDF files, qpdf no longer leaves all
          the files open. This makes it possible to merge numbers of
          files that may exceed the operating system's limit for the
          maximum number of open files.
         </para>
        </listitem>
        <listitem>
         <para>
          The <option>--rotate</option> option's syntax has been
          extended to make the page range optional. If you specify
          <option>--rotate=<replaceable>angle</replaceable></option>
          without specifying a page range, the rotation will be
          applied to all pages. This can be especially useful for
          adjusting a PDF created from a multi-page document that
          was scanned upside down.
         </para>
        </listitem>
        <listitem>
         <para>
          When merging multiple files, the <option>--verbose</option>
          option now prints information about each file as it operates
          on that file.
         </para>
        </listitem>
        <listitem>
         <para>
          When the <option>--progress</option> option is specified,
          qpdf will print a running indicator of its best guess at how
          far through the writing process it is. Note that, as with
          all progress meters, it's an approximation. This option is
          implemented in a way that makes it useful for software that
          uses the qpdf library; see API Enhancements below.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Properly decrypt files that use revision 3 of the standard
          security handler but use 40 bit keys (even though revision 3
          supports 128-bit keys).
         </para>
        </listitem>
        <listitem>
         <para>
          Limit depth of nested data structures to prevent crashes
          from certain types of malformed (malicious) PDFs.
         </para>
        </listitem>
        <listitem>
         <para>
          In &ldquo;newline before endstream&rdquo; mode, insert the
          required extra newline before the
          <literal>endstream</literal> at the end of object streams.
          This one case was previously omitted.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        API Enhancements
       </para>
       <itemizedlist>
        <listitem>
         <para>
          The first round of higher level &ldquo;helper&rdquo;
          interfaces has been introduced. These are designed to
          provide a more convenient way of interacting with certain
          document features than using
          <classname>QPDFObjectHandle</classname> directly. For
          details on helpers, see <xref
          linkend="ref.helper-classes"/>. Specific additional
          interfaces are described below.
         </para>
        </listitem>
        <listitem>
         <para>
          Add two new document helper classes:
          <classname>QPDFPageDocumentHelper</classname> for working
          with pages, and
          <classname>QPDFAcroFormDocumentHelper</classname> for
          working with interactive forms. No old methods have been
          removed, but <classname>QPDFPageDocumentHelper</classname>
          is now the preferred way to perform operations on pages
          rather than calling the old methods in
          <classname>QPDFObjectHandle</classname> and
          <classname>QPDF</classname> directly. Comments in the header
          files direct you to the new interfaces. Please see the
          header files and <filename>ChangeLog</filename> for
          additional details.
         </para>
        </listitem>
        <listitem>
         <para>
          Add three new object helper class:
          <classname>QPDFPageObjectHelper</classname> for pages,
          <classname>QPDFFormFieldObjectHelper</classname> for
          interactive form fields, and
          <classname>QPDFAnnotationObjectHelper</classname> for
          annotations. All three classes are fairly sparse at the
          moment, but they have some useful, basic functionality.
         </para>
        </listitem>
        <listitem>
         <para>
          A new example program
          <filename>examples/pdf-set-form-values.cc</filename> has
          been added that illustrates use of the new document and
          object helpers.
         </para>
        </listitem>
        <listitem>
         <para>
          The method
          <function>QPDFWriter::registerProgressReporter</function>
          has been added. This method allows you to register a
          function that is called by <classname>QPDFWriter</classname>
          to update your idea of the percentage it thinks it is
          through writing its output. Client programs can use this to
          implement reasonably accurate progress meters. The
          <command>qpdf</command> command line tool uses this to
          implement its <option>--progress</option> option.
         </para>
        </listitem>
        <listitem>
         <para>
          New methods
          <function>QPDFObjectHandle::newUnicodeString</function> and
          <function>QPDFObject::unparseBinary</function> have been
          added to allow for more convenient creation of strings that
          are explicitly encoded using big-endian UTF-16. This is
          useful for creating strings that appear outside of content
          streams, such as labels, form fields, outlines, document
          metadata, etc.
         </para>
        </listitem>
        <listitem>
         <para>
          A new class
          <classname>QPDFObjectHandle::Rectangle</classname> has been
          added to ease working with PDF rectangles, which are just
          arrays of four numeric values.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.0.2: March 6, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        When a loop is detected while following cross reference
        streams or tables, treat this as damage instead of silently
        ignoring the previous table. This prevents loss of otherwise
        recoverable data in some damaged files.
       </para>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
         Properly handle pages with no contents.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.0.1: March 4, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Disregard data check errors when uncompressing
        <literal>/FlateDecode</literal> streams. This is consistent
        with most other PDF readers and allows qpdf to recover data
        from another class of malformed PDF files.
       </para>
      </listitem>
      <listitem>
       <para>
        On the command line when specifying page ranges, support
        preceding a page number by &ldquo;r&rdquo; to indicate that it
        should be counted from the end. For example, the range
        <literal>r3-r1</literal> would indicate the last three pages
        of a document.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>8.0.0: February 25, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Packaging and Distribution Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          QPDF is now distributed as an <ulink
          url="https://appimage.org/">AppImage</ulink> in addition to
          all the other ways it is distributed. The AppImage can be
          found in the download area with the other packages. Thanks
          to Kurt Pfeifle and Simon Peter for their contributions.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          <function>QPDFObjectHandle::getUTF8Val</function> now
          properly treats non-Unicode strings as encoded with PDF Doc
          Encoding.
         </para>
        </listitem>
        <listitem>
         <para>
          Improvements to handling of objects in PDF files that are
          not of the expected type. In most cases, qpdf will be able
          to warn for such cases rather than fail with an exception.
          Previous versions of qpdf would sometimes fail with errors
          such as &ldquo;operation for dictionary object attempted on
          object of wrong type&rdquo;. This situation should be mostly
          or entirely eliminated now.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        Enhancements to the <command>qpdf</command> Command-line Tool.
        All new options listed here are documented in more detail in
        <xref linkend="ref.using"/>.
       </para>
       <itemizedlist>
        <listitem>
         <para>
          The option
          <option>--linearize-pass1=<replaceable>file</replaceable></option>
          has been added for debugging qpdf's linearization code.
         </para>
        </listitem>
        <listitem>
         <para>
          The option <option>--coalesce-contents</option> can be used
          to combine content streams of a page whose contents are an
          array of streams into a single stream.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
      <listitem>
       <para>
        API Enhancements. All new API calls are documented in their
        respective classes' header files. There are no non-compatible
        changes to the API.
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Add function <function>qpdf_check_pdf</function> to the C API.
          This function does basic checking that is a subset of what
          <command>qpdf --check</command> performs.
         </para>
        </listitem>
        <listitem>
         <para>
          Major enhancements to the lexical layer of qpdf. For a
          complete list of enhancements, please refer to the
          <filename>ChangeLog</filename> file. Most of the changes
          result in improvements to qpdf's ability handle erroneous
          files. It is also possible for programs to handle
          whitespace, comments, and inline images as tokens.
         </para>
        </listitem>
        <listitem>
         <para>
          New API for working with PDF content streams at a lexical
          level. The new class
          <classname>QPDFObjectHandle::TokenFilter</classname> allows
          the developer to provide token handlers. Token filters can be
          used with several different methods in
          <classname>QPDFObjectHandle</classname> as well as with a
          lower-level interface. See comments in
          <filename>QPDFObjectHandle.hh</filename> as well as the new
          examples <filename>examples/pdf-filter-tokens.cc</filename>
          and <filename>examples/pdf-count-strings.cc</filename> for
          details.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>7.1.1: February 4, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: files whose /ID fields were other than 16 bytes long
        can now be properly linearized
       </para>
      </listitem>
      <listitem>
       <para>
        A few compile and link issues have been corrected for some
        platforms.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>7.1.0: January 14, 2018</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        PDF files contain streams that may be compressed with various
        compression algorithms which, in some cases, may be enhanced
        by various predictor functions. Previously only the PNG up
        predictor was supported. In this version, all the PNG
        predictors as well as the TIFF predictor are supported. This
        increases the range of files that qpdf is able to handle.
       </para>
      </listitem>
      <listitem>
       <para>
        QPDF now allows a raw encryption key to be specified in place
        of a password when opening encrypted files, and will
        optionally display the encryption key used by a file. This is
        a non-standard operation, but it can be useful in certain
        situations. Please see the discussion of
        <option>--password-is-hex-key</option> in <xref
        linkend="ref.basic-options"/> or the comments around
        <function>QPDF::setPasswordIsHexKey</function> in
        <filename>QPDF.hh</filename> for additional details.
       </para>
      </listitem>
      <listitem>
       <para>
        Bug fix: numbers ending with a trailing decimal point are now
        properly recognized as numbers.
       </para>
      </listitem>
      <listitem>
       <para>
        Bug fix: when building qpdf from source on some platforms
        (especially MacOS), the build could get confused by older
        versions of qpdf installed on the system. This has been
        corrected.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>7.0.0: September 15, 2017</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Packaging and Distribution Changes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          QPDF's primary license is now <ulink
          url="http://www.apache.org/licenses/LICENSE-2.0">version 2.0
          of the Apache License</ulink> rather than version 2.0 of the
          Artistic License. You may still, at your option, consider
          qpdf to be licensed with version 2.0 of the Artistic
          license.
         </para>
        </listitem>
        <listitem>
         <para>
          QPDF no longer has a dependency on the PCRE (Perl-Compatible
          Regular Expression) library. QPDF now has an added
          dependency on the JPEG library.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
        Bug Fixes
       </para>
       <itemizedlist>
        <listitem>
         <para>
          This release contains many bug fixes for various infinite
          loops, memory leaks, and other memory errors that could be
          encountered with specially crafted or otherwise erroneous
          PDF files.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
        New Features
       </para>
       <itemizedlist>
        <listitem>
         <para>
          QPDF now supports reading and writing streams encoded with
          JPEG or RunLength encoding. Library API enhancements and
          command-line options have been added to control this
          behavior. See command-line options
          <option>--compress-streams</option> and
          <option>--decode-level</option> and methods
          <function>QPDFWriter::setCompressStreams</function> and
          <function>QPDFWriter::setDecodeLevel</function>.
         </para>
        </listitem>
        <listitem>
         <para>
          QPDF is much better at recovering from broken files. In most
          cases, qpdf will skip invalid objects and will preserve
          broken stream data by not attempting to filter broken
          streams. QPDF is now able to recover or at least not crash
          on dozens of broken test files I have received over the past
          few years.
         </para>
        </listitem>
        <listitem>
         <para>
          Page rotation is now supported and accessible from both the
          library and the command line.
         </para>
        </listitem>
        <listitem>
         <para>
          <classname>QPDFWriter</classname> supports writing files in
          a way that preserves PCLm compliance in support of
          driverless printing. This is very specialized and is only
          useful to applications that already know how to create PCLm
          files.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
        Enhancements to the <command>qpdf</command> Command-line Tool.
        All new options listed here are documented in more detail in
        <xref linkend="ref.using"/>.
       </para>
       <itemizedlist>
        <listitem>
         <para>
          Command-line arguments can now be read from files or
          standard input using <literal>@file</literal> or
          <literal>@-</literal> syntax. Please see <xref
          linkend="ref.invocation"/>.
         </para>
        </listitem>
        <listitem>
         <para>
          <option>--rotate</option>: request page rotation
         </para>
        </listitem>
        <listitem>
         <para>
          <option>--newline-before-endstream</option>: ensure that a
          newline appears before every <literal>endstream</literal>
          keyword in the file; used to prevent qpdf from breaking
          PDF/A compliance on already compliant files.
         </para>
        </listitem>
        <listitem>
         <para>
          <option>--preserve-unreferenced</option>: preserve
          unreferenced objects in the input PDF
         </para>
        </listitem>
        <listitem>
         <para>
	  <option>--split-pages</option>: break output into chunks
          with fixed numbers of pages
         </para>
        </listitem>
        <listitem>
         <para>
	  <option>--verbose</option>: print the name of each output
          file that is created
         </para>
        </listitem>
        <listitem>
         <para>
          <option>--compress-streams</option> and
          <option>--decode-level</option> replace
          <option>--stream-data</option> for improving granularity of
          controlling compression and decompression of stream data.
          The <option>--stream-data</option> option will remain
          available.
         </para>
        </listitem>
        <listitem>
         <para>
          When running <command>qpdf --check</command> with other
          options, checks are always run first. This enables qpdf to
          perform its full recovery logic before outputting other
          information. This can be especially useful when manually
          recovering broken files, looking at qpdf's regenerated cross
          reference table, or other similar operations.
         </para>
        </listitem>
        <listitem>
         <para>
          Process <command>--pages</command> earlier so that other
          options like <option>--show-pages</option> or
          <option>--split-pages</option> can operate on the file after
          page splitting/merging has occurred.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
     <itemizedlist>
      <listitem>
       <para>
        API Changes. All new API calls are documented in their
        respective classes' header files.
       </para>
       <itemizedlist>
        <listitem>
         <para>
          <function>QPDFObjectHandle::rotatePage</function>: apply
          rotation to a page object
         </para>
        </listitem>
        <listitem>
         <para>
          <function>QPDFWriter::setNewlineBeforeEndstream</function>:
          force newline to appear before <literal>endstream</literal>
         </para>
        </listitem>
        <listitem>
         <para>
          <function>QPDFWriter::setPreserveUnreferencedObjects</function>:
          preserve unreferenced objects that appear in the input PDF.
          The default behavior is to discard them.
         </para>
        </listitem>
        <listitem>
         <para>
          New <classname>Pipeline</classname> types
          <classname>Pl_RunLength</classname> and
          <classname>Pl_DCT</classname> are available for developers
          who wish to produce or consume RunLength or DCT stream data
          directly. The <filename>examples/pdf-create.cc</filename>
          example illustrates their use.
         </para>
        </listitem>
        <listitem>
         <para>
	  <function>QPDFWriter::setCompressStreams</function> and
	  <function>QPDFWriter::setDecodeLevel</function> methods
	  control handling of different types of stream compression.
         </para>
        </listitem>
        <listitem>
         <para>
	  Add new C API functions
	  <function>qpdf_set_compress_streams</function>,
	  <function>qpdf_set_decode_level</function>,
	  <function>qpdf_set_preserve_unreferenced_objects</function>,
	  and <function>qpdf_set_newline_before_endstream</function>
	  corresponding to the new <classname>QPDFWriter</classname>
	  methods.
         </para>
        </listitem>
       </itemizedlist>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>6.0.0: November 10, 2015</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Implement <option>--deterministic-id</option> command-line
        option and <function>QPDFWriter::setDeterministicID</function>
        as well as C API function
        <function>qpdf_set_deterministic_ID</function> for generating
        a deterministic ID for non-encrypted files. When this option
        is selected, the ID of the file depends on the contents of the
        output file, and not on transient items such as the timestamp
        or output file name.
       </para>
      </listitem>
      <listitem>
       <para>
        Make qpdf more tolerant of files whose xref table entries are
        not the correct length.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.1.3: May 24, 2015</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: fix-qdf was not properly handling files that
        contained object streams with more than 255 objects in them.
       </para>
      </listitem>
      <listitem>
       <para>
        Bug fix: qpdf was not properly initializing Microsoft's secure
        crypto provider on fresh Windows installations that had not
        had any keys created yet.
       </para>
      </listitem>
      <listitem>
       <para>
        Fix a few errors found by Gynvael Coldwind and
	Mateusz Jurczyk of the Google Security Team. Please see the
        ChangeLog for details.
       </para>
      </listitem>
      <listitem>
       <para>
        Properly handle pages that have no contents at all. There were
        many cases in which qpdf handled this fine, but a few methods
        blindly obtained page contents with handling the possibility
        that there were no contents.
       </para>
      </listitem>
      <listitem>
       <para>
        Make qpdf more robust for a few more kinds of problems that
        may occur in invalid PDF files.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.1.2: June 7, 2014</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: linearizing files could create a corrupted output
        file under extremely unlikely file size circumstances. See
        ChangeLog for details. The odds of getting hit by this are
        very low, though one person did.
       </para>
      </listitem>
      <listitem>
       <para>
        Bug fix: qpdf would fail to write files that had streams with
        decode parameters referencing other streams.
       </para>
      </listitem>
      <listitem>
       <para>
        New example program: <command>pdf-split-pages</command>:
        efficiently split PDF files into individual pages. The example
        program does this more efficiently than using <command>qpdf
        --pages</command> to do it.
       </para>
      </listitem>
      <listitem>
       <para>
        Packaging fix: Visual C++ binaries did not support Windows XP.
        This has been rectified by updating the compilers used to
        generate the release binaries.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.1.1: January 14, 2014</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Performance fix: copying foreign objects could be very slow
        with certain types of files.  This was most likely to be
        visible during page splitting and was due to traversing the
        same objects multiple times in some cases.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.1.0: December 17, 2013</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Added runtime option
        (<function>QUtil::setRandomDataProvider</function>) to supply
        your own random data provider.  You can use this if you want
        to avoid using the OS-provided secure random number generation
        facility or stdlib's less secure version.  See comments in
        include/qpdf/QUtil.hh for details.
       </para>
      </listitem>
      <listitem>
       <para>
        Fixed image comparison tests to not create 12-bit-per-pixel
        images since some versions of tiffcmp have bugs in comparing
        them in some cases.  This increases the disk space required by
        the image comparison tests, which are off by default anyway.
       </para>
      </listitem>
      <listitem>
       <para>
        Introduce a number of small fixes for compilation on the
        latest clang in MacOS and the latest Visual C++ in Windows.
       </para>
      </listitem>
      <listitem>
       <para>
        Be able to handle broken files that end the xref table header
        with a space instead of a newline.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.0.1: October 18, 2013</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Thanks to a detailed review by Florian Weimer and the Red Hat
        Product Security Team, this release includes a number of
        non-user-visible security hardening changes.  Please see the
        ChangeLog file in the source distribution for the complete
        list.
       </para>
      </listitem>
      <listitem>
       <para>
        When available, operating system-specific secure random number
        generation is used for generating initialization vectors and
        other random values used during encryption or file creation.
        For the Windows build, this results in an added dependency on
        Microsoft's cryptography API.  To disable the OS-specific
        cryptography and use the old version, pass the
        <option>--enable-insecure-random</option> option to
        <command>./configure</command>.
       </para>
      </listitem>
      <listitem>
       <para>
        The <command>qpdf</command> command-line tool now issues a
        warning when <option>-accessibility=n</option> is specified
        for newer encryption versions stating that the option is
        ignored.  qpdf, per the spec, has always ignored this flag,
        but it previously did so silently.  This warning is issued
        only by the command-line tool, not by the library.  The
        library's handling of this flag is unchanged.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>5.0.0: July 10, 2013</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: previous versions of qpdf would lose objects with
        generation != 0 when generating object streams.  Fixing this
        required changes to the public API.
       </para>
      </listitem>
      <listitem>
       <para>
        Removed methods from public API that were only supposed to be
        called by QPDFWriter and couldn't realistically be called
        anywhere else.  See ChangeLog for details.
       </para>
      </listitem>
      <listitem>
       <para>
        New <type>QPDFObjGen</type> class added to represent an object
        ID/generation pair.
        <function>QPDFObjectHandle::getObjGen()</function> is now
        preferred over
        <function>QPDFObjectHandle::getObjectID()</function> and
        <function>QPDFObjectHandle::getGeneration()</function> as it
        makes it less likely for people to accidentally write code
        that ignores the generation number.  See
        <filename>QPDF.hh</filename> and
        <filename>QPDFObjectHandle.hh</filename> for additional notes.
       </para>
      </listitem>
      <listitem>
       <para>
        Add <option>--show-npages</option> command-line option to the
        <command>qpdf</command> command to show the number of pages in
        a file.
       </para>
      </listitem>
      <listitem>
       <para>
        Allow omission of the page range within
        <option>--pages</option> for the <command>qpdf</command>
        command.  When omitted, the page range is implicitly taken to
        be all the pages in the file.
       </para>
      </listitem>
      <listitem>
       <para>
        Various enhancements were made to support different types of
        broken files or broken readers.  Details can be found in
        <filename>ChangeLog</filename>.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>4.1.0: April 14, 2013</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Note to people including qpdf in distributions: the
        <filename>.la</filename> files generated by libtool are now
        installed by qpdf's <command>make install</command> target.
        Before, they were not installed.  This means that if your
        distribution does not want to include <filename>.la</filename>
        files, you must remove them as part of your packaging process.
       </para>
      </listitem>
      <listitem>
       <para>
        Major enhancement: API enhancements have been made to support
        parsing of content streams.  This enhancement includes the
        following changes:
        <itemizedlist>
         <listitem>
          <para>
           <function>QPDFObjectHandle::parseContentStream</function>
           method parses objects in a content stream and calls
           handlers in a callback class.  The example
           <filename>examples/pdf-parse-content.cc</filename>
           illustrates how this may be used.
          </para>
         </listitem>
         <listitem>
          <para>
           <type>QPDFObjectHandle</type> can now represent operators
           and inline images, object types that may only appear in
           content streams.
          </para>
         </listitem>
         <listitem>
          <para>
           Method <function>QPDFObjectHandle::getTypeCode()</function>
           returns an enumerated type value representing the
           underlying object type.  Method
           <function>QPDFObjectHandle::getTypeName()</function>
           returns a text string describing the name of the type of a
           <type>QPDFObjectHandle</type> object.  These methods can be
           used for more efficient parsing and debugging/diagnostic
           messages.
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
      <listitem>
       <para>
        <command>qpdf --check</command> now parses all pages' content
        streams in addition to doing other checks.  While there are
        still many types of errors that cannot be detected, syntactic
        errors in content streams will now be reported.
       </para>
      </listitem>
      <listitem>
       <para>
        Minor compilation enhancements have been made to facilitate
        easier for support for a broader range of compilers and
        compiler versions.
        <itemizedlist>
         <listitem>
          <para>
           Warning flags have been moved into a separate variable in
           <filename>autoconf.mk</filename>
          </para>
         </listitem>
         <listitem>
          <para>
           The configure flag <option>--enable-werror</option> work
           for Microsoft compilers
          </para>
         </listitem>
         <listitem>
          <para>
           All MSVC CRT security warnings have been resolved.
          </para>
         </listitem>
         <listitem>
          <para>
           All C-style casts in C++ Code have been replaced by C++
           casts, and many casts that had been included to suppress
           higher warning levels for some compilers have been removed,
           primarily for clarity.  Places where integer type coercion
           occurs have been scrutinized.  A new casting policy has
           been documented in the manual.  This is of concern mainly
           to people porting qpdf to new platforms or compilers.  It
           is not visible to programmers writing code that uses the
           library
          </para>
         </listitem>
         <listitem>
          <para>
           Some internal limits have been removed in code that
           converts numbers to strings.  This is largely invisible to
           users, but it does trigger a bug in some older versions of
           mingw-w64's C++ library.  See
           <filename>README-windows.md</filename> in the source
           distribution if you think this may affect you.  The copy of
           the DLL distributed with qpdf's binary distribution is not
           affected by this problem.
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
      <listitem>
       <para>
        The RPM spec file previously included with qpdf has been
        removed.  This is because virtually all Linux distributions
        include qpdf now that it is a dependency of CUPS filters.
       </para>
      </listitem>
      <listitem>
       <para>
        A few bug fixes are included:
        <itemizedlist>
         <listitem>
          <para>
           Overridden compressed objects are properly handled.
           Before, there were certain constructs that could cause qpdf
           to see old versions of some objects.  The most usual
           manifestation of this was loss of filled in form values for
           certain files.
          </para>
         </listitem>
         <listitem>
          <para>
           Installation no longer uses GNU/Linux-specific versions of
           some commands, so <command>make install</command> works on
           Solaris with native tools.
          </para>
         </listitem>
         <listitem>
          <para>
           The 64-bit mingw Windows binary package no longer includes
           a 32-bit DLL.
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>4.0.1: January 17, 2013</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Fix detection of binary attachments in test suite to avoid
        false test failures on some platforms.
       </para>
      </listitem>
      <listitem>
       <para>
        Add clarifying comment in <filename>QPDF.hh</filename> to
        methods that return the user password explaining that it is no
        longer possible with newer encryption formats to recover the
        user password knowing the owner password.  In earlier
        encryption formats, the user password was encrypted in the
        file using the owner password.  In newer encryption formats, a
        separate encryption key is used on the file, and that key is
        independently encrypted using both the user password and the
        owner password.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>4.0.0: December 31, 2012</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Major enhancement: support has been added for newer encryption
        schemes supported by version X of Adobe Acrobat.  This
        includes use of 127-character passwords, 256-bit encryption
        keys, and the encryption scheme specified in ISO 32000-2, the
        PDF 2.0 specification.  This scheme can be chosen from the
        command line by specifying use of 256-bit keys.  qpdf also
        supports the deprecated encryption method used by Acrobat IX.
        This encryption style has known security weaknesses and should
        not be used in practice.  However, such files exist &ldquo;in
        the wild,&rdquo; so support for this scheme is still useful.
        New methods
        <function>QPDFWriter::setR6EncryptionParameters</function>
        (for the PDF 2.0 scheme) and
        <function>QPDFWriter::setR5EncryptionParameters</function>
        (for the deprecated scheme) have been added to enable these
        new encryption schemes.  Corresponding functions have been
        added to the C API as well.
       </para>
      </listitem>
      <listitem>
       <para>
        Full support for Adobe extension levels in PDF version
        information.  Starting with PDF version 1.7, corresponding to
        ISO 32000, Adobe adds new functionality by increasing the
        extension level rather than increasing the version.  This
        support includes addition of the
        <function>QPDF::getExtensionLevel</function> method for
        retrieving the document's extension level, addition of
        versions of
        <function>QPDFWriter::setMinimumPDFVersion</function> and
        <function>QPDFWriter::forcePDFVersion</function> that accept
        an extension level, and extended syntax for specifying forced
        and minimum versions on the command line as described in <xref
        linkend="ref.advanced-transformation"/>.  Corresponding
        functions have been added to the C API as well.
       </para>
      </listitem>
      <listitem>
       <para>
        Minor fixes to prevent qpdf from referencing objects in the
        file that are not referenced in the file's overall structure.
        Most files don't have any such objects, but some files have
        contain unreferenced objects with errors, so these fixes
        prevent qpdf from needlessly rejecting or complaining about
        such objects.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new generalized methods for reading and writing files
        from/to programmer-defined sources.  The method
        <function>QPDF::processInputSource</function> allows the
        programmer to use any input source for the input file, and
        <function>QPDFWriter::setOutputPipeline</function> allows the
        programmer to write the output file through any pipeline.
        These methods would make it possible to perform any number of
        specialized operations, such as accessing external storage
        systems, creating bindings for qpdf in other programming
        languages that have their own I/O systems, etc.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new method <function>QPDF::getEncryptionKey</function> for
        retrieving the underlying encryption key used in the file.
       </para>
      </listitem>
      <listitem>
       <para>
        This release includes a small handful of non-compatible API
        changes.  While effort is made to avoid such changes, all the
        non-compatible API changes in this version were to parts of
        the API that would likely never be used outside the library
        itself.  In all cases, the altered methods or structures were
        parts of the <classname>QPDF</classname> that were public to
        enable them to be called from either
        <classname>QPDFWriter</classname> or were part of validation
        code that was over-zealous in reporting problems in parts of
        the file that would not ordinarily be referenced.  In no case
        did any of the removed methods do anything worse that falsely
        report error conditions in files that were broken in ways that
        didn't matter.  The following public parts of the
        <classname>QPDF</classname> class were changed in a
        non-compatible way:
        <itemizedlist>
         <listitem>
          <para>
           Updated nested <classname>QPDF::EncryptionData</classname>
           class to add fields needed by the newer encryption formats,
           member variables changed to private so that future changes
           will not require breaking backward compatibility.
          </para>
         </listitem>
         <listitem>
          <para>
           Added additional parameters to
           <function>compute_data_key</function>, which is used by
           <classname>QPDFWriter</classname> to compute the encryption
           key used to encrypt a specific object.
          </para>
         </listitem>
         <listitem>
          <para>
           Removed the method
           <function>flattenScalarReferences</function>.  This method
           was previously used prior to writing a new PDF file, but it
           has the undesired side effect of causing qpdf to read
           objects in the file that were not referenced.  Some
           otherwise files have unreferenced objects with errors in
           them, so this could cause qpdf to reject files that would
           be accepted by virtually all other PDF readers.  In fact,
           qpdf relied on only a very small part of what
           flattenScalarReferences did, so only this part has been
           preserved, and it is now done directly inside
           <classname>QPDFWriter</classname>.
          </para>
         </listitem>
         <listitem>
          <para>
           Removed the method <function>decodeStreams</function>.
           This method was used by the <option>--check</option> option
           of the <command>qpdf</command> command-line tool to force
           all streams in the file to be decoded, but it also suffered
           from the problem of opening otherwise unreferenced streams
           and thus could report false positive.  The
           <option>--check</option> option now causes qpdf to go
           through all the motions of writing a new file based on the
           original one, so it will always reference and check exactly
           those parts of a file that any ordinary viewer would check.
          </para>
         </listitem>
         <listitem>
          <para>
           Removed the method
           <function>trimTrailerForWrite</function>.  This method was
           used by <classname>QPDFWriter</classname> to modify the
           original QPDF object by removing fields from the trailer
           dictionary that wouldn't apply to the newly written file.
           This functionality, though generally harmless, was a poor
           implementation and has been replaced by having QPDFWriter
           filter these out when copying the trailer rather than
           modifying the original QPDF object.  (Note that qpdf never
           modifies the original file itself.)
          </para>
         </listitem>
        </itemizedlist>
       </para>
      </listitem>
      <listitem>
       <para>
        Allow the PDF header to appear anywhere in the first 1024
        bytes of the file.  This is consistent with what other readers
        do.
       </para>
      </listitem>
      <listitem>
       <para>
        Fix the <command>pkg-config</command> files to list zlib and
        pcre in <function>Requires.private</function> to better
        support static linking using <command>pkg-config</command>.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>3.0.2: September 6, 2012</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: <function>QPDFWriter::setOutputMemory</function> did
        not work when not used with
        <function>QPDFWriter::setStaticID</function>, which made it
        pretty much useless.  This has been fixed.
       </para>
      </listitem>
      <listitem>
       <para>
        New API call
        <function>QPDFWriter::setExtraHeaderText</function> inserts
        additional text near the header of the PDF file.  The intended
        use case is to insert comments that may be consumed by a
        downstream application, though other use cases may exist.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>3.0.1: August 11, 2012</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Version 3.0.0 included addition of files for
        <command>pkg-config</command>, but this was not mentioned in
        the release notes.  The release notes for 3.0.0 were updated
        to mention this.
       </para>
      </listitem>
      <listitem>
       <para>
        Bug fix: if an object stream ended with a scalar object not
        followed by space, qpdf would incorrectly report that it
        encountered a premature EOF.  This bug has been in qpdf since
        version&nbsp;2.0.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>3.0.0: August 2, 2012</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Acknowledgment: I would like to express gratitude for the
        contributions of Tobias Hoffmann toward the release of qpdf
        version 3.0.  He is responsible for most of the implementation
        and design of the new API for manipulating pages, and
        contributed code and ideas for many of the improvements made
        in version 3.0.  Without his work, this release would
        certainly not have happened as soon as it did, if at all.
       </para>
      </listitem>
      <listitem>
       <para>
        <emphasis>Non-compatible API change:</emphasis> The version of
        <function>QPDFObjectHandle::replaceStreamData</function> that
        uses a <classname>StreamDataProvider</classname> no longer
        requires (or accepts) a <varname>length</varname> parameter.
        See <xref linkend="ref.upgrading-to-3.0"/> for an explanation.
        While care is taken to avoid non-compatible API changes in
        general, an exception was made this time because the new
        interface offers an opportunity to significantly simplify
        calling code.
       </para>
      </listitem>
      <listitem>
       <para>
        Support has been added for large files.  The test suite
        verifies support for files larger than 4 gigabytes, and manual
        testing has verified support for files larger than 10
        gigabytes.  Large file support is available for both 32-bit
        and 64-bit platforms as long as the compiler and underlying
        platforms support it.
       </para>
      </listitem>
      <listitem>
       <para>
        Support for page selection (splitting and merging PDF files)
        has been added to the <command>qpdf</command> command-line
        tool.  See <xref linkend="ref.page-selection"/>.
       </para>
      </listitem>
      <listitem>
       <para>
        Options have been added to the <command>qpdf</command>
        command-line tool for copying encryption parameters from
        another file.  See <xref linkend="ref.basic-options"/>.
       </para>
      </listitem>
      <listitem>
       <para>
        New methods have been added to the <classname>QPDF</classname>
        object for adding and removing pages.  See <xref
        linkend="ref.adding-and-remove-pages"/>.
       </para>
      </listitem>
      <listitem>
       <para>
        New methods have been added to the <classname>QPDF</classname>
        object for copying objects from other PDF files.  See <xref
        linkend="ref.foreign-objects"/>
       </para>
      </listitem>
      <listitem>
       <para>
        A new method <function>QPDFObjectHandle::parse</function> has
        been added for constructing
        <classname>QPDFObjectHandle</classname> objects from a string
        description.
       </para>
      </listitem>
      <listitem>
       <para>
        Methods have been added to <classname>QPDFWriter</classname>
        to allow writing to an already open stdio <type>FILE*</type>
        addition to writing to standard output or a named file.
        Methods have been added to <classname>QPDF</classname> to be
        able to process a file from an already open stdio
        <type>FILE*</type>.  This makes it possible to read and write
        PDF from secure temporary files that have been unlinked prior
        to being fully read or written.
       </para>
      </listitem>
      <listitem>
       <para>
        The <function>QPDF::emptyPDF</function> can be used to allow
        creation of PDF files from scratch.  The example
        <filename>examples/pdf-create.cc</filename> illustrates how it
        can be used.
       </para>
      </listitem>
      <listitem>
       <para>
        Several methods to take
        <classname>PointerHolder&lt;Buffer&gt;</classname> can now
        also accept <type>std::string</type> arguments.
       </para>
      </listitem>
      <listitem>
       <para>
        Many new convenience methods have been added to the library,
        most in <classname>QPDFObjectHandle</classname>.  See
        <filename>ChangeLog</filename> for a full list.
       </para>
      </listitem>
      <listitem>
       <para>
        When building on a platform that supports ELF shared libraries
        (such as Linux), symbol versions are enabled by default.  They
        can be disabled by passing
        <option>--disable-ld-version-script</option> to
        <command>./configure</command>.
       </para>
      </listitem>
      <listitem>
       <para>
        The file <filename>libqpdf.pc</filename> is now installed to
        support <command>pkg-config</command>.
       </para>
      </listitem>
      <listitem>
       <para>
        Image comparison tests are off by default now since they are
        not needed to verify a correct build or port of qpdf.  They
        are needed only when changing the actual PDF output generated
        by qpdf.  You should enable them if you are making deep
        changes to qpdf itself.  See <filename>README.md</filename> for
        details.
       </para>
      </listitem>
      <listitem>
       <para>
        Large file tests are off by default but can be turned on with
        <command>./configure</command> or by setting an environment
        variable before running the test suite.  See
        <filename>README.md</filename> for details.
       </para>
      </listitem>
      <listitem>
       <para>
        When qpdf's test suite fails, failures are not printed to the
        terminal anymore by default.  Instead, find them in
        <filename>build/qtest.log</filename>.  For packagers who are
        building with an autobuilder, you can add the
        <option>--enable-show-failed-test-output</option> option to
        <command>./configure</command> to restore the old behavior.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.3.1: December 28, 2011</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Fix thread-safety problem resulting from non-thread-safe use
        of the PCRE library.
       </para>
      </listitem>
      <listitem>
       <para>
        Made a few minor documentation fixes.
       </para>
      </listitem>
      <listitem>
       <para>
        Add workaround for a bug that appears in some versions of
        ghostscript to the test suite
       </para>
      </listitem>
      <listitem>
       <para>
        Fix minor build issue for Visual C++ 2010.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.3.0: August 11, 2011</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Bug fix: when preserving existing encryption on encrypted
        files with cleartext metadata, older qpdf versions would
        generate password-protected files with no valid password.
        This operation now works.  This bug only affected files
        created by copying existing encryption parameters; explicit
        encryption with specification of cleartext metadata worked
        before and continues to work.
       </para>
      </listitem>
      <listitem>
       <para>
        Enhance <classname>QPDFWriter</classname> with a new
        constructor that allows you to delay the specification of the
        output file.  When using this constructor, you may now call
        <function>QPDFWriter::setOutputFilename</function> to specify
        the output file, or you may use
        <function>QPDFWriter::setOutputMemory</function> to cause
        <classname>QPDFWriter</classname> to write the resulting PDF
        file to a memory buffer.  You may then use
        <function>QPDFWriter::getBuffer</function> to retrieve the
        memory buffer.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new API call <function>QPDF::replaceObject</function> for
        replacing objects by object ID
       </para>
      </listitem>
      <listitem>
       <para>
        Add new API call <function>QPDF::swapObjects</function> for
        swapping two objects by object ID
       </para>
      </listitem>
      <listitem>
       <para>
        Add <function>QPDFObjectHandle::getDictAsMap</function> and
        <function>QPDFObjectHandle::getArrayAsVector</function> to
        allow retrieval of dictionary objects as maps and array
        objects as vectors.
       </para>
      </listitem>
      <listitem>
       <para>
        Add functions <function>qpdf_get_info_key</function> and
        <function>qpdf_set_info_key</function> to the C API for
        manipulating string fields of the document's
        <literal>/Info</literal> dictionary.
       </para>
      </listitem>
      <listitem>
       <para>
        Add functions <function>qpdf_init_write_memory</function>,
        <function>qpdf_get_buffer_length</function>, and
        <function>qpdf_get_buffer</function> to the C API for writing
        PDF files to a memory buffer instead of a file.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.2.4: June 25, 2011</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Fix installation and compilation issues; no functionality
        changes.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.2.3: April 30, 2011</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Handle some damaged streams with incorrect characters
        following the stream keyword.
       </para>
      </listitem>
      <listitem>
       <para>
        Improve handling of inline images when normalizing content
        streams.
       </para>
      </listitem>
      <listitem>
       <para>
        Enhance error recovery to properly handle files that use
        object 0 as a regular object, which is specifically disallowed
        by the spec.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.2.2: October 4, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Add new function <function>qpdf_read_memory</function>
        to the C API to call
        <function>QPDF::processMemoryFile</function>.  This was an
        omission in qpdf 2.2.1.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.2.1: October 1, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Add new method <function>QPDF::setOutputStreams</function>
        to replace <varname>std::cout</varname> and
        <varname>std::cerr</varname> with other streams for generation
        of diagnostic messages and error messages.  This can be useful
        for GUIs or other applications that want to capture any output
        generated by the library to present to the user in some other
        way.  Note that QPDF does not write to
        <varname>std::cout</varname> (or the specified output stream)
        except where explicitly mentioned in
        <filename>QPDF.hh</filename>, and that the only use of the
        error stream is for warnings.  Note also that output of
        warnings is suppressed when
        <literal>setSuppressWarnings(true)</literal> is called.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new method <function>QPDF::processMemoryFile</function>
        for operating on PDF files that are loaded into memory rather
        than in a file on disk.
       </para>
      </listitem>
      <listitem>
       <para>
        Give a warning but otherwise ignore empty PDF objects by
        treating them as null.  Empty object are not permitted by the
        PDF specification but have been known to appear in some actual
        PDF files.
       </para>
      </listitem>
      <listitem>
       <para>
        Handle inline image filter abbreviations when the appear as
        stream filter abbreviations.  The PDF specification does not
        allow use of stream filter abbreviations in this way, but
        Adobe Reader and some other PDF readers accept them since they
        sometimes appear incorrectly in actual PDF files.
       </para>
      </listitem>
      <listitem>
       <para>
        Implement miscellaneous enhancements to
        <classname>PointerHolder</classname> and
        <classname>Buffer</classname> to support other changes.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.2.0: August 14, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Add new methods to <classname>QPDFObjectHandle</classname>
        (<function>newStream</function> and
        <function>replaceStreamData</function> for creating new
        streams and replacing stream data.  This makes it possible to
        perform a wide range of operations that were not previously
        possible.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new helper method in
        <classname>QPDFObjectHandle</classname>
        (<function>addPageContents</function>) for appending or
        prepending new content streams to a page.  This method makes
        it possible to manipulate content streams without having to be
        concerned whether a page's contents are a single stream or an
        array of streams.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new method in <classname>QPDFObjectHandle</classname>:
        <function>replaceOrRemoveKey</function>, which replaces a
        dictionary key
        with a given value unless the value is null, in which case it
        removes the key instead.
       </para>
      </listitem>
      <listitem>
       <para>
        Add new method in <classname>QPDFObjectHandle</classname>:
        <function>getRawStreamData</function>, which returns the raw
        (unfiltered) stream data into a buffer.  This complements the
        <function>getStreamData</function> method, which returns the
        filtered (uncompressed) stream data and can only be used when
        the stream's data is filterable.
       </para>
      </listitem>
      <listitem>
       <para>
        Provide two new examples:
        <command>pdf-double-page-size</command> and
        <command>pdf-invert-images</command> that illustrate the newly
        added interfaces.
       </para>
      </listitem>
      <listitem>
       <para>
        Fix a memory leak that would cause loss of a few bytes for
        every object involved in a cycle of object references.  Thanks
        to Jian Ma for calling my attention to the leak.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1.5: April 25, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Remove restriction of file identifier strings to 16 bytes.
        This unnecessary restriction was preventing qpdf from being
        able to encrypt or decrypt files with identifier strings that
        were not exactly 16 bytes long.  The specification imposes no
        such restriction.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1.4: April 18, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Apply the same padding calculation fix from version 2.1.2 to
        the main cross reference stream as well.
       </para>
      </listitem>
      <listitem>
       <para>
        Since <command>qpdf --check</command> only performs limited
        checks, clarify the output to make it clear that there still
        may be errors that qpdf can't check.  This should make it less
        surprising to people when another PDF reader is unable to read
        a file that qpdf thinks is okay.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1.3: March 27, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Fix bug that could cause a failure when rewriting PDF files
        that contain object streams with unreferenced objects that in
        turn reference indirect scalars.
       </para>
      </listitem>
      <listitem>
       <para>
        Don't complain about (invalid) AES streams that aren't a
        multiple of 16 bytes.  Instead, pad them before decrypting.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1.2: January 24, 2010</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Fix bug in padding around first half cross reference stream in
        linearized files.  The bug could cause an assertion failure
        when linearizing certain unlucky files.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1.1: December 14, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        No changes in functionality; insert missing include in an
        internal library header file to support gcc 4.4, and update
        test suite to ignore broken Adobe Reader installations.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.1: October 30, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        This is the first version of qpdf to include Windows support.
        On Windows, it is possible to build a DLL. Additionally, a
        partial C-language API has been introduced, which makes it
        possible to call qpdf functions from non-C++ environments. I
        am very grateful to Žarko <!-- Gajić --> Gajic (<ulink
        url="http://zarko-gajic.iz.hr/">http://zarko-gajic.iz.hr/</ulink>)
        for tirelessly testing numerous pre-release versions of this
        DLL and providing many excellent suggestions on improving the
        interface.
       </para>
       <para>
        For programming to the C interface, please see the header file
        <filename>qpdf/qpdf-c.h</filename> and the example
        <filename>examples/pdf-linearize.c</filename>.
       </para>
      </listitem>
      <listitem>
       <para>
        Žarko Gajic has written a Delphi wrapper for qpdf, which can
        be downloaded from qpdf's download side.  Žarko's Delphi
        wrapper is released with the same licensing terms as qpdf
        itself and comes with this disclaimer: &ldquo;Delphi wrapper
        unit <filename>qpdf.pas</filename> created by Žarko Gajic
        (<ulink
        url="http://zarko-gajic.iz.hr/">http://zarko-gajic.iz.hr/</ulink>).
        Use at your own risk and for whatever purpose you want.  No
        support is provided.  Sample code is provided.&rdquo;
       </para>
      </listitem>
      <listitem>
       <para>
        Support has been added for AES encryption and crypt filters.
        Although qpdf does not presently support files that use
        PKI-based encryption, with the addition of AES and crypt
        filters, qpdf is now be able to open most encrypted files
        created with newer versions of Acrobat or other PDF creation
        software.  Note that I have not been able to get very many
        files encrypted in this way, so it's possible there could
        still be some cases that qpdf can't handle.  Please report
        them if you find them.
       </para>
      </listitem>
      <listitem>
       <para>
        Many error messages have been improved to include more
        information in hopes of making qpdf a more useful tool for PDF
        experts to use in manually recovering damaged PDF files.
       </para>
      </listitem>
      <listitem>
       <para>
        Attempt to avoid compressing metadata streams if possible.
        This is consistent with other PDF creation applications.
       </para>
      </listitem>
      <listitem>
       <para>
        Provide new command-line options for AES encrypt, cleartext
        metadata, and setting the minimum and forced PDF versions of
        output files.
       </para>
      </listitem>
      <listitem>
       <para>
        Add additional methods to the <classname>QPDF</classname>
        object for querying the document's permissions.  Although qpdf
        does not enforce these permissions, it does make them
        available so that applications that use qpdf can enforce
        permissions.
       </para>
      </listitem>
      <listitem>
       <para>
        The <option>--check</option> option to <command>qpdf</command>
        has been extended to include some additional information.
       </para>
      </listitem>
      <listitem>
       <para>
        There have been a handful of non-compatible API changes.  For
        details, see <xref linkend="ref.upgrading-to-2.1"/>.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.6: May 3, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Do not attempt to uncompress streams that have decode
        parameters we don't recognize.  Earlier versions of qpdf would
        have rejected files with such streams.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.5: March 10, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Improve error handling in the LZW decoder, and fix a small
        error introduced in the previous version with regard to
        handling full tables.  The LZW decoder has been more strongly
        verified in this release.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.4: February 21, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Include proper support for LZW streams encoded without the
        &ldquo;early code change&rdquo; flag.  Special thanks to Atom
        Smasher who reported the problem and provided an input file
        compressed in this way, which I did not previously have.
       </para>
      </listitem>
      <listitem>
       <para>
        Implement some improvements to file recovery logic.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.3: February 15, 2009</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Compile cleanly with gcc 4.4.
       </para>
      </listitem>
      <listitem>
       <para>
        Handle strings encoded as UTF-16BE properly.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.2: June 30, 2008</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        Update test suite to work properly with a
        non-<command>bash</command> <filename>/bin/sh</filename> and
        with Perl 5.10.  No changes were made to the actual qpdf
        source code itself for this release.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0.1: May 6, 2008</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        No changes in functionality or interface.  This release
        includes fixes to the source code so that qpdf compiles
        properly and passes its test suite on a broader range of
        platforms.  See <filename>ChangeLog</filename> in the source
        distribution for details.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
   <varlistentry>
    <term>2.0: April 29, 2008</term>
    <listitem>
     <itemizedlist>
      <listitem>
       <para>
        First public release.
       </para>
      </listitem>
     </itemizedlist>
    </listitem>
   </varlistentry>
  </variablelist>
 </appendix>
 <appendix id="ref.upgrading-to-2.1">
  <title>Upgrading from 2.0 to 2.1</title>
  <para>
   Although, as a general rule, we like to avoid introducing
   source-level incompatibilities in qpdf's interface, there were a
   few non-compatible changes made in this version.  A considerable
   amount of source code that uses qpdf will probably compile without
   any changes, but in some cases, you may have to update your code.
   The changes are enumerated here.  There are also some new
   interfaces; for those, please refer to the header files.
  </para>
  <itemizedlist>
   <listitem>
    <para>
     QPDF's exception handling mechanism now uses
     <classname>std::logic_error</classname> for internal errors and
     <classname>std::runtime_error</classname> for runtime errors in
     favor of the now removed <classname>QEXC</classname> classes used
     in previous versions.  The <classname>QEXC</classname> exception
     classes predated the addition of the
     <filename>&lt;stdexcept&gt;</filename> header file to the C++
     standard library.  Most of the exceptions thrown by the qpdf
     library itself are still of type <classname>QPDFExc</classname>
     which is now derived from
     <classname>std::runtime_error</classname>.  Programs that caught
     an instance of <classname>std::exception</classname> and
     displayed it by calling the <function>what()</function> method
     will not need to be changed.
    </para>
   </listitem>
   <listitem>
    <para>
     The <classname>QPDFExc</classname> class now internally
     represents various fields of the error condition and provides
     interfaces for querying them.  Among the fields is a numeric
     error code that can help applications act differently on (a small
     number of) different error conditions.  See
     <filename>QPDFExc.hh</filename> for details.
    </para>
   </listitem>
   <listitem>
    <para>
     Warnings can be retrieved from qpdf as instances of
     <classname>QPDFExc</classname> instead of strings.
    </para>
   </listitem>
   <listitem>
    <para>
     The nested <classname>QPDF::EncryptionData</classname> class's
     constructor takes an additional argument.  This class is
     primarily intended to be used by
     <classname>QPDFWriter</classname>.  There's not really anything
     useful an end-user application could do with it.  It probably
     shouldn't really be part of the public interface to begin with.
     Likewise, some of the methods for computing internal encryption
     dictionary parameters have changed to support
     <literal>/R=4</literal> encryption.
    </para>
   </listitem>
   <listitem>
    <para>
     The method <function>QPDF::getUserPassword</function> has been
     removed since it didn't do what people would think it did.  There
     are now two new methods:
     <function>QPDF::getPaddedUserPassword</function> and
     <function>QPDF::getTrimmedUserPassword</function>.  The first one
     does what the old <function>QPDF::getUserPassword</function>
     method used to do, which is to return the password with possible
     binary padding as specified by the PDF specification.  The second
     one returns a human-readable password string.
    </para>
   </listitem>
   <listitem>
    <para>
     The enumerated types that used to be nested in
     <classname>QPDFWriter</classname> have moved to top-level
     enumerated types and are now defined in the file
     <filename>qpdf/Constants.h</filename>.  This enables them to be
     shared by both the C and C++ interfaces.
    </para>
   </listitem>
  </itemizedlist>
 </appendix>
 <appendix id="ref.upgrading-to-3.0">
  <title>Upgrading to 3.0</title>
  <para>
   For the most part, the API for qpdf version 3.0 is backward
   compatible with versions 2.1 and later.  There are two exceptions:
   <itemizedlist>
    <listitem>
     <para>
      The method
      <function>QPDFObjectHandle::replaceStreamData</function> that
      uses a <classname>StreamDataProvider</classname> to provide the
      stream data no longer takes a <varname>length</varname>
      parameter.  While it would have been easy enough to keep the
      parameter for backward compatibility, in this case, the
      parameter was removed since this provides the user an
      opportunity to simplify the calling code.  This method was
      introduced in version 2.2.  At the time, the
      <varname>length</varname> parameter was required in order to
      ensure that calls to the stream data provider returned the same
      length for a specific stream every time they were invoked.  In
      particular, the linearization code depends on this.  Instead,
      qpdf 3.0 and newer check for that constraint explicitly.  The
      first time the stream data provider is called for a specific
      stream, the actual length is saved, and subsequent calls are
      required to return the same number of bytes.  This means the
      calling code no longer has to compute the length in advance,
      which can be a significant simplification.  If your code fails
      to compile because of the extra argument and you don't want to
      make other changes to your code, just omit the argument.
     </para>
    </listitem>
    <listitem>
     <para>
      Many methods take <type>long long</type> instead of other
      integer types.  Most if not all existing code should compile
      fine with this change since such parameters had always
      previously been smaller types.  This change was required to
      support files larger than two gigabytes in size.
     </para>
    </listitem>
   </itemizedlist>
  </para>
 </appendix>
 <appendix id="ref.upgrading-to-4.0">
  <title>Upgrading to 4.0</title>
  <para>
   While version 4.0 includes a few non-compatible API changes, it is
   very unlikely that anyone's code would have used any of those parts
   of the API since they generally required information that would
   only be available inside the library.  In the unlikely event that
   you should run into trouble, please see the ChangeLog.  See also
   <xref linkend="ref.release-notes"/> for a complete list of the
   non-compatible API changes made in this version.
  </para>
 </appendix>
</book>