home *** CD-ROM | disk | FTP | other *** search
/ PC World 2002 September / PCWorld_2002-09_cd.bin / Software / Vyzkuste / pdfedit / pdf995.exe / pdf995 / res / convert / wrfont.ps < prev    next >
Encoding:
Text File  |  2002-02-23  |  18.3 KB  |  663 lines

  1. %    Copyright (C) 1991, 1995, 1996 Aladdin Enterprises.  All rights reserved.
  2. % This software is provided AS-IS with no warranty, either express or
  3. % implied.
  4. % This software is distributed under license and may not be copied,
  5. % modified or distributed except as expressly authorized under the terms
  6. % of the license contained in the file LICENSE in this distribution.
  7. % For more information about licensing, please refer to
  8. % http://www.ghostscript.com/licensing/. For information on
  9. % commercial licensing, go to http://www.artifex.com/licensing/ or
  10. % contact Artifex Software, Inc., 101 Lucas Valley Road #110,
  11. % San Rafael, CA  94903, U.S.A., +1(415)492-9861.
  12.  
  13. % $Id: wrfont.ps,v 1.2.6.1 2002/02/22 19:45:55 ray Exp $
  14. % wrfont.ps
  15. % Write out a Type 1 font in readable, reloadable form.
  16. % Note that this does NOT work on protected fonts, such as Adobe fonts
  17. % (unless you have loaded unprot.ps first, in which case you may be
  18. % violating the Adobe license).
  19.  
  20. % ****** NOTE: This file must be kept consistent with gs_pfile.ps.
  21.  
  22. /wrfont_dict 100 dict def
  23. wrfont_dict begin
  24.  
  25. % ------ Options ------ %
  26.  
  27. % Define whether to use eexec encryption for the font.
  28. % eexec encryption is only useful for compatibility with Adobe Type Manager
  29. % and other programs; it only slows Ghostscript down.
  30.    /eexec_encrypt false def
  31.  
  32. % Define whether to write out the CharStrings in binary or in hex.
  33. % Binary takes less space on the file, but isn't guaranteed portable.
  34.    /binary_CharStrings false def
  35.  
  36. % Define whether to use binary token encodings when possible.
  37. % Binary tokens are smaller and load faster, but are a Level 2 feature.
  38.    /binary_tokens false def
  39.  
  40. % Define whether to encrypt the CharStrings on the file.  (CharStrings
  41. % are always encrypted in memory.)  Unencrypted CharStrings load about
  42. % 20% slower, but make the files compress much better for transport.
  43.    /encrypt_CharStrings true def
  44.  
  45. % Define whether the font must provide standard PostScript language
  46. % equivalents for any facilities it uses that are provided in Ghostscript
  47. % but are not part of the standard PostScript language.
  48.    /standard_only true def
  49.  
  50. % Define the value of lenIV to use in writing out the font.
  51. % use_lenIV = 0 produces the smallest output, but this may not be
  52. % compatible with old Adobe interpreters.  use_lenIV = -1 means
  53. % use the value of lenIV from the font.
  54.    /use_lenIV -1 def
  55.  
  56. % Define whether to produce the smallest possible output, relying
  57. % as much as possible on Ghostscript-specific support code.
  58. % Taking full advantage of this requires the following settings:
  59. % binary_CharStrings = true, binary_tokens = true, standard_only = false.
  60.    /smallest_output false def
  61.  
  62. % Define whether to write out all currently known Encodings by name,
  63. % or only StandardEncoding and ISOLatin1Encoding.
  64.    /name_all_Encodings false def
  65.  
  66. % ---------------- Runtime support ---------------- %
  67.  
  68. /.packedfilefilter where
  69.  { pop }
  70.  { (gs_pfile.ps) runlibfile }
  71. ifelse
  72.  
  73. % ------ Output utilities ------ %
  74.  
  75. % By convention, the output file is named psfile.
  76.  
  77. % Define some utilities for writing the output file.
  78.    /wtstring 2000 string def
  79.    /wb {psfile exch write} bind def
  80.    /wnb {/wb load repeat} bind def
  81.    /w1 {psfile exch write} bind def
  82.    /ws {psfile exch writestring} bind def
  83.    /wl {ws (\n) ws} bind def
  84.    /wt {wtstring cvs ws ( ) ws} bind def
  85.    /wd        % Write a dictionary.
  86.     { dup length wo {dict dup begin} wol { we } forall
  87.       {end} wol
  88.     } bind def
  89.    /wld        % Write a large dictionary more efficiently.
  90.            % Ignore the readonly attributes.
  91.     { dup length wo {dict dup begin} wol
  92.       0 exch
  93.        { exch wo wo () wl
  94.      1 add dup 200 eq
  95.       { wo ({def} repeat) wl 0 }
  96.      if
  97.        }
  98.       forall
  99.       dup 0 ne
  100.        { wo ({def} repeat) wl }
  101.        { pop }
  102.       ifelse
  103.       (end) ws
  104.     } bind def
  105.    /we        % Write a dictionary entry.
  106.     { exch wo wo /def cvx wo (\n) ws
  107.     } bind def
  108.    /wcs        % Write a CharString (or Subrs entry)
  109.     { dup type /stringtype eq
  110.        { 4330 exch changelenIV 0 ge
  111.           {    % Add some leading garbage bytes.
  112.         wtstring changelenIV 2 index length getinterval
  113.         .type1decrypt exch pop
  114.         wtstring exch 0 exch length changelenIV add getinterval
  115.       }
  116.       {    % Drop some leading garbage bytes.
  117.         wtstring .type1decrypt exch pop
  118.         changelenIV neg 1 index length 1 index sub getinterval
  119.       }
  120.      ifelse
  121.          binary_tokens encrypt_CharStrings and
  122.       { % Suppress recognizing the readonly status of the string.
  123.         4330 exch dup .type1encrypt exch pop wo
  124.       }
  125.       { encrypt_CharStrings
  126.          { 4330 exch dup .type1encrypt exch pop
  127.          } if
  128.         smallest_output
  129.          { wo
  130.          }
  131.          { readonly dup length wo
  132.            binary_tokens not { ( ) ws } if
  133.            readproc ws wx
  134.          }
  135.         ifelse
  136.       }
  137.      ifelse
  138.        }
  139.        { wo        % PostScript procedure
  140.        }
  141.       ifelse
  142.     } bind def
  143.  
  144. % Construct the inversion of the system name table.
  145.    /SystemNames where
  146.     { pop /snit 256 dict def
  147.       0 1 255
  148.        { dup SystemNames exch get
  149.          dup null ne { exch snit 3 1 roll put } { pop pop } ifelse
  150.        }
  151.       for
  152.     }
  153.     { /snit 1 dict def
  154.     }
  155.    ifelse
  156.  
  157. % Write an object, using binary tokens if requested and possible.
  158.    /woa        % write in ascii
  159.     { psfile exch write==only
  160.     } bind def
  161.  
  162.             % Lookup table for ASCII output.
  163.  
  164.    /intbytes    % int nbytes -> byte*
  165.     { { dup 255 and exch -8 bitshift } repeat pop
  166.     } bind def
  167.    /wotta 10 dict dup begin
  168.       { /booleantype /integertype }
  169.       { { ( ) ws woa } def }
  170.      forall
  171.         % Iterate over arrays so we can print operators.
  172.      /arraytype
  173.       { dup xcheck {(}) ({)} {(]) ([)} ifelse ws exch dup wol exch ws wop
  174.       } bind def
  175.      /dicttype
  176.       { ( ) ws wd } def
  177.      /nametype
  178.       { dup xcheck { ( ) ws } if woa
  179.       } bind def
  180.         % Map back operators to their names,
  181.         % so we can write procedures.
  182.      /nulltype
  183.       { pop ( null) ws
  184.       } bind def
  185.      /operatortype
  186.       { wtstring cvs cvn cvx wo
  187.       } bind def
  188.         % Convert reals to integers if possible.
  189.      /realtype
  190.       { dup cvi 1 index eq { cvi wo } { ( ) ws woa } ifelse
  191.       } bind def
  192.         % == truncates strings longer than 200 characters!
  193.      /stringtype
  194.       { (\() ws dup
  195.      { dup dup 32 lt exch 127 ge or
  196.         { (\\) ws dup -6 bitshift 48 add w1
  197.           dup -3 bitshift 7 and 48 add w1
  198.           7 and 48 add
  199.         }
  200.         { dup dup -2 and 40 eq exch 92 eq or {(\\) ws} if
  201.         }
  202.        ifelse w1
  203.      }
  204.     forall
  205.     (\)) ws wop
  206.       } bind def
  207.      /packedarraytype
  208.       { ([) ws dup { wo } forall
  209.     encodingnames 1 index known
  210.         % This is an encoding, but not one of the standard ones.
  211.         % Use the built-in encoding only if it is available.
  212.      { encodingnames exch get wo
  213.        ({findencoding}stopped{pop) ws
  214.        (}{counttomark 1 add 1 roll cleartomark}ifelse)
  215.      }
  216.      { pop ()
  217.      }
  218.     ifelse
  219.     (/packedarray where{pop counttomark packedarray exch pop}{]readonly}ifelse) ws
  220.     wl
  221.       }
  222.      def
  223.    end def
  224.  
  225.             % Lookup table for binary output.
  226.  
  227.    /wottb 8 dict dup begin
  228.    wotta currentdict copy pop
  229.      /integertype
  230.       { dup dup 127 le exch -128 ge and
  231.          { 136 wb 255 and wb }
  232.      { dup dup 32767 le exch -32768 ge and
  233.         { 134 wb 2 intbytes wb wb }
  234.         { 132 wb 4 intbytes wb wb wb wb }
  235.        ifelse
  236.      }
  237.     ifelse
  238.       } bind def
  239.      /nametype
  240.       { dup snit exch known
  241.          { dup xcheck { 146 } { 145 } ifelse wb
  242.        snit exch get wb
  243.      }
  244.      { wotta /nametype get exec
  245.      }
  246.     ifelse
  247.       } bind def
  248.      /stringtype
  249.       { dup dup length dup 255 le { 142 2 } { 2 intbytes 143 3 } ifelse wnb
  250.     ws wop
  251.       } bind def
  252.    end def
  253.  
  254.    /wop        % Write object protection
  255.      { wcheck not { /readonly cvx wo } if
  256.      } bind def
  257.    /wo        % Write an object.
  258.      { dup type binary_tokens { wottb } { wotta } ifelse
  259.        exch get exec
  260.      } bind def
  261.    /wol        % Write a list of objects.
  262.      { { wo } forall
  263.      } bind def
  264.  
  265. % Write a hex string for Subrs or CharStrings.
  266.    /wx        % string ->
  267.     { binary_CharStrings
  268.        { ws
  269.        }
  270.        { % Some systems choke on very long lines, so
  271.      % we break up the hexstring into chunks of 50 characters.
  272.       { dup length 25 le {exit} if
  273.         dup 0 25 getinterval psfile exch writehexstring (\n) ws
  274.         dup length 25 sub 25 exch getinterval
  275.       } loop
  276.      psfile exch writehexstring
  277.        } ifelse
  278.     } bind def
  279.  
  280. % ------ CharString encryption utilities ------ %
  281.  
  282. /enc_dict 20 dict def
  283. 1 dict begin
  284. /bind { } def        % make sure we can print out the procedures
  285. enc_dict begin
  286.  
  287. (type1enc.ps) runlibfile
  288. enc_dict /.type1decrypt undef        % we don't need this
  289.  
  290. end end
  291.  
  292. enc_dict { 1 index where { pop pop pop } { def } ifelse } forall
  293.  
  294. % ------ Other utilities ------ %
  295.  
  296. % Test whether two values are equal (for default dictionary entries).
  297.    /valueeq        % <obj1> <obj2> valueeq <bool>
  298.     { 2 copy eq
  299.        { pop pop true }
  300.        {    % Special hack for comparing FontMatrix values
  301.      dup type /arraytype eq 2 index type /arraytype eq and
  302.       { dup length 2 index length eq
  303.          { true 0 1 3 index length 1 sub
  304.         {    % Stack: arr1 arr2 true index
  305.           3 index 1 index get 3 index 3 -1 roll get eq not
  306.            { pop false exit }
  307.           if
  308.         }
  309.            for 3 1 roll pop pop
  310.          }
  311.          { pop pop false
  312.          }
  313.         ifelse
  314.       }
  315.       { pop pop false
  316.       }
  317.      ifelse
  318.        }
  319.       ifelse
  320.     } bind def
  321.  
  322. % ------ The main program ------ %
  323.  
  324. % Define the dictionary of keys to skip because they are treated specially.
  325. /.fontskipkeys mark
  326.   /CharStrings dup
  327.   /Encoding dup
  328.   /FDepVector dup
  329.   /FID dup
  330.   /FontInfo dup
  331.   /Metrics dup
  332.   /Metrics2 dup
  333.   /Private dup
  334. .dicttomark def
  335. /.minfontskipkeys mark
  336.   .fontskipkeys { } forall
  337.   /FontName dup
  338.   /UniqueID dup
  339. .dicttomark def
  340. /.privateskipkeys mark
  341.   /ND dup
  342.   /NP dup
  343.   /RD dup
  344.   /Subrs dup
  345. .dicttomark def
  346. /.minprivateskipkeys mark
  347.   .privateskipkeys { } forall
  348.   /MinFeature dup
  349.   /Password dup
  350.   /UniqueID dup
  351. .dicttomark def
  352.  
  353. % Define the procedures for the Private dictionary.
  354. % These must be defined without `bind',
  355. % for the sake of the DISKFONTS feature.
  356. 4 dict begin
  357.  /-! {string currentfile exch readhexstring pop} def
  358.  /-| {string currentfile exch readstring pop} def
  359.  /|- {readonly def} def
  360.  /| {readonly put} def
  361. currentdict end /encrypted_procs exch def
  362. 4 dict begin
  363.  /-! {string currentfile exch readhexstring pop
  364.    4330 exch dup .type1encrypt exch pop} def
  365.  /-| {string currentfile exch readstring pop
  366.    4330 exch dup .type1encrypt exch pop} def
  367.  /|- {readonly def} def
  368.  /| {readonly put} def
  369. currentdict end /unencrypted_procs exch def
  370.  
  371. % Construct an inverse dictionary of encodings.
  372. /encodingnames mark
  373.  StandardEncoding /StandardEncoding
  374.  ISOLatin1Encoding /ISOLatin1Encoding
  375.  SymbolEncoding /SymbolEncoding
  376.  DingbatsEncoding /DingbatsEncoding
  377.  /resourceforall where
  378.   { pop (*) { cvn dup findencoding exch } 100 string /Encoding resourceforall }
  379.  if
  380. .dicttomark def
  381.  
  382. % Invert the standard encodings.
  383. .knownEncodings length 256 mul dict begin
  384.   0 .knownEncodings
  385.    {  { currentdict 1 index known { pop } { 1 index def } ifelse
  386.     1 add
  387.       }
  388.      forall
  389.    }
  390.   forall pop
  391. currentdict end /inverseencodings exch def
  392.  
  393. /writefont        % <psfile> writefont - (writes the current font)
  394.  { /psfile exch def
  395.    /Font currentfont def
  396.    /FontInfo Font /FontInfo .knownget not { 0 dict } if def
  397.    /FontType Font /FontType get def
  398.    /hasPrivate Font /Private known def
  399.    /Private hasPrivate { Font /Private get } { 0 dict } ifelse def
  400.    /readproc binary_CharStrings { (-| ) } { (-! ) } ifelse def
  401.    /privateprocs
  402.      encrypt_CharStrings binary_tokens not and
  403.       { encrypted_procs } { unencrypted_procs } ifelse
  404.      def
  405.    /addlenIV false def
  406.    /changelenIV use_lenIV 0 lt
  407.     { 0 }
  408.     { use_lenIV Private /lenIV .knownget not
  409.        { 4 /addlenIV use_lenIV 4 ne def } if sub }
  410.    ifelse def
  411.    /minimize
  412.      smallest_output
  413.      FontType 1 eq and
  414.      Font /UniqueID known and
  415.    def
  416.    (%!FontType) ws FontType wtstring cvs ws (-1.0: ) ws
  417.      currentfont /FontName get wt
  418.      FontInfo /version .knownget not { (001.001) } if wl
  419.    FontInfo /CreationDate .knownget { (%%Creation Date: ) ws wl } if
  420.    FontInfo /VMusage .knownget
  421.     { (%%VMusage: ) ws dup wt wtstring cvs wl }
  422.    if
  423.    (systemdict begin) wl
  424.  
  425. % If we're going to use eexec, create the filters now.
  426.    /realpsfile psfile def
  427.    eexec_encrypt
  428.     { /eexecfilter psfile binary_CharStrings not
  429.        { pop /bxstring 35 string def
  430.       { pop dup length 0 ne
  431.          { realpsfile exch writehexstring realpsfile (\n) writestring }
  432.          { pop }
  433.         ifelse bxstring
  434.       }
  435.      /NullEncode filter dup /hexfilter exch def
  436.        }
  437.       if 55665 /eexecEncode filter def
  438.     }
  439.    if
  440.  
  441. % Turn on binary tokens if relevant.
  442.    binary_tokens { (currentobjectformat 1 setobjectformat) wl } if
  443.  
  444. % If the file has a UniqueID, write out a check against loading it twice.
  445.    minimize
  446.     { Font /FontName get wo
  447.       Font /UniqueID get wo
  448.       Private length addlenIV { 1 add } if wo
  449.       Font length 1 add wo        % +1 for FontFile
  450.       ( .checkexistingfont) wl
  451.     }
  452.     { Font /UniqueID known
  453.        { ({} FontDirectory) ws Font /FontName get dup wo ( known) wl
  454.      ( {) ws wo ( findfont dup /UniqueID known) wl
  455.      (    { dup /UniqueID get) ws Font /UniqueID get wo ( eq exch /FontType get 1 eq and }) wl
  456.      (    { pop false } ifelse) wl
  457.      (    { pop save /restore load } if) wl
  458.      ( } if) wl
  459.        }
  460.       if
  461.     }
  462.    ifelse
  463.  
  464. % If we are writing unencrypted CharStrings for a standard environment,
  465. % write out the encryption procedures.
  466.    privateprocs unencrypted_procs eq standard_only and
  467.     { (systemdict /.type1encrypt known) wl
  468.       ( { save /restore load } { { } } ifelse) wl
  469.       (userdict begin) wl
  470.       enc_dict { we } forall
  471.       (end exec) wl
  472.     }
  473.    if
  474.  
  475. % Write out the creation of the font dictionary and FontInfo.
  476.    minimize not
  477.     { Font length 1 add wo {dict begin} wol        % +1 for FontFile
  478.     }
  479.    if
  480.    (/FontInfo ) ws FontInfo wd {readonly def} wol
  481.  
  482. % Write out the other fixed entries in the font dictionary.
  483.    Font begin
  484.    Font
  485.     { minimize
  486.        { .minfontskipkeys 2 index known
  487.       { pop pop
  488.       }
  489.       { //.compactfontdefault 2 index .knownget
  490.          { 1 index valueeq { pop pop } { we } ifelse }
  491.          { we }
  492.         ifelse
  493.       }
  494.      ifelse
  495.        }
  496.        { .fontskipkeys 2 index known { pop pop } { we } ifelse
  497.        }
  498.       ifelse
  499.     } forall
  500.    /Encoding
  501.    encodingnames Encoding known
  502.    name_all_Encodings
  503.    Encoding StandardEncoding eq or
  504.    Encoding ISOLatin1Encoding eq or and
  505.     { encodingnames Encoding get cvx }
  506.     { Encoding }
  507.    ifelse
  508.    dup /StandardEncoding cvx eq minimize and
  509.     { pop pop }
  510.     { we }
  511.    ifelse
  512.  
  513. % Write the FDepVector, if any.
  514.    Font /FDepVector .knownget
  515.     { {/FDepVector [} wol
  516.        { /FontName get wo {findfont} wol () wl } forall
  517.       {] readonly def} wol
  518.     }
  519.    if
  520.  
  521. % Write out the Metrics, if any.
  522.    Font /Metrics .knownget
  523.     { (/Metrics ) ws wld {readonly def} wol
  524.     }
  525.    if
  526.    Font /Metrics2 .knownget
  527.     { (/Metrics2 ) ws wld {readonly def} wol
  528.     }
  529.    if
  530.  
  531. % Start the eexec-encrypted section, if applicable.
  532.   eexec_encrypt
  533.    { {currentdict currentfile eexec} wol () wl
  534.      /psfile eexecfilter store
  535.      (\000\000\000\000) ws {begin} wol
  536.    }
  537.   if
  538.  
  539. % Create and initialize the Private dictionary, if any.
  540.    hasPrivate
  541. {
  542.    Private
  543.    minimize
  544.     { begin {Private dup begin}
  545.     }
  546.     {  dup length privateprocs length add dict copy begin
  547.        privateprocs { readonly def } forall
  548.        /Private wo
  549.        currentdict length 1 add wo {dict dup begin}
  550.     }
  551.    ifelse wol () wl
  552.    currentdict
  553.     { 1 index minimize { .minprivateskipkeys } { .privateskipkeys } ifelse
  554.       exch known
  555.        { pop pop }
  556.        { 1 index /lenIV eq use_lenIV 0 ge and { pop use_lenIV } if we }
  557.       ifelse
  558.     } forall
  559.    addlenIV { /lenIV use_lenIV we } if
  560. }
  561. if
  562.  
  563. % Write the Subrs entries, if any.
  564.    currentdict /Subrs known
  565.     { (/Subrs[) wl
  566.       Subrs
  567.        { dup null ne
  568.       { wcs minimize not { () wl } if }
  569.       { pop /null cvx wo }
  570.      ifelse
  571.        } forall
  572.       {] dup {readonly pop} forall readonly def} wol () wl
  573.     }
  574.    if
  575.  
  576. % Wrap up the Private dictionary.
  577.    hasPrivate
  578.     { end            % Private
  579.       minimize
  580.        { {end readonly pop} }    % Private
  581.        { {end readonly def} }    % Private in font
  582.       ifelse wol
  583.     }
  584.    if
  585.  
  586. % Write the CharStrings entries.
  587. % Detect identical (eq) entries, which bdftops produces.
  588.    currentdict /CharStrings known
  589. {
  590.    /CharStrings wo CharStrings length wo
  591.    minimize
  592.     { encrypt_CharStrings not wo ( .readCharStrings) wl
  593.       CharStrings length dict
  594.       CharStrings
  595.        { exch inverseencodings 1 index .knownget not { dup } if wo
  596.         % Stack: vdict value key
  597.      3 copy pop .knownget { wo pop pop } { 3 copy put pop wcs } ifelse
  598.        } forall
  599.     }
  600.     { {dict dup Private begin begin} wol () wl
  601.       CharStrings length dict
  602.       CharStrings
  603.        { 2 index 1 index known
  604.       { exch wo 1 index exch get wo {load def} wol () wl
  605.       }
  606.       { 2 index 1 index 3 index put
  607.         exch wo wcs ( |-) wl
  608.       }
  609.      ifelse
  610.        } forall
  611.       {end end} wol
  612.     }
  613.    ifelse
  614.    pop
  615.     { readonly def }    % CharStrings in font
  616.    wol
  617. }
  618. if
  619.  
  620. % Terminate the output.
  621.    end            % Font
  622.    eexec_encrypt
  623.     { {end mark currentfile closefile} wol () wl
  624.       eexecfilter dup flushfile closefile    % psfile is eexecfilter
  625.       binary_CharStrings not { hexfilter dup flushfile closefile } if
  626.       /psfile realpsfile store
  627.       8
  628.        { (0000000000000000000000000000000000000000000000000000000000000000)
  629.          wl
  630.        }
  631.       repeat {cleartomark} wol
  632.     }
  633.    if
  634.     { FontName currentdict end definefont pop
  635.     }
  636.    wol
  637.    Font /UniqueID known { /exec cvx wo } if
  638.    binary_tokens { /setobjectformat cvx wo } if
  639.    ( end) wl        % systemdict
  640.  
  641.  } bind def
  642.  
  643. % ------ Other utilities ------ %
  644.  
  645. % Prune garbage characters and OtherSubrs out of the current font,
  646. % if the relevant dictionaries are writable.
  647. /prunefont
  648.  { currentfont /CharStrings get wcheck
  649.     { currentfont /CharStrings get dup [ exch
  650.        { pop dup (S????00?) .stringmatch not { pop } if
  651.        } forall
  652.       ] { 2 copy undef pop } forall pop
  653.     }
  654.    if
  655.  } bind def
  656.  
  657. end            % wrfont_dict
  658.  
  659. /writefont { wrfont_dict begin writefont end } def
  660.