home *** CD-ROM | disk | FTP | other *** search
/ The Datafile PD-CD 3 / PDCD_3.iso / tex / texsrc2 / Src / dviutil / dvicopy / web (.txt) < prev   
Texinfo Document  |  1991-04-09  |  163KB  |  3,596 lines

  1. % This is DVICOPY.WEB in text format, as of February 13, 1991.
  2. % Copyright (C) 1990,91 Peter Breitenlohner (peb@@dm0mpi11.bitnet)
  3. % This program is free software; you can redistribute it and/or modify
  4. % it under the terms of the GNU General Public License as published by
  5. % the Free Software Foundation; either version 1, or (at your option)
  6. % any later version.
  7. % You should have received a copy of the GNU General Public License
  8. % along with this program; if not, write to the Free Software
  9. % Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  10. % Version 0.9 was finished May 21, 1990.
  11. % Version 1.0 pixel rounding for real devices (August 6, 1990).
  12. % Version 1.1 major rearrangements for DVIprint (October 7, 1990).
  13. % Version 1.2 fixed some bugs, introduced page selection (February 13, 1991).
  14. % Here is TeX material that gets inserted after \input webmac
  15. \def\hang{\hangindent 3em\indent\ignorespaces}
  16. \font\ninerm=cmr9
  17. \let\mc=\ninerm % medium caps for names like SAIL
  18. \def\PASCAL{Pascal}
  19. \font\tenlogo=logo10 % font used for the METAFONT logo
  20. \font\ninelogo=logo9 % font used for the METAFONT logo
  21. \let\logo=\tenlogo
  22. \def\MF{{\logo META}\-{\logo FONT}}
  23. \mathchardef\RA="3221 % right arrow
  24. \def\(#1){} % this is used to make section names sort themselves better
  25. \def\9#1{} % this is used for sort keys in the index
  26. \def\title{DVI\lowercase{copy}}
  27. \def\contentspagenumber{1}
  28. \def\topofcontents{\null
  29.   \def\titlepage{F} % include headline on the contents page
  30.   \def\rheader{\mainfont\hfil \contentspagenumber}
  31.   \vfill
  32.   \centerline{\titlefont The {\ttitlefont DVIcopy} processor}
  33.   \vskip 5pt
  34.   \centerline{Copyright (C) 1990,91 Peter Breitenlohner}
  35.   \centerline{Distributed under terms of GNU General Public License}
  36.   \vskip 15pt
  37.   \centerline{(Version 1.2, February 1991)}
  38.   \vfill}
  39. \def\botofcontents{\vfill
  40.   \centerline{\hsize 5in\baselineskip9pt
  41.     \vbox{\ninerm \let\logo=\ninelogo \noindent
  42.     This program was developed at the
  43.     Max-Planck-Institut f\"ur Physik
  44.     (Werner-Heisenberg-Institut), Munich, Germany.
  45.     `\TeX' is a trademark of the American Mathematical Society.
  46.     `\MF' is a trademark of Addison-Wesley
  47.     Publishing Company.}}}
  48. \pageno=\contentspagenumber \advance\pageno by 1
  49. @* Introduction.
  50. The \.{DVIcopy} utility program copies (selected pages of) binary
  51. device-independent (``\.{DVI}'') files that are produced by document
  52. compilers such as \TeX, and replaces all references to characters from
  53. virtual fonts by the typesetting instructions specified for them in
  54. binary virtual-font (``\.{VF}'') files.
  55. This program has two chief purposes: (1)~It can be used as preprocessor
  56. for existing \.{DVI}-related software in cases where this software is
  57. unable to handle virtual fonts or (given suitable \.{VF} files) where
  58. this software cannot handle fonts with more than 128~characters;
  59. and (2)~it serves as an example of a program that reads \.{DVI} and
  60. \.{VF} files correctly, for system programmers who are developing
  61. \.{DVI}-related software.
  62. Goal number (1) is important since quite a few existing programs have
  63. to be adapted to the extened capabilities of Version~3 of \TeX\ which
  64. will require some time. Moreover some existing programs are `as is' and
  65. the source code is, unfortunately, not available.
  66. Goal number (2) needs perhaps a bit more explanation. Programs for
  67. typesetting need to be especially careful about how they do arithmetic; if
  68. rounding errors accumulate, margins won't be straight, vertical rules
  69. won't line up, and so on (see the documentaion of \.{DVItype} for more
  70. details). This program is written as if it were a \.{DVI}-driver for a
  71. hypothetical typesetting device |out_file|, the output file receiving
  72. the copy of the input |dvi_file|. In addition all code related to
  73. |out_file| is concentrated in two chapters at the end of this program
  74. and quite independent of the rest of the code concerned with the
  75. decoding of \.{DVI} and \.{VF} files and with font substitutions. Thus
  76. it should be relatively easy to replace the device dependent code of
  77. this program by the corresponding code required for a real typesetting
  78. device. Having this in mind \.{DVItype}'s pixel rounding algorithms are
  79. included as conditional code not used by \.{DVIcopy}.
  80. The |banner| and |preamble_comment| strings defined here should be
  81. changed whenever \.{DVIcopy} gets modified.
  82. @d banner=='This is DVIcopy, Version 1.2' {printed when the program starts}
  83. @d title=='DVIcopy' {the name of this program, used in some messages}
  84. @d copyright=='Copyright (C) 1990,91 Peter Breitenlohner'
  85. @d preamble_comment=='DVIcopy 1.2 output from '
  86. @d comm_length=24 {length of |preamble_comment|}
  87. @d from_length=6 {length of its |' from '| part}
  88. @ This program is written in standard \PASCAL, except where it is necessary
  89. to use extensions; for example, \.{DVIcopy} must read files whose names
  90. are dynamically specified, and that would be impossible in pure \PASCAL.
  91. All places where nonstandard constructions are used have been listed in
  92. the index under ``system dependencies.''
  93. @!@^system dependencies@>
  94. One of the extensions to standard \PASCAL\ that we shall deal with is the
  95. ability to move to a random place in a binary file; another is to
  96. determine the length of a binary file. Such extensions are not necessary
  97. for reading \.{DVI} files; since \.{DVIcopy} is (a model for) a
  98. production program it should, however, be made as efficient as possible
  99. for a particular system. If \.{DVIcopy} is being used with
  100. \PASCAL s for which random file positioning is not efficiently available,
  101. the following definition should be changed from |true| to |false|; in such
  102. cases, \.{DVIcopy} will not include the optional feature that reads the
  103. postamble first.
  104. @d random_reading==true {should we skip around in the file?}
  105. @ The program begins with a fairly normal header, made up of pieces that
  106. @^system dependencies@>
  107. will mostly be filled in later. The \.{DVI} input comes from file
  108. |dvi_file|, the \.{DVI} output goes to file |out_file|, and messages
  109. go to \PASCAL's standard |output| file.
  110. The \.{TFM} and \.{VF} files are defined later since their external
  111. names are determined dynamically.
  112. If it is necessary to abort the job because of a fatal error, the program
  113. calls the `|jump_out|' procedure, which goes to the label |final_end|.
  114. @d final_end = 9999 {go here to wrap it up}
  115. @p @t\4@>@<Compiler directives@>@/
  116. program DVI_copy(@!dvi_file,@!out_file,@!output);
  117. label final_end;
  118. const @<Constants in the outer block@>@/
  119. type @<Types in the outer block@>@/
  120. var @<Globals in the outer block@>@/
  121. @<Error handling procedures@>@/
  122. procedure initialize; {this procedure gets things started properly}
  123.   var @<Local variables for initialization@>@/
  124.   begin print_ln(banner);@/
  125.   print_ln(copyright);
  126.   print_ln('Distributed under terms of GNU General Public License');@/
  127.   @<Set initial values@>@/
  128.   end;
  129. @ The definition of |max_font_type| should be adapted to the number of
  130. font types used by the program; the first two values have a fixed meaning:
  131. |new_font_type=0| indicates that a font has been defined but has
  132. not yet been used, and |vf_font_type=1| indicates a virtual font;
  133. font type values |>=out_font_type=2| indicate real fonts and different
  134. font types could be used to distinguish various kinds of font files
  135. (\.{GF} or \.{PK} or \.{PXL}).
  136. @!@^font types@>
  137. @d new_font_type=0 {this font has been defined but has not yet been used}
  138. @d vf_font_type=1 {this font is a virtual font}
  139. @d out_font_type=2 {this font is a real font}
  140. @d max_font_type=2
  141. @ The following parameters can be changed at compile time to extend or
  142. reduce \.{DVIcopy}'s capacity.
  143. @d max_select=10 {maximum number of page selection ranges}
  144. @<Constants...@>=
  145. @!max_fonts=100; {maximum number of distinct fonts}
  146. @!max_chars=10000; {maximum number of different characters among all fonts}
  147. @!max_widths=3000; {maximum number of different characters widths}
  148. @!max_packets=5000; {maximum number of different characters packets;
  149.   must be less than 65536}
  150. @!max_bytes=30000; {maximum number of bytes for characters packets}
  151. @!max_recursion=10; {\.{VF} files shouldn't recurse beyond this level}
  152. @!stack_size=100; {\.{DVI} files shouldn't |push| beyond this depth}
  153. @!terminal_line_length=150; {maximum number of characters input in a single
  154.   line of input from the terminal}
  155. @!name_length=50; {a file name shouldn't be longer than this}
  156. @ As mentioned above, \.{DVIcopy} has two chief purposes: (1)~It produces
  157. a copy of the input \.{DVI} file with all references to characters from
  158. virtual fonts replaced by their expansion as specified in the character
  159. packets of \.{VF} files; and (2)~it serves as an example of a program
  160. that reads \.{DVI} and \.{VF} files correctly, for system programmers
  161. who are developing \.{DVI}-related software.
  162. In fact, a very large section of code (starting with the second half of
  163. this first chapter `Introduction' and ending with the fourteenth chapter
  164. `The main program') is used in identical form in \.{DVIcopy} and in
  165. \.{DVIprint}, a prototype \.{DVI}-driver for certain types of laser
  166. printers. This has been made possible mostly by using several \.{WEB}
  167. coding tricks, such as not to make the resulting \PASCAL\ program
  168. inefficient in any way.
  169. Parts of the program that are needed in \.{DVIprint} but not in
  170. \.{DVIcopy} are delimited by the codewords `$|device|\ldots|ecived|$';
  171. these are mostly the pixel rounding algorithms used to convert the
  172. \.{DVI} units of a \.{DVI} file to the raster units of a real output
  173. device and have been copied more or less verbatim from \.{DVItype}.
  174. @d device==@{ {change this to `$\\{device}\equiv\null$' when output
  175.   for a real device is produced}
  176. @d ecived==@t@>@} {change this to `$\\{ecived}\equiv\null$' when output
  177.   for a real device is produced}
  178. @f device==begin
  179. @f ecived==end
  180. @ On some systems it is necessary to use various integer subrange types
  181. in order to make \.{\title} efficient; this is true in particular for
  182. frequently used variables such as loop indices. Consider an integer
  183. variable |x| with values in the range |0..255|: on most small systems
  184. |x| should be a one or two byte integer whereas on most large systems
  185. |x| should be a four byte integer.
  186. Clearly the author of a program knows best which range of values is
  187. required for each variable; thus \.{\title} never uses \PASCAL's |integer|
  188. type. All integer variables are declared as one of the integer subrange
  189. types defined below as \.{WEB} macros or \PASCAL\ types; these definitions
  190. can be used without system-dependent changes, provided the signed 32~bit
  191. integers are a subset of the standard type |integer|, and the compiler
  192. automatically uses the optimal representation for integer subranges
  193. (both conditions need not be satisfied for a particular system).
  194. @^system dependencies@>
  195. The complementary problem of storing large arrays of integer type
  196. variables as compactly as possible is addressed differently; here
  197. \.{\title} uses a \PASCAL\ |type|~declaration for each kind of array
  198. element.
  199. Note that the primary purpose of these definitions is optimizations, not
  200. range checking. All places where optimization for a particular system is
  201. highly desirable have been listed in the index under ``optimization.''
  202. @!@^optimization@>
  203. @d int_32 == integer {signed 32~bit integers}
  204. @<Types...@>=
  205. @!int_31 = 0..@"7FFFFFFF; {unsigned 31~bit integer}
  206. @!int_24u = 0..@"FFFFFF; {unsigned 24~bit integer}
  207. @!int_24 = -@"800000..@"7FFFFF; {signed 24~bit integer}
  208. @!int_23 = 0..@"7FFFFF; {unsigned 23~bit integer}
  209. @!int_16u = 0..@"FFFF; {unsigned 16~bit integer}
  210. @!int_16 = -@"8000..@"7FFF; {signed 16~bit integer}
  211. @!int_15 = 0..@"7FFF; {unsigned 15~bit integer}
  212. @!int_8u = 0..@"FF; {unsigned 8~bit integer}
  213. @!int_8 = -@"80..@"7F; {signed 8~bit integer}
  214. @!int_7 = 0..@"7F; {unsigned 7~bit integer}
  215. @ Some of this code is optional for use when debugging only;
  216. such material is enclosed between the delimiters |debug| and $|gubed|$.
  217. Other parts, delimited by |stat| and $|tats|$, are optionally included
  218. if statistics about \.{\title}'s memory usage are desired.
  219. @d debug==@{ {change this to `$\\{debug}\equiv\null$' when debugging}
  220. @d gubed==@t@>@} {change this to `$\\{gubed}\equiv\null$' when debugging}
  221. @f debug==begin
  222. @f gubed==end
  223. @d stat==@{ {change this to `$\\{stat}\equiv\null$'
  224.   when gathering usage statistics}
  225. @d tats==@t@>@} {change this to `$\\{tats}\equiv\null$'
  226.   when gathering usage statistics}
  227. @f stat==begin
  228. @f tats==end
  229. @ The \PASCAL\ compiler used to develop this program has ``compiler
  230. directives'' that can appear in comments whose first character is a dollar sign.
  231. In production versions of \.{\title} these directives tell the compiler that
  232. @^system dependencies@>
  233. it is safe to avoid range checks and to leave out the extra code it inserts
  234. for the \PASCAL\ debugger's benefit, although interrupts will occur if
  235. there is arithmetic overflow.
  236. @<Compiler directives@>=
  237. @{@&$C-,A+,D-@} {no range check, catch arithmetic overflow, no debug overhead}
  238. @!debug @{@&$C+,D+@}@+ gubed {but turn everything on when debugging}
  239. @ Labels are given symbolic names by the following definitions. We insert
  240. the label `|exit|:' just before the `\ignorespaces|end|\unskip' of a
  241. procedure in which we have used the `|return|' statement defined below;
  242. the label `|restart|' is occasionally used at the very beginning of a
  243. procedure; and the label `|reswitch|' is occasionally used just prior to
  244. a \&{case} statement in which some cases change the conditions and we wish to
  245. branch to the newly applicable case.
  246. Loops that are set up with the \&{loop} construction defined below are
  247. commonly exited by going to `|done|' or to `|found|' or to `|not_found|',
  248. and they are sometimes repeated by going to `|continue|'.
  249. @d exit=10 {go here to leave a procedure}
  250. @d restart=20 {go here to start a procedure again}
  251. @d reswitch=21 {go here to start a case statement again}
  252. @d continue=22 {go here to resume a loop}
  253. @d done=30 {go here to exit a loop}
  254. @d found=31 {go here when you've found it}
  255. @d not_found=32 {go here when you've found something else}
  256. @ The term |print| is used instead of |write| when this program writes on
  257. |output|, so that all such output could easily be redirected if desired;
  258. the term |d_print| is used for conditional output if we are debugging.
  259. @d print(#)==write(output,#)
  260. @d print_ln(#)==write_ln(output,#)
  261. @d new_line==write_ln(output) {start new line}
  262. @d print_nl(#)==  {print information starting on a new line}
  263.   begin new_line; print(#);
  264.   end
  265. @d d_print(#)==@!debug print(#) @; @+ gubed
  266. @d d_print_ln(#)==@! debug print_ln(#) @; @+ gubed
  267. @ Here are some macros for common programming idioms.
  268. @d incr(#) == #:=#+1 {increase a variable by unity}
  269. @d decr(#) == #:=#-1 {decrease a variable by unity}
  270. @d Incr_Decr(#) == #
  271. @d Incr(#) == #:=#+Incr_Decr {increase a variable}
  272. @d Decr(#) == #:=#-Incr_Decr {decrease a variable}
  273. @d loop == @+ while true do@+ {repeat over and over until a |goto| happens}
  274. @d do_nothing == {empty statement}
  275. @d return == goto exit {terminate a procedure call}
  276. @f return == nil
  277. @f loop == xclause
  278. @ We assume that |case| statements may include a default case that applies
  279. if no matching label is found. Thus, we shall use constructions like
  280. @^system dependencies@>
  281. $$\vbox{\halign{#\hfil\cr
  282. |case x of|\cr
  283. 1: $\langle\,$code for $x=1\,\rangle$;\cr
  284. 3: $\langle\,$code for $x=3\,\rangle$;\cr
  285. |othercases| $\langle\,$code for |x<>1| and |x<>3|$\,\rangle$\cr
  286. |endcases|\cr}}$$
  287. since most \PASCAL\ compilers have plugged this hole in the language by
  288. incorporating some sort of default mechanism. For example, the compiler
  289. used to develop \.{WEB} and \TeX\ allows `|others|:' as a default label,
  290. and other \PASCAL s allow syntaxes like `\ignorespaces|else|\unskip' or
  291. `\&{otherwise}' or `\\{otherwise}:', etc. The definitions of |othercases|
  292. and |endcases| should be changed to agree with local conventions. (Of
  293. course, if no default mechanism is available, the |case| statements of
  294. this program must be extended by listing all remaining cases.
  295. Donald~E. Knuth, the author of the \.{WEB} system program \.{TANGLE},
  296. @^Knuth, Donald Ervin@>
  297. would have taken the trouble to modify \.{TANGLE} so that such extensions
  298. were done automatically, if he had not wanted to encourage \PASCAL\
  299. compiler writers to make this important change in \PASCAL, where it belongs.)
  300. @d othercases == others: {default for cases not listed explicitly}
  301. @d endcases == @+end {follows the default case in an extended |case| statement}
  302. @f othercases == else
  303. @f endcases == end
  304. @* The character set.
  305. Like all programs written with the  \.{WEB} system, \.{\title} can be
  306. used with any character set. But it uses ASCII code internally, because
  307. the programming for portable input-output is easier when a fixed internal
  308. code is used, and because \.{DVI} and \.{VF} files use ASCII code for
  309. file names and certain other strings.
  310. The next few sections of \.{\title} have therefore been copied from the
  311. analogous ones in the \.{WEB} system routines. They have been considerably
  312. simplified, since \.{\title} need not deal with the controversial
  313. ASCII codes less than @'40 or greater than @'176.
  314. If such codes appear in the \.{DVI} file,
  315. they will be printed as question marks.
  316. @<Types...@>=
  317. @!ASCII_code=" ".."~"; {a subrange of the integers}
  318. @ The original \PASCAL\ compiler was designed in the late 60s, when six-bit
  319. character sets were common, so it did not make provision for lower case
  320. letters. Nowadays, of course, we need to deal with both upper and lower case
  321. alphabets in a convenient way, especially in a program like \.{\title}.
  322. So we shall assume that the \PASCAL\ system being used for \.{\title}
  323. has a character set containing at least the standard visible characters
  324. of ASCII code (|"!"| through |"~"|).
  325. Some \PASCAL\ compilers use the original name |char| for the data type
  326. associated with the characters in text files, while other \PASCAL s
  327. consider |char| to be a 64-element subrange of a larger data type that has
  328. some other name.  In order to accommodate this difference, we shall use
  329. the name |text_char| to stand for the data type of the characters in the
  330. output file.  We shall also assume that |text_char| consists of
  331. the elements |chr(first_text_char)| through |chr(last_text_char)|,
  332. inclusive. The following definitions should be adjusted if necessary.
  333. @^system dependencies@>
  334. @d text_char == char {the data type of characters in text files}
  335. @d first_text_char=0 {ordinal number of the smallest element of |text_char|}
  336. @d last_text_char=127 {ordinal number of the largest element of |text_char|}
  337. @<Types...@>=
  338. @!text_file=packed file of text_char;
  339. @ @<Local variables for init...@>=
  340. @!i:int_16; {loop index for initializations}
  341. @ The \.{\title} processor converts between ASCII code and
  342. the user's external character set by means of arrays |xord| and |xchr|
  343. that are analogous to \PASCAL's |ord| and |chr| functions.
  344. @<Globals...@>=
  345. @!xord: array [text_char] of ASCII_code;
  346.   {specifies conversion of input characters}
  347. @!xchr: array [0..255] of text_char;
  348.   {specifies conversion of output characters}
  349. @ Under our assumption that the visible characters of standard ASCII are
  350. all present, the following assignment statements initialize the
  351. |xchr| array properly, without needing any system-dependent changes.
  352. @<Set init...@>=
  353. for i:=0 to @'37 do xchr[i]:='?';
  354. xchr[@'40]:=' ';
  355. xchr[@'41]:='!';
  356. xchr[@'42]:='"';
  357. xchr[@'43]:='#';
  358. xchr[@'44]:='$';
  359. xchr[@'45]:='%';
  360. xchr[@'46]:='&';
  361. xchr[@'47]:='''';@/
  362. xchr[@'50]:='(';
  363. xchr[@'51]:=')';
  364. xchr[@'52]:='*';
  365. xchr[@'53]:='+';
  366. xchr[@'54]:=',';
  367. xchr[@'55]:='-';
  368. xchr[@'56]:='.';
  369. xchr[@'57]:='/';@/
  370. xchr[@'60]:='0';
  371. xchr[@'61]:='1';
  372. xchr[@'62]:='2';
  373. xchr[@'63]:='3';
  374. xchr[@'64]:='4';
  375. xchr[@'65]:='5';
  376. xchr[@'66]:='6';
  377. xchr[@'67]:='7';@/
  378. xchr[@'70]:='8';
  379. xchr[@'71]:='9';
  380. xchr[@'72]:=':';
  381. xchr[@'73]:=';';
  382. xchr[@'74]:='<';
  383. xchr[@'75]:='=';
  384. xchr[@'76]:='>';
  385. xchr[@'77]:='?';@/
  386. xchr[@'100]:='@@';
  387. xchr[@'101]:='A';
  388. xchr[@'102]:='B';
  389. xchr[@'103]:='C';
  390. xchr[@'104]:='D';
  391. xchr[@'105]:='E';
  392. xchr[@'106]:='F';
  393. xchr[@'107]:='G';@/
  394. xchr[@'110]:='H';
  395. xchr[@'111]:='I';
  396. xchr[@'112]:='J';
  397. xchr[@'113]:='K';
  398. xchr[@'114]:='L';
  399. xchr[@'115]:='M';
  400. xchr[@'116]:='N';
  401. xchr[@'117]:='O';@/
  402. xchr[@'120]:='P';
  403. xchr[@'121]:='Q';
  404. xchr[@'122]:='R';
  405. xchr[@'123]:='S';
  406. xchr[@'124]:='T';
  407. xchr[@'125]:='U';
  408. xchr[@'126]:='V';
  409. xchr[@'127]:='W';@/
  410. xchr[@'130]:='X';
  411. xchr[@'131]:='Y';
  412. xchr[@'132]:='Z';
  413. xchr[@'133]:='[';
  414. xchr[@'134]:='\';
  415. xchr[@'135]:=']';
  416. xchr[@'136]:='^';
  417. xchr[@'137]:='_';@/
  418. xchr[@'140]:='`';
  419. xchr[@'141]:='a';
  420. xchr[@'142]:='b';
  421. xchr[@'143]:='c';
  422. xchr[@'144]:='d';
  423. xchr[@'145]:='e';
  424. xchr[@'146]:='f';
  425. xchr[@'147]:='g';@/
  426. xchr[@'150]:='h';
  427. xchr[@'151]:='i';
  428. xchr[@'152]:='j';
  429. xchr[@'153]:='k';
  430. xchr[@'154]:='l';
  431. xchr[@'155]:='m';
  432. xchr[@'156]:='n';
  433. xchr[@'157]:='o';@/
  434. xchr[@'160]:='p';
  435. xchr[@'161]:='q';
  436. xchr[@'162]:='r';
  437. xchr[@'163]:='s';
  438. xchr[@'164]:='t';
  439. xchr[@'165]:='u';
  440. xchr[@'166]:='v';
  441. xchr[@'167]:='w';@/
  442. xchr[@'170]:='x';
  443. xchr[@'171]:='y';
  444. xchr[@'172]:='z';
  445. xchr[@'173]:='{';
  446. xchr[@'174]:='|';
  447. xchr[@'175]:='}';
  448. xchr[@'176]:='~';
  449. for i:=@'177 to 255 do xchr[i]:='?';
  450. @ The following system-independent code makes the |xord| array contain a
  451. suitable inverse to the information in |xchr|.
  452. @<Set init...@>=
  453. for i:=first_text_char to last_text_char do xord[chr(i)]:=@'40;
  454. for i:=" " to "~" do xord[xchr[i]]:=i;
  455. @* Reporting errors to the user.
  456. The \.{\title} processor does not verify that every single bit read from
  457. one of its binary input files is meaningful and consistent; there are
  458. other programs, e.g., \.{DVItype}, \.{TFtoPL}, and \.{VFtoPL}, specially
  459. designed for that purpose.
  460. On the other hand, \.{\title} is designed to avoid unpredictable results
  461. due to undetected arithmetic overflow, or due to violation of integer
  462. subranges or array bounds under {\it all\/} circumstances. Thus a fair
  463. amount of checking is done when reading and analyzing the input data,
  464. even in cases where such checking reduces the efficiency of the program
  465. to some extent.
  466. @ A global variable called |history| will contain one of four values
  467. at the end of every run: |spotless| means that no unusual messages were
  468. printed; |harmless_message| means that a message of possible interest
  469. was printed but no serious errors were detected; |error_message| means that
  470. at least one error was found; |fatal_message| means that the program
  471. terminated abnormally. The value of |history| does not influence the
  472. behavior of the program; it is simply computed for the convenience
  473. of systems that might want to use such information.
  474. @d spotless=0 {|history| value for normal jobs}
  475. @d harmless_message=1 {|history| value when non-serious info was printed}
  476. @d error_message=2 {|history| value when an error was noted}
  477. @d fatal_message=3 {|history| value when we had to stop prematurely}
  478. @d mark_harmless==@t@>@+if history=spotless then history:=harmless_message
  479. @d mark_error==history:=error_message
  480. @d mark_fatal==history:=fatal_message
  481. @<Glob...@>=@!history:spotless..fatal_message; {how bad was this run?}
  482. @ @<Set init...@>=history:=spotless;
  483. @ If an input (\.{DVI}, \.{TFM}, \.{VF}, or other) file is badly malformed,
  484. the whole process must be aborted; \.{\title} will give up, after issuing
  485. an error message about what caused the error. These messages will, however,
  486. in most cases just indicate which input file caused the error. One of the
  487. programs \.{DVItype}, \.{TFtoPL} or \.{VFtoVP} should then be used to
  488. diagnose the error in full detail.
  489. Such errors might be discovered inside of subroutines inside of subroutines,
  490. so a procedure called |jump_out| has been introduced. This procedure, which
  491. transfers control to the label |final_end| at the end of the program,
  492. contains the only non-local |@!goto| statement in \.{\title}.
  493. @^system dependencies@>
  494. Some \PASCAL\ compilers do not implement non-local |goto| statements. In
  495. such cases the |goto final_end| in |jump_out| should simply be replaced
  496. by a call on some system procedure that quietly terminates the program.
  497. @^system dependencies@>
  498. @d abort(#)==begin print_ln(' ',#,'.'); jump_out;
  499.     end
  500. @<Error handling...@>=
  501. @<Basic printing procedures@>@;
  502. procedure close_files_and_terminate; forward;
  503. procedure jump_out;
  504. begin mark_fatal; close_files_and_terminate;
  505. goto final_end;
  506. @ Sometimes the program's behavior is far different from what it should
  507. be, and \.{\title} prints an error message that is really for the
  508. \.{\title} maintenance person, not the user. In such cases the program
  509. says |confusion(|indication of where we are|)|.
  510. @<Error handling...@>=
  511. procedure confusion(@!p:pckt_pointer);
  512. begin print(' !This can''t happen ('); print_packet(p); print_ln(').');
  513. @.This can't happen@>
  514. jump_out;
  515. @ An overflow stop occurs if \.{\title}'s tables aren't large enough.
  516. @<Error handling...@>=
  517. procedure overflow(@!p:pckt_pointer;@!n:int_16u);
  518. begin print(' !Sorry, ',title,' capacity exceeded ['); print_packet(p);
  519. @.Sorry, {\title} capacity exceeded@>
  520. print_ln('=',n:1,'].');
  521. jump_out;
  522. @* Binary data and binary files.
  523. A detailed description of the \.{DVI} file format can be found in the
  524. documentation of \TeX, \.{DVItype}, or \.{GFtoDVI}; here we just define
  525. symbolic names for some of the \.{DVI} command bytes.
  526. @d set_char_0=0 {typeset character 0 and move right}
  527. @d set1=128 {typeset a character and move right}
  528. @d set_rule=132 {typeset a rule and move right}
  529. @d put1=133 {typeset a character}
  530. @d put_rule=137 {typeset a rule}
  531. @d nop=138 {no operation}
  532. @d bop=139 {beginning of page}
  533. @d eop=140 {ending of page}
  534. @d push=141 {save the current positions}
  535. @d pop=142 {restore previous positions}
  536. @d right1=143 {move right}
  537. @d w0=147 {move right by |w|}
  538. @d w1=148 {move right and set |w|}
  539. @d x0=152 {move right by |x|}
  540. @d x1=153 {move right and set |x|}
  541. @d down1=157 {move down}
  542. @d y0=161 {move down by |y|}
  543. @d y1=162 {move down and set |y|}
  544. @d z0=166 {move down by |z|}
  545. @d z1=167 {move down and set |z|}
  546. @d fnt_num_0=171 {set current font to 0}
  547. @d fnt1=235 {set current font}
  548. @d xxx1=239 {extension to \.{DVI} primitives}
  549. @d xxx4=242 {potentially long extension to \.{DVI} primitives}
  550. @d fnt_def1=243 {define the meaning of a font number}
  551. @d pre=247 {preamble}
  552. @d post=248 {postamble beginning}
  553. @d post_post=249 {postamble ending}
  554. @d dvi_id=2 {identifies \.{DVI} files}
  555. @ A \.{DVI}, \.{VF}, or \.{TFM} file is a sequence of 8-bit bytes.
  556. The bytes appear physically in what is called a `|packed file of 0..255|'
  557. in \PASCAL\ lingo. One, two, three, or four consecutive bytes are often
  558. interpreted as (signed or unsigned) integers.
  559. We might as well define the corresponding data types.
  560. @!@^system dependencies@>
  561. @<Types...@>=
  562. @!signed_byte=-@"80..@"7F; {signed one-byte quantity}
  563. @!eight_bits=0..@"FF; {unsigned one-byte quantity}
  564. @!signed_pair=-@"8000..@"7FFF; {signed two-byte quantity}
  565. @!sixteen_bits=0..@"FFFF; {unsigned two-byte quantity}
  566. @!signed_trio=-@"800000..@"7FFFFF; {signed three-byte quantity}
  567. @!twentyfour_bits=0..@"FFFFFF; {unsigned three-byte quantity}
  568. @!signed_quad=int_32; {signed four-byte quantity}
  569. @ Packing is system dependent, and many \PASCAL\ systems fail to implement
  570. such files in a sensible way (at least, from the viewpoint of producing
  571. good production software).  For example, some systems treat all
  572. byte-oriented files as text, looking for end-of-line marks and such
  573. things. Therefore some system-dependent code is often needed to deal with
  574. binary files, even though most of the program in this section of
  575. \.{\title} is written in standard \PASCAL.
  576. @^system dependencies@>
  577. One common way to solve the problem is to consider files of |integer|
  578. numbers, and to convert an integer in the range $-2^{31}\L x<2^{31}$ to
  579. a sequence of four bytes $(a,b,c,d)$ using the following code, which
  580. avoids the controversial integer division of negative numbers:
  581. $$\vbox{\halign{#\hfil\cr
  582. |if x>=0 then a:=x div @'100000000|\cr
  583. |else begin x:=(x+@'10000000000)+@'10000000000; a:=x div @'100000000+128;|\cr
  584. \quad|end|\cr
  585. |x:=x mod @'100000000;|\cr
  586. |b:=x div @'200000; x:=x mod @'200000;|\cr
  587. |c:=x div @'400; d:=x mod @'400;|\cr}}$$
  588. The four bytes are then kept in a buffer and output one by one. (On 36-bit
  589. computers, an additional division by 16 is necessary at the beginning.
  590. Another way to separate an integer into four bytes is to use/abuse
  591. \PASCAL's variant records, storing an integer and retrieving bytes that are
  592. packed in the same place; {\sl caveat implementor!\/}) It is also desirable
  593. in some cases to read a hundred or so integers at a time, maintaining a
  594. larger buffer.
  595. @ We shall stick to simple \PASCAL\ in the standard version of this program,
  596. for reasons of clarity, even if such simplicity is sometimes unrealistic.
  597. @<Types...@>=
  598. @!byte_file=packed file of eight_bits; {files that contain binary data}
  599. @ For some operating systems it may be convenient or even necessary to
  600. close the input files.
  601. @d close_in(#)==do_nothing {close an input file}
  602. @ Character packets extracted from \.{VF} files will be stored in a large
  603. array |byte_mem|. Other packets of bytes, e.g., character packets
  604. extracted from a \.{GF} or \.{PK} or \.{PXL} file could be stored in the
  605. same way. A `|pckt_pointer|' variable, which signifies a packet,
  606. is an index into another array |pckt_start|. The actual sequence of bytes
  607. in the packet pointed to by |p| appears in positions |pckt_start[p]| to
  608. |pckt_start[p+1]-1|, inclusive, in |byte_mem|.
  609. Packets will also be used to store sequences of |ASCII_code|s; in this
  610. respect the |byte_mem| array is very similar to \TeX's string pool and
  611. part of the following code has, in fact, been copied more or less
  612. verbatim from \TeX.
  613. In other respects the packets resemble the identifiers used by
  614. \.{TANGLE} and \.{WEAVE} (also stored in an array called |byte_mem|)
  615. since there is, in general, at most one packet with a given contents;
  616. thus part of the code below has been adapted from the corresponding code
  617. in these programs.
  618. Some \PASCAL\ compilers won't pack integers into a single byte unless the
  619. integers lie in the range |-128..127|. To accommodate such systems we
  620. access the array |byte_mem| only via macros that can easily be redefined.
  621. @^system dependencies@>
  622. @d bi(#) == # {convert from |eight_bits| to |packed_byte|}
  623. @d bo(#) == # {convert from |packed_byte| to |eight_bits|}
  624. @<Types...@>=
  625. @!packed_byte = eight_bits; {elements of |byte_mem| array}
  626. @!byte_pointer = 0..max_bytes; {an index into |byte_mem|}
  627. @!pckt_pointer = 0..max_packets; {an index into |pckt_start|}
  628. @ The global variable |byte_ptr| points to the first unused location in
  629. |byte_mem| and |pckt_ptr| points to the first unused location in
  630. |pckt_start|.
  631. @<Globals...@>=
  632. @!byte_mem: packed array [byte_pointer] of packed_byte; {bytes of packets}
  633. @!pckt_start: array [pckt_pointer] of byte_pointer;
  634.   {directory into |byte_mem|}
  635. @!byte_ptr: byte_pointer;
  636. @!pckt_ptr: pckt_pointer;
  637. @ Several of the elementary operations with packets are performed using
  638. \.{WEB} macros instead of \PASCAL\ procedures, because many of the
  639. operations are done quite frequently and we want to avoid the
  640. overhead of procedure calls. For example, here is
  641. a simple macro that computes the length of a packet.
  642. @.WEB@>
  643. @d pckt_length(#)==(pckt_start[#+1]-pckt_start[#]) {the number of bytes
  644.   in packet number \#}
  645. @ Packets are created by appending bytes to |byte_mem|.
  646. The |append_byte| macro, defined here, does not check to see if the
  647. value of |byte_ptr| has gotten too high; this test is supposed to be
  648. made before |append_byte| is used. There is also a |flush_byte|
  649. macro, which erases the last byte appended.
  650. To test if there is room to append |l| more bytes to |byte_mem|,
  651. we shall write |pckt_room(l)|, which aborts \.{\title} and gives an
  652. apologetic error message if there isn't enough room.
  653. @d append_byte(#) == {put byte \# at the end of |byte_mem|}
  654. begin byte_mem[byte_ptr]:=bi(#); incr(byte_ptr);
  655. @d flush_byte == decr(byte_ptr) {forget the last byte in |byte_mem|}
  656. @d pckt_room(#) == {make sure that |byte_mem| hasn't overflowed}
  657.   if max_bytes-byte_ptr<# then overflow(str_bytes,max_bytes)
  658. @d append_one(#) ==
  659. begin pckt_room(1); append_byte(#);
  660. @ The length of the current packet is called |cur_pckt_length|:
  661. @d cur_pckt_length == (byte_ptr - pckt_start[pckt_ptr])
  662. @ Once a sequence of bytes has been appended to |byte_mem|, it
  663. officially becomes a packet when the |make_packet| function is called.
  664. This function returns as its value the identification number of either
  665. an existing packet with the same contents or, if no such packet exists,
  666. of the new packet. Thus two packets have the same contents if and only
  667. if they have the same identification number. In order to locate the
  668. packet with a given contents, or to find out that no such packet exists,
  669. we need a hash table. The hash table is kept by the method of simple
  670. chaining, where the heads of the individual lists appear in the |p_hash|
  671. array. If |h| is a hash code, the hash table list starts at |p_hash[h]|
  672. and proceeds through |p_link| pointers.
  673. @d hash_size=353 {should be prime, must be |>256|}
  674. @<Types...@>=
  675. @!hash_code=0..hash_size;
  676. @ @<Glob...@>=
  677. @!p_link:array[pckt_pointer] of pckt_pointer; {hash table}
  678. @!p_hash:array[hash_code] of pckt_pointer;
  679. @ Initially |byte_mem| and all the hash lists are empty; |empty_packet|
  680. is the empty packet.
  681. @d empty_packet=0 {the empty packet}
  682. @d invalid_packet==max_packets {used when there is no packet}
  683. @<Set init...@>=
  684. pckt_ptr:=1; byte_ptr:=1;
  685. pckt_start[0]:=1; pckt_start[1]:=1;
  686. for h:=0 to hash_size-1 do p_hash[h]:=0;
  687. @ @<Local variables for init...@>=
  688. @!h:hash_code; {index into hash-head arrays}
  689. @ Here now is the |make_packet| function used to create packets (and
  690. strings).
  691. @p function make_packet:pckt_pointer;
  692. label found;
  693. var i,@!k:byte_pointer; {indices into |byte_mem|}
  694. @!h:hash_code; {hash code}
  695. @!s,@!l:byte_pointer; {start and length of the given packet}
  696. @!p:pckt_pointer; {where the packet is being sought}
  697. begin s:=pckt_start[pckt_ptr]; l:=byte_ptr-s; {compute start and length}
  698. if l=0 then p:=empty_packet
  699. else  begin @<Compute the packet hash code |h|@>;
  700.   @<Compute the packet location |p|@>;
  701.   if pckt_ptr=max_packets then overflow(str_packets,max_packets);
  702.   incr(pckt_ptr); pckt_start[pckt_ptr]:=byte_ptr;
  703.   end;
  704. found:make_packet:=p;
  705. @ A simple hash code is used: If the sequence of bytes is
  706. $b_1b_2\ldots b_n$, its hash value will be
  707. $$(2^{n-1}b_1+2^{n-2}b_2+\cdots+b_n)\,\bmod\,|hash_size|.$$
  708. @<Compute the packet hash...@>=
  709. h:=bo(byte_mem[s]); i:=s+1;
  710. while i<byte_ptr do
  711.   begin h:=(h+h+bo(byte_mem[i])) mod hash_size; incr(i);
  712.   end
  713. @ If the packet is new, it will be placed in position |p=pckt_ptr|,
  714. otherwise |p| will point to its existing location.
  715. @<Compute the packet location...@>=
  716. p:=p_hash[h];
  717. while p<>0 do
  718.   begin if pckt_length(p)=l then
  719.       @<Compare packet |p| with current packet, |goto found| if equal@>;
  720.   p:=p_link[p];
  721.   end;
  722. p:=pckt_ptr; {the current packet is new}
  723. p_link[p]:=p_hash[h]; p_hash[h]:=p {insert |p| at beginning of hash list}
  724. @ @<Compare packet |p|...@>=
  725. begin i:=s; k:=pckt_start[p];
  726. while (i<byte_ptr)and(byte_mem[i]=byte_mem[k]) do
  727.   begin incr(i); incr(k);
  728.   end;
  729. if i=byte_ptr then {all bytes agree}
  730.   begin byte_ptr:=pckt_start[pckt_ptr]; goto found;
  731.   end;
  732. @ Some packets are initialized with predefined strings of |ASCII_code|s;
  733. a few macros permit us to do the initialization with a compact program.
  734. Since this initialization is done when |byte_mem| is still empty, and
  735. since |byte_mem| is supposed to be large enough for all the predefined
  736. strings, |pckt_room| is used only if we are debugging.
  737. @d pid0(#)==#:=make_packet
  738. @d pid1(#)==byte_mem[byte_ptr-1]:=bi(#); pid0
  739. @d pid2(#)==byte_mem[byte_ptr-2]:=bi(#); pid1
  740. @d pid3(#)==byte_mem[byte_ptr-3]:=bi(#); pid2
  741. @d pid4(#)==byte_mem[byte_ptr-4]:=bi(#); pid3
  742. @d pid5(#)==byte_mem[byte_ptr-5]:=bi(#); pid4
  743. @d pid6(#)==byte_mem[byte_ptr-6]:=bi(#); pid5
  744. @d pid7(#)==byte_mem[byte_ptr-7]:=bi(#); pid6
  745. @d pid8(#)==byte_mem[byte_ptr-8]:=bi(#); pid7
  746. @d pid9(#)==byte_mem[byte_ptr-9]:=bi(#); pid8
  747. @d pid10(#)==byte_mem[byte_ptr-10]:=bi(#); pid9
  748. @d pid_init(#)==
  749.   @!debug pckt_room(#); @+ gubed @;
  750.   Incr(byte_ptr)(#)
  751. @d id1==pid_init(1); pid1
  752. @d id2==pid_init(2); pid2
  753. @d id3==pid_init(3); pid3
  754. @d id4==pid_init(4); pid4
  755. @d id5==pid_init(5); pid5
  756. @d id6==pid_init(6); pid6
  757. @d id7==pid_init(7); pid7
  758. @d id8==pid_init(8); pid8
  759. @d id9==pid_init(9); pid9
  760. @d id10==pid_init(10); pid10
  761. @ Here we initialize some strings used as argument of the |overflow| and
  762. |confusion| procedures.
  763. @<Initialize predefined strings@>=
  764. id5("f")("o")("n")("t")("s")(str_fonts);
  765. id5("c")("h")("a")("r")("s")(str_chars);
  766. id6("w")("i")("d")("t")("h")("s")(str_widths);
  767. id7("p")("a")("c")("k")("e")("t")("s")(str_packets);
  768. id5("b")("y")("t")("e")("s")(str_bytes);
  769. id9("r")("e")("c")("u")("r")("s")("i")("o")("n")(str_recursion);
  770. id5("s")("t")("a")("c")("k")(str_stack);
  771. id10("n")("a")("m")("e")("l")("e")("n")("g")("t")("h")(str_name_length);
  772. @ @<Glob...@>=
  773. @!str_fonts,@!str_chars,@!str_widths,@!str_packets,@!str_bytes,
  774. @!str_recursion,@!str_stack,@!str_name_length:pckt_pointer;
  775. @ Some packets, e.g., the preamble comments of \.{DVI} and \.{VF} files,
  776. are needed only temporarily. In such cases |new_packet| is used to
  777. create a packet (which might duplicate an existing packet) and
  778. |flush_packet| is used to discard it; the calls to |new_packet| and
  779. |flush_packet| must occur in balanced pairs, without any intervening
  780. calls to |make_packet|.
  781. @p function new_packet: pckt_pointer;
  782. begin if pckt_ptr=max_packets then overflow(str_packets,max_packets);
  783. new_packet:=pckt_ptr; incr(pckt_ptr); pckt_start[pckt_ptr]:=byte_ptr;
  784. procedure flush_packet;
  785. begin decr(pckt_ptr); byte_ptr:=pckt_start[pckt_ptr];
  786. @ The |print_packet| procedure prints the contents of a packet; such a
  787. packets should, of course, consists of a sequence of |ASCII_code|s.
  788. @<Basic printing...@>=
  789. procedure print_packet(p:pckt_pointer);
  790. var k:byte_pointer;
  791. begin for k:=pckt_start[p] to pckt_start[p+1]-1 do
  792.   print(xchr[bo(byte_mem[k])]);
  793. @ When we interpret a packet we will use two (global or local) variables:
  794. |cur_loc| will point to the byte to be used next, and |cur_limit| will
  795. point to the start of the next packet. The macro |pckt_extract| will be
  796. used to extract one byte; it should, however, never be used with
  797. |cur_loc>=cur_limit|.
  798. @d pckt_extract(#) ==
  799. @!debug if cur_loc>=cur_limit then confusion(str_packets) @+ else @/
  800. gubed @;
  801.   begin #:=bo(byte_mem[cur_loc]); incr(cur_loc); @+ end
  802. @<Globals...@>=
  803. @!cur_pckt: pckt_pointer; {the current packet}
  804. @!cur_loc: byte_pointer; {current location in a packet}
  805. @!cur_limit: byte_pointer; {start of next packet}
  806. @ We will need routines to extract one, two, three, or four bytes from
  807. |byte_mem|, from the \.{DVI} file, or from a \.{VF} file and assemble
  808. them into (signed or unsigned) integers and these routines should be
  809. optimized for efficiency. Here we define \.{WEB} macros to be used for
  810. the body of these routines; thus the changes for system dependent
  811. optimization have to be applied only once.
  812. @^system dependencies@>
  813. @^optimization@>
  814. In addition we demonstrates how these macros can be used to define
  815. functions that extract one, two, three, or four bytes from a character
  816. packet and assemble them into signed or unsigned integers (assuming that
  817. |cur_loc| and |cur_limit| are initialized suitably).
  818. @d begin_byte(#) ==
  819. var a:eight_bits;
  820. begin #(a)
  821. @d comp_sbyte(#) == if a<128 then #:=a @+ else #:=a-256
  822. @d comp_ubyte(#) == #:=a
  823. @f begin_byte == begin
  824. @p function pckt_sbyte:int_8; {returns the next byte, signed}
  825. @!begin_byte(pckt_extract); comp_sbyte(pckt_sbyte);
  826. function pckt_ubyte:int_8u; {returns the next byte, unsigned}
  827. @!begin_byte(pckt_extract); comp_ubyte(pckt_ubyte);
  828. @ @d begin_pair(#) ==
  829. var a,@!b:eight_bits;
  830. begin #(a); #(b)
  831. @d comp_spair(#) == if a<128 then #:=a*256+b @+ else #:=(a-256)*256+b
  832. @d comp_upair(#) == #:=a*256+b
  833. @f begin_pair == begin
  834. @p function pckt_spair:int_16; {returns the next two bytes, signed}
  835. @!begin_pair(pckt_extract); comp_spair(pckt_spair);
  836. function pckt_upair:int_16u; {returns the next two bytes, unsigned}
  837. @!begin_pair(pckt_extract); comp_upair(pckt_upair);
  838. @ @d begin_trio(#) ==
  839. var a,@!b,@!c:eight_bits;
  840. begin #(a); #(b); #(c)
  841. @d comp_strio(#) ==
  842. if a<128 then #:=(a*256+b)*256+c @+ else #:=((a-256)*256+b)*256+c
  843. @d comp_utrio(#) == #:=(a*256+b)*256+c
  844. @f begin_trio == begin
  845. @p function pckt_strio:int_24; {returns the next three bytes, signed}
  846. @!begin_trio(pckt_extract); comp_strio(pckt_strio);
  847. function pckt_utrio:int_24u; {returns the next three bytes, unsigned}
  848. @!begin_trio(pckt_extract); comp_utrio(pckt_utrio);
  849. @ @d begin_quad(#) ==
  850. var a,@!b,@!c,@!d:eight_bits;
  851. begin #(a); #(b); #(c); #(d)
  852. @d comp_squad(#) ==
  853. if a<128 then #:=((a*256+b)*256+c)*256+d
  854. else #:=(((a-256)*256+b)*256+c)*256+d
  855. @f begin_quad == begin
  856. @p function pckt_squad:int_32; {returns the next four bytes, signed}
  857. @!begin_quad(pckt_extract); comp_squad(pckt_squad);
  858. @ A similar set of routines is needed for the inverse task of
  859. decomposing a \.{DVI} command into a sequence of bytes to be appended
  860. to |byte_mem| or, in the case of \.{DVIcopy}, to be written to the
  861. output file. Again we define \.{WEB} macros to be used for the body
  862. of these routines; thus the changes for system dependent optimization
  863. have to be applied only once.
  864. @^system dependencies@>
  865. @^optimization@>
  866. First, the |pckt_one| outputs one byte, negative values are represented
  867. in two's complement notation.
  868. @d begin_one == begin
  869. @d comp_one(#) ==
  870. if x<0 then Incr(x)(256);
  871. @f begin_one == begin
  872. @p @!device
  873. procedure pckt_one(@!x:int_32); {output one byte}
  874. @!begin_one; pckt_room(1); comp_one(append_byte);
  875. ecived
  876. @ The |pckt_two| outputs two bytes, negative values are represented in
  877. two's complement notation.
  878. @d begin_two == begin
  879. @d comp_two(#) ==
  880. if x<0 then Incr(x)(@"10000);
  881. #(x div @"100); #(x mod @"100)
  882. @f begin_two == begin
  883. @p @!device
  884. procedure pckt_two(@!x:int_32); {output two byte}
  885. @!begin_two; pckt_room(2); comp_two(append_byte);
  886. ecived
  887. @ The |pckt_four| procedure outputs four bytes in two's complement
  888. notation, without risking arithmetic overflow.
  889. @d begin_four == begin
  890. @d comp_four(#) ==
  891. if x>=0 then #(x div @"1000000)
  892. else  begin Incr(x)(@"40000000); Incr(x)(@"40000000);
  893.   #((x div @"1000000) + 128);
  894.   end;
  895. x:=x mod @"1000000; #(x div @"10000);
  896. x:=x mod @"10000; #(x div @"100);
  897. #(x mod @"100)
  898. @f begin_four == begin
  899. @p procedure pckt_four(@!x:int_32); {output four bytes}
  900. @!begin_four; pckt_room(4); comp_four(append_byte);
  901. @ Next, the |pckt_char| procedure outputs a |set_char| or \\{set} command
  902. or, if |upd=false|, a |put| command.
  903. @d begin_char ==
  904. var o:eight_bits; {|set1| or |put1|}
  905. begin
  906. @d comp_char(#) ==
  907. if (not upd)or(res>127)or(ext<>0) then
  908.   begin o:=dvi_char_cmd[upd]; {|set1| or |put1|}
  909.   if ext<0 then Incr(ext)(@"1000000);
  910.   if ext=0 then #(o) @+ else @;
  911.     begin if ext<@"100 then #(o+1) @+ else @;
  912.       begin if ext<@"10000 then #(o+2) @+ else @;
  913.         begin #(o+3); #(ext div @"10000); ext:=ext mod @"10000;
  914.         end;
  915.       #(ext div @"100); ext:=ext mod @"100;
  916.       end;
  917.     #(ext);
  918.     end;
  919.   end;
  920. #(res)
  921. @f begin_char == begin
  922. @p procedure pckt_char(@!upd:boolean;@!ext:int_32;@!res:eight_bits);
  923.   {output \\{set} or |put|}
  924. @!begin_char; pckt_room(5); comp_char(append_byte);
  925. @ Then, the |pckt_unsigned| procedure outputs a |fnt| or |xxx|
  926. command with its first parameter (normally unsigned); a |fnt| command
  927. is converted into |fnt_num| whenever this is possible.
  928. @d begin_unsigned == begin
  929. @d comp_unsigned(#) ==
  930. if (x<@"100)and(x>=0) then
  931.   if (o=fnt1)and(x<64) then Incr(x)(fnt_num_0) @+ else #(o)
  932.   begin if (x<@"10000)and(x>=0) then #(o+1) @+ else @;
  933.     begin if (x<@"1000000)and(x>=0) then #(o+2) @+ else @;
  934.       begin #(o+3);
  935.       if x>=0 then #(x div @"1000000)
  936.       else  begin Incr(x)(@"40000000); Incr(x)(@"40000000);
  937.         #((x div @"1000000) + 128);
  938.         end;
  939.       x:=x mod @"1000000;
  940.       end;
  941.     #(x div @"10000); x:=x mod @"10000;
  942.     end;
  943.   #(x div @"100); x:=x mod @"100;
  944.   end;
  945. @f begin_unsigned == begin
  946. @p procedure pckt_unsigned(@!o:eight_bits;@!x:int_32);
  947.   {output |fnt_num|, |fnt|, or |xxx|}
  948. @!begin_unsigned; pckt_room(5); comp_unsigned(append_byte);
  949. @ Finally, the |pckt_signed| procedure outputs a movement (|right|, |w|,
  950. |x|, |down|, |y|, or |z|) command with its (signed) parameter.
  951. @d begin_signed ==
  952. var xx:int_31; {`absolute value' of |x|}
  953. begin
  954. @d comp_signed(#) ==
  955. if x>=0 then xx:=x @+ else xx:=-(x+1);
  956. if xx<@"80 then
  957.   begin #(o); @+ if x<0 then Incr(x)(@"100); @+ end
  958. else  begin if xx<@"8000 then
  959.     begin #(o+1); @+ if x<0 then Incr(x)(@"10000); @+ end
  960.   else  begin if xx<@"800000 then
  961.       begin #(o+2); @+ if x<0 then Incr(x)(@"1000000); @+ end
  962.     else  begin #(o+3);
  963.       if x>=0 then #(x div @"1000000)
  964.       else  begin x:=@"7FFFFFFF-xx; #((x div @"1000000) + 128); @+ end;
  965.       x:=x mod @"1000000;
  966.       end;
  967.     #(x div @"10000); x:=x mod @"10000;
  968.     end;
  969.   #(x div @"100); x:=x mod @"100;
  970.   end;
  971. @f begin_signed == begin
  972. @p procedure pckt_signed(@!o:eight_bits;@!x:int_32);
  973.   {output |right|, |w|, |x|, |down|, |y|, or |z|}
  974. @!begin_signed; pckt_room(5); comp_signed(append_byte);
  975. @ The |hex_packet| procedure prints the contents of a packet in
  976. hexadecimal form.
  977. @<Basic printing...@>=
  978. @!debug procedure hex_packet(@!p:pckt_pointer); {prints a packet in hex}
  979. var j,@!k,@!l:byte_pointer; {indices into |byte_mem|}
  980. @!d:int_8u;
  981. begin j:=pckt_start[p]-1; k:=pckt_start[p+1]-1;
  982. print_ln(' packet=',p:1,' start=',j+1:1,' length=',k-j:1);
  983. for l:=j+1 to k do
  984.   begin d:=(bo(byte_mem[l])) div 16;
  985.   if d<10 then print(xchr[d+"0"]) @+ else print(xchr[d-10+"A"]);
  986.   d:=(bo(byte_mem[l])) mod 16;
  987.   if d<10 then print(xchr[d+"0"]) @+ else print(xchr[d-10+"A"]);
  988.   if (l=k)or(((l-j) mod 16)=0) then new_line
  989.   else if ((l-j) mod 4)=0 then print('  ')
  990.   else print(' ');
  991.   end;
  992. gubed
  993. @* File names.
  994. The structure of file names is different for different systems; therefore
  995. this part of the program will, in most cases, require system dependent
  996. modifications. Here we assume that a file name consists of three parts:
  997. an area or directory specifying where the file can be found, a name
  998. proper and an extension; \.{\title} assumes that these three parts appear
  999. in order stated above but this need not be true in all cases.
  1000. The font names extracted from \.{DVI} and \.{VF} files consist of an area
  1001. part and a name proper; these are stored as packets consisting of the
  1002. length of the area part followed by the area and the name proper.
  1003. When we print an external font name we simple print the area and the name
  1004. contained in the `file name packet' without delimiter between them.
  1005. This may need to be modified for some systems.
  1006. @^system dependencies@>
  1007. @<Basic printing...@>=
  1008. procedure print_font(@!f:font_number);
  1009. var p:pckt_pointer; {the font name packet}
  1010. @!k:byte_pointer; {index into |byte_mem|}
  1011. @!m:int_31; {font magnification}
  1012. begin print(' = '); p:=font_name(f);
  1013. for k:=pckt_start[p]+1 to pckt_start[p+1]-1 do
  1014.   print(xchr[bo(byte_mem[k])]);
  1015. m:=round((font_scaled(f)/font_design(f))*out_mag);
  1016. if m<>1000 then print(' scaled ',m:1);
  1017. @ Before a font file can be opened for input we must build a string
  1018. with its external name.
  1019. @<Glob...@>=
  1020. @!cur_name:packed array[1..name_length] of char; {external name,
  1021.   with no lower case letters}
  1022. @!cur_name_length:int_15; {this many characters are actually relevant in
  1023.   |cur_name|}
  1024. @ For \.{TFM} and \.{VF} files we just append the apropriate extension
  1025. to the file name packet; in addition a system dependent area part
  1026. (usually different for \.{TFM} and \.{VF} files) is prepended if
  1027. the file name packet contains no area part.
  1028. @^system dependencies@>
  1029. @d append_to_name(#)==
  1030.   if cur_name_length<name_length then
  1031.     begin incr(cur_name_length); cur_name[cur_name_length]:=#;
  1032.     end
  1033.   else overflow(str_name_length,name_length)
  1034. @d make_font_name_end(#)==
  1035.   append_to_name(#[l]); make_name
  1036. @d make_font_name(#)==
  1037.   cur_name_length:=0; for l:=1 to # do make_font_name_end
  1038. @ For files with character raster data (e.g., \.{GF} or \.{PK} files) the
  1039. the extension and\slash or area part will in most cases depend on the
  1040. resolution of the output device (corrected for font magnification).
  1041. If the special character |res_char| occurs in the extension and\slash or
  1042. default area, a character string representing the device resolution will
  1043. be substituted.
  1044. @^system dependencies@>
  1045. @d res_char=='?' {character to be replaced by font resolution}
  1046. @d res_ASCII="?" {|xord[res_char]|}
  1047. @d append_res_to_name(#)==
  1048.   begin c:=#;
  1049.   @!device if c=res_char then
  1050.     for ll:=n_res_digits downto 1 do append_to_name(res_digits[ll])
  1051.   else ecived@;@/
  1052.   append_to_name(c);
  1053.   end
  1054. @d make_font_res_end(#)==
  1055.   append_res_to_name(#[l]); make_name
  1056. @d make_font_res(#)==
  1057.   make_res; cur_name_length:=0; for l:=1 to # do make_font_res_end
  1058. @ @<Glob...@>=
  1059. @!device
  1060. @!f_res:int_16u; {font resolution}
  1061. @!res_digits:array [1..5] of char;
  1062. @!n_res_digits:int_7; {number of significant characters in |res_digits|}
  1063. ecived
  1064. @ The |make_res| procedure creates a sequence of characters representing
  1065. to the font resolution |f_res|.
  1066. @p @!device procedure make_res;
  1067. var r:int_16u;
  1068. begin n_res_digits:=0; r:=f_res;
  1069. repeat incr(n_res_digits);
  1070.   res_digits[n_res_digits]:=xchr["0"+(r mod 10)]; r:=r div 10;
  1071. until r=0;
  1072. ecived
  1073. @ The |make_name| procedure used to build the external file name. The
  1074. global variable |cur_name_length| contains the length of a default area
  1075. which has been copied to |cur_name| before |make_name| is called.
  1076. @^system dependencies@>
  1077. @p procedure make_name(@!e:pckt_pointer);
  1078. var b:eight_bits; {a byte extracted from |byte_mem|}
  1079. @!n:pckt_pointer; {file name packet}
  1080. @!cur_loc,@!cur_limit:byte_pointer; {indices into |byte_mem|}
  1081. @!device
  1082. @!ll:int_15; {loop index}
  1083. ecived@;@/
  1084. @!c:char; {a character to be appended to |cur_name|}
  1085. begin n:=font_name(cur_fnt);
  1086. cur_loc:=pckt_start[n]; cur_limit:=pckt_start[n+1];
  1087. pckt_extract(b); {length of area part}
  1088. if b>0 then cur_name_length:=0;
  1089. while cur_loc<cur_limit do
  1090.   begin pckt_extract(b);
  1091.   if (b>="a")and(b<="z") then Decr(b)("a"-"A"); {convert to upper case}
  1092.   append_to_name(xchr[b]);
  1093.   end;
  1094. cur_loc:=pckt_start[e]; cur_limit:=pckt_start[e+1];
  1095. while cur_loc<cur_limit do
  1096.   begin pckt_extract(b); append_res_to_name(xchr[b]);
  1097.   end;
  1098. while cur_name_length<name_length do
  1099.   begin incr(cur_name_length); cur_name[cur_name_length]:=' ';
  1100.   end;
  1101. @* Defining fonts.
  1102. A detailed description of the \.{TFM} file format can be found in the
  1103. documentation of \TeX, \MF, or \.{TFtoPL}.
  1104. @ \.{DVI} file format does not include information about character widths, since
  1105. that would tend to make the files a lot longer. But a program that reads
  1106. a \.{DVI} file is supposed to know the widths of the characters that appear
  1107. in \\{set\_char} commands. Therefore \.{\title} looks at the font metric
  1108. (\.{TFM}) files for the fonts that are involved.
  1109. @.TFM {\rm files}@>
  1110. The character-width data appears also in other files (e.g., in \.{VF} files
  1111. or in \.{GF} and \.{PK} files that specify bit patterns for digitized
  1112. characters); thus, it is usually possible for \.{DVI} reading programs
  1113. to get by with accessing only one file per font. For \.{VF} reading
  1114. programs there is, however, a problem: (1)~when reading the character
  1115. packets from a \.{VF} file the \.{TFM} width for its local fonts should
  1116. be known in order to analyze and optimize the packets (e.g., determine
  1117. if a packet must indeed be enclosed with |push| and |pop| as implied by
  1118. the \.{VF} format); and (2)~ in order to avoid infinite recursion such
  1119. programs must not try to read a \.{VF} file for a font before a
  1120. character from that font is actually used. Thus \.{\title} reads the
  1121. \.{TFM} file whenever a new font is encountered and delays the decision
  1122. whether this is a virtual font or not.
  1123. @ First of all we need to know for each font~|f| such things as its
  1124. external name, design and scaled size, and the approximate size of
  1125. inter-word spaces. In addition we need to know the range |bc..ec| of
  1126. valid characters for this font, and for each character~|c| in~|f|  we
  1127. need to know if this character exists and if so what is the width of~|c|.
  1128. Depending on the font type of~|f| we may want to know a few other things
  1129. about character~|c| in~|f| such as the character packet from a \.{VF}
  1130. file or the raster data from a \.{PK} file.
  1131. @^font types@>
  1132. In \.{\title} we want to be able to handle the full range
  1133. |@t$-2^{31}$@><=c<@t$2^{31}$@>| of character codes; each character code
  1134. is decomposed into a character residue |0<=res<256| and character
  1135. extension |@t$-2^{23}$@><=ext<@t$2^{23}$@>| such that |c=256*ext+res|.
  1136. At present \.{VFtoVP}, \.{VPtoVF}, and the standard version of \TeX\ use
  1137. only characters in the range |0<=c<256| (i.e., |ext=0|), there are,
  1138. however, extensions of \TeX\ which use characters with |ext<>0|.
  1139. In any case characters with |ext<>0| will be used rather infrequently
  1140. and we want to handle this possibility without too much overhead.
  1141. Some of the data for each character~|c| depend only on its residue:
  1142. first of all its width and escapement; others, such as \.{VF} packets or
  1143. raster data will also depend on its extension. The later will be stored
  1144. as packets in |byte_mem|, and the packets for characters with the same
  1145. residue but different extension will be chained.
  1146. Thus we have to maintain several variables for each character
  1147. residue~|bc<=res<=ec| from each font~|f|; we store each type of variable
  1148. in a large array such that the array index |font_chars(f)+res| points to
  1149. the value for characters with residue |res| from font~|f|.
  1150. @ Quite often a particular width value is shared by several characters in
  1151. a font or even by characters from different fonts; the later will
  1152. probably occur in particular for virtual fonts and the local fonts used
  1153. by them. Thus the array |widths| is used to store all different \.{TFM}
  1154. width values of all legal characters in all fonts; a variable of type
  1155. |width_pointer| is an index into |widths| or is zero if a characters does
  1156. not exist. If the output is for a real typesetting device the |pix_widths|
  1157. array contains the same width values converted to (horizontal) pixels.
  1158. In order to locate a given width value we use again a hash
  1159. table with simple chaining; this time the heads of the individual lists
  1160. appear in the |w_hash| array and the lists proceed through |w_link|
  1161. pointers.
  1162. @d min_pix_value==-@"8000 {smallest allowed pixel value}
  1163. @d max_pix_value==@"7FFF {largest allowed pixel value; this range may not
  1164.   suffice for high resolution output devices}
  1165. @<Types...@>=
  1166. @!width_pointer=0..max_widths; {an index into |widths|}
  1167. @!device
  1168. @!pix_value=min_pix_value..max_pix_value; {a pixel coordinate or displacement}
  1169. ecived
  1170. @ @<Glob...@>=
  1171. @!widths:array[width_pointer] of int_32; {the different width values}
  1172. @!device
  1173. @!pix_widths:array[width_pointer] of pix_value; {the widths in pixels}
  1174. ecived @; @/
  1175. @!w_link:array[width_pointer] of width_pointer; {hash table}
  1176. @!w_hash:array[hash_code] of width_pointer;
  1177. @!n_widths:width_pointer; {first unoccupied position in |widths|}
  1178. @ Initially the |widths| array and all the hash lists are empty, except
  1179. for one entry: the width value zero; in addition we set |widths[0]:=0|.
  1180. @d invalid_width=0 {width pointer for invalid characters}
  1181. @d zero_width=1 {a width pointer to the value zero}
  1182. @<Set init...@>=
  1183. w_hash[0]:=1; w_link[1]:=0; widths[0]:=0; widths[1]:=0; n_widths:=2;
  1184. @!device pix_widths[0]:=0; pix_widths[1]:=0; @+ ecived @;
  1185. for h:=1 to hash_size-1 do w_hash[h]:=0;
  1186. @ The |make_width| function returns an index into |widths| and, if
  1187. necessary, adds a new width value; thus two characters will have the
  1188. same |width_pointer| if and only if their widths agree.
  1189. @d h_pixel_round(#)==round(h_conv*(#))
  1190. @d v_pixel_round(#)==round(v_conv*(#))
  1191. @^system dependencies@>
  1192. @p function make_width(@!w:int_32):width_pointer;
  1193. label found;
  1194. var h:hash_code; {hash code}
  1195. @!p:width_pointer; {where the identifier is being sought}
  1196. @!x:int_16; {intermediate value}
  1197. begin widths[n_widths]:=w;
  1198. @<Compute the width hash code |h|@>;
  1199. @<Compute the width location |p|, |goto| found unless the value is new@>;
  1200. if n_widths=max_widths then overflow(str_widths,max_widths);
  1201. incr(n_widths);
  1202. @!device pix_widths[p]:=h_pixel_round(w); @+ ecived @;
  1203. found:make_width:=p;
  1204. @ A simple hash code is used: If the width value consists of the four
  1205. bytes $b_0b_1b_2b_3$, its hash value will be
  1206. $$(8*b_0+4*b_1+2*b_2+b_3)\,\bmod\,|hash_size|.$$
  1207. @<Compute the width hash...@>=
  1208. if w>=0 then x:=w div @"1000000
  1209. else  begin w:=w+@"40000000; w:=w+@"40000000; x:=(w div @"1000000)+@"80;
  1210.   end;
  1211. w:=w mod @"1000000; x:=x+x+(w div @"10000);
  1212. w:=w mod @"10000; x:=x+x+(w div @"100);
  1213. h:=(x+x+(w mod @"100)) mod hash_size
  1214. @ If the width is new, it has been placed into position |p=n_widths|,
  1215. otherwise |p| will point to its existing location.
  1216. @<Compute the width location...@>=
  1217. p:=w_hash[h];
  1218. while p<>0 do
  1219.   begin if widths[p]=widths[n_widths] then goto found;
  1220.   p:=w_link[p];
  1221.   end;
  1222. p:=n_widths; {the current width is new}
  1223. w_link[p]:=w_hash[h]; w_hash[h]:=p {insert |p| at beginning of hash list}
  1224. @ The |char_widths| array is used to store the |width_pointer|s for all
  1225. different characters among all fonts. For a real typesetting device the
  1226. |char_pixels| array is used to store the horizontal character escapements:
  1227. Initially we use the |pix_widths| values, but these will be replaced by
  1228. the character escapements specified in a \.{PK} or \.{GF} file;
  1229. these values may differ by a small amount.
  1230. The |char_packets| array is used to store the |pckt_pointer|s for all
  1231. different characters among all fonts; they can point to character
  1232. packets from \.{VF} files or, e.g., raster packets from \.{PK} files.
  1233. @<Types...@>=
  1234. @!char_offset=-255..max_chars; {|char_pointer| offset for a font}
  1235. @!char_pointer=0..max_chars; {index into |char_widths| or similar arrays}
  1236. @ @<Glob...@>=
  1237. @!char_widths:array[char_pointer] of width_pointer; {width pointers}
  1238. @!device
  1239. @!char_pixels:array[char_pointer] of pix_value; {character escapements}
  1240. ecived @; @/
  1241. @!char_packets:array[char_pointer] of pckt_pointer; {packet pointers}
  1242. @!n_chars:char_pointer; {first unused position in |char_widths|}
  1243. @ @<Set init...@>=
  1244. n_chars:=0;
  1245. @ The current number of known fonts is |nf|; each known font has an
  1246. internal number |f|, where |0<=f<nf|. For the moment we need for each
  1247. known font: |font_check|, |font_scaled|, |font_design|, |font_space|,
  1248. |font_name|, |font_bc|, |font_ec|, |font_chars|, and |font_type|.
  1249. Here |font_scaled|, |font_design|, and |font_space| are measured in
  1250. \.{DVI} units and |font_chars| is of type |char_offset|:
  1251. the width pointer for character~|c| of the font is stored in
  1252. |char_widths[char_offset+c]| (for |font_bc<=c<=font_ec|).
  1253. Lateron we will need additional information depending on the font type:
  1254. \.{VF} or real (\.{GF}, \.{PK}, or \.{PXL}).
  1255. @<Types...@>=
  1256. @!f_type=new_font_type..max_font_type; {type of a font}
  1257. @!font_number=0..max_fonts;
  1258. @ @<Glob...@>=
  1259. @!nf:font_number;
  1260. @ These data are stored in several arrays and we use \.{WEB} macros
  1261. to access the various fields. Thus it would be simple to store the
  1262. data in an array of record structures and adapt the \.{WEB} macros
  1263. accordingly.
  1264. We will say, e.g., |font_name(f)| for the name field of font~|f|, and
  1265. |font_width(f)(c)| for the width pointer of character~|c| in font~|f|
  1266. and |font_packet(f)(c)| for its character packet (this character
  1267. exists provided |font_bc(f)<=c<=font_ec(f)| and
  1268. |font_width(f)(c)<>invalid_width|). The actual width of character~|c| in
  1269. font~|f| is stored in |widths[font_width(f)(c)]|; the horizontal
  1270. escapement is given by |font_pixel(f)(c)|.
  1271. @d font_check(#)==fnt_check[#] {checksum}
  1272. @d font_scaled(#)==fnt_scaled[#] {scaled or `at' size}
  1273. @d font_design(#)==fnt_design[#] {design size}
  1274. @d font_space(#)==fnt_space[#] {boundary between ``small'' and ``large''
  1275.   spaces}
  1276. @d font_name(#)==fnt_name[#] {area plus name packet}
  1277. @d font_bc(#)==fnt_bc[#] {first character}
  1278. @d font_ec(#)==fnt_ec[#] {last character}
  1279. @d font_chars(#)==fnt_chars[#] {character info offset}
  1280. @d font_type(#)==fnt_type[#] {type of this font}
  1281. @d font_font(#)==fnt_font[#] {use depends on |font_type|}
  1282. @d font_width_end(#)==#]
  1283. @d font_width(#)==char_widths[font_chars(#)+font_width_end
  1284. @d font_pixel(#)==char_pixels[font_chars(#)+font_width_end
  1285. @d font_packet(#)==char_packets[font_chars(#)+font_width_end
  1286. @<Glob...@>=
  1287. @!fnt_check:array [font_number] of int_32; {checksum}
  1288. @!fnt_scaled:array [font_number] of int_31; {scaled size}
  1289. @!fnt_design:array [font_number] of int_31; {design size}
  1290. @!device
  1291. @!fnt_space:array [font_number] of int_32; {boundary between ``small''
  1292.   and ``large'' spaces}
  1293. ecived @;
  1294. @!fnt_name:array [font_number] of pckt_pointer; {pointer to area plus
  1295.   name packet}
  1296. @!fnt_bc:array [font_number] of eight_bits; {first character}
  1297. @!fnt_ec:array [font_number] of eight_bits; {last character}
  1298. @!fnt_chars:array [font_number] of char_offset; {character info offset}
  1299. @!fnt_type:array [font_number] of f_type; {type of font}
  1300. @!fnt_font:array [font_number] of font_number; {use depends on |font_type|}
  1301. @ @d invalid_font==max_fonts {used when there is no valid font}
  1302. @<Set init...@>=
  1303. @!device font_space(invalid_font):=0; @+ ecived @;@/
  1304. nf:=0;
  1305. @ A \.{VF}, or \.{GF}, or \.{PK} file may contain information for
  1306. several characters with the same residue but with different extension;
  1307. all except the first of the corresponding packets in |byte_mem| will
  1308. contain a pointer to the previous one and |font_packet(f)(res)|
  1309. identifies the last such packet.
  1310. A character packet in |byte_mem| starts with a flag byte
  1311. $$\hbox{|flag=@"40*ext_flag+@"20*chain_flag+type_flag|}$$
  1312. with |0<=ext_flag<=3|, |0<=chain_flag<=1|, |0<=type_flag<=@"1F|,
  1313. followed by |ext_flag| bytes with the character extension for this
  1314. packet and, if |chain_flag=1|, by a two byte packet pointer to the
  1315. previous packet for the same font and character residue. The actual
  1316. character packet follows after these header bytes and the
  1317. interpretation of the |type_flag| depends on whether this is a \.{VF}
  1318. packet or a packet for raster data.
  1319. The empty packet is interpreted as a special case of a packet with
  1320. |flag=0|.
  1321. @d ext_flag=@"40
  1322. @d chain_flag=@"20
  1323. @<Types...@>=
  1324. @!type_flag=0..chain_flag-1; {the range of values for the |type_flag|}
  1325. @ The global variable |cur_fnt| is the internal font number of the
  1326. currently selected font, or equals |invalid_font| if no font has
  1327. been selected; |cur_res| and |cur_ext| are the residue and extension
  1328. part of the current character code. The type of a character packet
  1329. located by the |find_packet| function defined below is |cur_type|.
  1330. While building a character packet for a character, |pckt_ext| and
  1331. |pckt_res| are the extension and residue of this character; |pckt_dup|
  1332. indicates whether a packet for this extension exists already.
  1333. @<Glob...@>=
  1334. @!cur_fnt:font_number; {the currently selected font}
  1335. @!cur_ext:int_24; {the current character extension}
  1336. @!cur_res:int_8u; {the current character residue}
  1337. @!cur_type:type_flag; {type of the current character packet}
  1338. @!pckt_ext:int_24; {character extension for the current character packet}
  1339. @!pckt_res:int_8u; {character residue for the current character packet}
  1340. @!pckt_dup:boolean; {is there a previous packet for the same extension?}
  1341. @!pckt_prev:pckt_pointer; {a previous packet for the same extension}
  1342. @!pckt_m_msg,@!pckt_s_msg,@!pckt_d_msg:int_7; {counts for various character
  1343.   packet error messages}
  1344. @ @<Set init...@>=
  1345. cur_fnt:=invalid_font; pckt_m_msg:=0; pckt_s_msg:=0; pckt_d_msg:=0;
  1346. @ The |find_packet| functions is used to locate the character packet for
  1347. the character with residue~|cur_res| and extension~|cur_ext| from
  1348. font~|cur_fnt| and returns |false| if no packet exists for any extension;
  1349. otherwise the result is |true| and the global variables |cur_packet|,
  1350. |cur_type|, |cur_loc|, and |cur_limit| are initialized. In case none of
  1351. the character packets has the correct extension, the last one in the
  1352. chain (the one defined first) is used instead and |cur_ext| is changed
  1353. accordingly.
  1354. @p function find_packet:boolean;
  1355. label found,exit;
  1356. var p,@!q:pckt_pointer; {current and next packet}
  1357. @!f:eight_bits; {a flag byte}
  1358. @!e:int_24; {extension for a packet}
  1359. begin @<Locate a character packet and |goto found| if found@>;
  1360. if font_packet(cur_fnt)(cur_res)=invalid_packet then
  1361.   begin if pckt_m_msg<10 then {stop telling after first 10 times}
  1362.     begin print_ln('---missing character packet for character ',cur_res:1,
  1363. @.missing character packet...@>
  1364.       ' font ',cur_fnt:1);
  1365.     incr(pckt_m_msg); mark_error;
  1366.     if pckt_m_msg=10 then print_ln('---further messages suppressed.');
  1367.     end;
  1368.   find_packet:=false; return;
  1369.   end;
  1370. if pckt_s_msg<10 then {stop telling after first 10 times}
  1371.   begin print_ln('---substituted character packet with extension ',
  1372. @.substituted character packet...@>
  1373.     e:1,' instead of ',cur_ext:1,' for character ',cur_res:1,
  1374.     ' font ',cur_fnt:1);
  1375.   incr(pckt_s_msg); mark_error;
  1376.   if pckt_s_msg=10 then print_ln('---further messages suppressed.');
  1377.   end;
  1378. cur_ext:=e;
  1379. found: cur_pckt:=p; cur_type:=f; find_packet:=true;
  1380. exit: end;
  1381. @ @<Locate a character packet and |goto found| if found@>=
  1382. q:=font_packet(cur_fnt)(cur_res);
  1383. while q<>invalid_packet do
  1384.   begin p:=q; q:=invalid_packet;
  1385.   cur_loc:=pckt_start[p]; cur_limit:=pckt_start[p+1];
  1386.   if p=empty_packet then
  1387.     begin e:=0; f:=0;
  1388.     end
  1389.   else  begin pckt_extract(f);
  1390.     case (f div ext_flag) of
  1391.     0: e:=0;
  1392.     1: e:=pckt_ubyte;
  1393.     2: e:=pckt_upair;
  1394.     3: e:=pckt_strio;
  1395.     end; {there are no other cases}
  1396.     if (f mod ext_flag)>=chain_flag then q:=pckt_upair;
  1397.     f:=f mod chain_flag;
  1398.     end;
  1399.   if e=cur_ext then goto found;
  1400.   end
  1401. @ The |start_packet| procedure is used to create the header bytes of a
  1402. character packet for the character with residue~|cur_res| and
  1403. extension~|cur_ext| from font~|cur_fnt|; if a previous such a packet
  1404. exists, we try to build an exact duplicate, i.e., use the chain field of
  1405. that previous packet.
  1406. @p procedure start_packet(@!t:type_flag);
  1407. label found,not_found;
  1408. var p,@!q:pckt_pointer; {current and next packet}
  1409. @!f:int_8u; {a flag byte}
  1410. @!e:int_32; {extension for a packet}
  1411. @!cur_loc: byte_pointer; {current location in a packet}
  1412. @!cur_limit: byte_pointer; {start of next packet}
  1413. begin @<Locate a character packet and |goto found| if found@>;
  1414. q:=font_packet(cur_fnt)(cur_res); pckt_dup:=false; goto not_found;
  1415. found: pckt_dup:=true; pckt_prev:=p;
  1416. not_found: pckt_ext:=cur_ext; pckt_res:=cur_res; pckt_room(6);
  1417. @!debug if byte_ptr<>pckt_start[pckt_ptr] then confusion(str_packets);
  1418. gubed @;@/
  1419. if q=invalid_packet then f:=t @+ else f:=t+chain_flag;
  1420. e:=cur_ext;
  1421. if e<0 then Incr(e)(@"1000000);
  1422. if e=0 then append_byte(f) @+ else @;
  1423.   begin if e<@"100 then append_byte(f+ext_flag) @+ else @;
  1424.     begin if e<@"10000 then append_byte(f+ext_flag+ext_flag) @+ else @;
  1425.       begin append_byte(f+ext_flag+ext_flag+ext_flag);
  1426.       append_byte(e div @"10000); e:=e mod @"10000;
  1427.       end;
  1428.     append_byte(e div @"100); e:=e mod @"100;
  1429.     end;
  1430.   append_byte(e);
  1431.   end;
  1432. if q<>invalid_packet then
  1433.   begin append_byte(q div @"100); append_byte(q mod @"100);
  1434.   end;
  1435. @ The |build_packet| procedure is used to finish a character packet.
  1436. If a previous packet for the same character extension exists, the new
  1437. one is discarded; if the two packets are identical, as it occasionally
  1438. occurs for raster files, this is done without an error message.
  1439. @p procedure build_packet;
  1440. var k,@!l:byte_pointer; {indices into |byte_mem|}
  1441. begin if pckt_dup then
  1442.   begin k:=pckt_start[pckt_prev+1]; l:=pckt_start[pckt_ptr];
  1443.   if (byte_ptr-l)<>(k-pckt_start[pckt_prev]) then pckt_dup:=false;
  1444.   while pckt_dup and(byte_ptr>l) do
  1445.     begin flush_byte; decr(k);
  1446.     if byte_mem[byte_ptr]<>byte_mem[k] then pckt_dup:=false;
  1447.     end;
  1448.   if (not pckt_dup)and(pckt_d_msg<10) then {stop telling after first 10 times}
  1449.     begin print('---duplicate packet for character ',pckt_res:1);
  1450. @.duplicate packet for character...@>
  1451.     if pckt_ext<>0 then print('.',pckt_ext:1);
  1452.     print_ln(' font ',cur_fnt:1);
  1453.     incr(pckt_d_msg); mark_error;
  1454.     if pckt_d_msg=10 then print_ln('---further messages suppressed.');
  1455.     end;
  1456.   byte_ptr:=l;
  1457.   end
  1458. else font_packet(cur_fnt)(pckt_res):=make_packet;
  1459. @ In order to read \.{TFM} files the program uses the binary file
  1460. variable |tfm_file|.
  1461. @<Glob...@>=
  1462. @!tfm_file:byte_file; {a \.{TFM} file}
  1463. @!tfm_ext:pckt_pointer; {extension for \.{TFM} files}
  1464. @ @<Initialize predefined strings@>=
  1465. id4(".")("T")("F")("M")(tfm_ext); {file name extension for \.{TFM} files}
  1466. @ If no font directory has been specified, \.{\title} is supposed to use
  1467. the default \.{TFM} directory, which is a system-dependent place where
  1468. the \.{TFM} files for standard fonts are kept.
  1469. The string variable |TFM_default_area| contains the name of this area.
  1470. @^system dependencies@>
  1471. @d TFM_default_area_name=='TeXfonts:' {change this to the correct name}
  1472. @d TFM_default_area_name_length=9 {change this to the correct length}
  1473. @<Glob...@>=
  1474. @!TFM_default_area:packed array[1..TFM_default_area_name_length] of char;
  1475. @ @<Set init...@>=
  1476. TFM_default_area:=TFM_default_area_name;
  1477. @ If a \.{TFM} file is badly malformed, we say |bad_font|; for a \.{TFM}
  1478. file the |bad_tfm| procedure is used to give an error message which
  1479. refers the user to \.{TFtoPL} and \.{PLtoTF}, and terminates \.{\title}.
  1480. @<Error handling...@>=
  1481. procedure bad_tfm;
  1482. begin print('Bad TFM file'); print_font(cur_fnt); print_ln('!');
  1483. @.Bad TFM file@>
  1484. abort('Use TFtoPL/PLtoTF to diagnose and correct the problem');
  1485. @.Use TFtoPL/PLtoTF@>
  1486. procedure bad_font;
  1487. begin new_line;
  1488. case font_type(cur_fnt) of
  1489.   new_font_type: bad_tfm;
  1490.   @<Cases for |bad_font|@>@;@/
  1491.   end; {there are no other cases}
  1492. @ To prepare |tfm_file| for input we |reset| it.
  1493. @<TFM: Open |tfm_file|@>=
  1494. make_font_name(TFM_default_area_name_length)(TFM_default_area)(tfm_ext);
  1495. reset(tfm_file,cur_name);
  1496. if eof(tfm_file) then
  1497. @^system dependencies@>
  1498.   abort('---not loaded, TFM file can''t be opened!')
  1499. @.TFM file can\'t be opened@>
  1500. @ It turns out to be convenient to read four bytes at a time, when we
  1501. are inputting from \.{TFM} files. The input goes into global variables
  1502. |tfm_b0|, |tfm_b1|, |tfm_b2|, and |tfm_b3|, with |tfm_b0| getting the
  1503. first byte and |tfm_b3| the fourth.
  1504. @<Glob...@>=
  1505. @!tfm_b0,@!tfm_b1,@!tfm_b2,@!tfm_b3: eight_bits; {four bytes input at once}
  1506. @ Reading a \.{TFM} file should be done as efficient as possible for a
  1507. particular system; on many systems this means that a large number of
  1508. bytes from |tfm_file| is read into a buffer and will then be extracted
  1509. from that buffer. In order to simplify such system dependent changes
  1510. we use the \.{WEB} macro |tfm_byte| to extract the next \.{TFM} byte;
  1511. this macro and |eof(tfm_file)| are used only in the |read_tfm_word|
  1512. procedure which sets |tfm_b0| through |tfm_b3| to the next four bytes
  1513. in the current \.{TFM} file. Here we give simple minded definitions in
  1514. terms of standard \PASCAL.
  1515. @^system dependencies@>
  1516. @^optimization@>
  1517. @d tfm_byte(#)==read(tfm_file,#) {read next \.{TFM} byte}
  1518. @p procedure read_tfm_word;
  1519. begin tfm_byte(tfm_b0); tfm_byte(tfm_b1);
  1520. tfm_byte(tfm_b2); tfm_byte(tfm_b3);
  1521. if eof(tfm_file) then bad_font;
  1522. @ Here are three procedures used to check the consistency of font files:
  1523. First, the |check_check_sum| procedure compares two check sum values: a
  1524. warning is given if they differ and are both non-zero; if the second
  1525. value is not zero it may replace the first one.
  1526. Next, the |check_design_size| procedure compares two design size
  1527. values: a warning is given if they differ by more than a small amount.
  1528. Finally, the |check_width| function compares the character width value
  1529. for character |cur_res| read from a \.{VF} or raster file for font
  1530. |cur_fnt| with the value previously read from the \.{TFM} file and
  1531. returns the width pointer for that value; a warning is given if the two
  1532. values differ.
  1533. @p procedure check_check_sum(@!c:int_32;@!u:boolean);
  1534.   {compare |font_check(cur_fnt)| with |c|}
  1535. begin if (c<>font_check(cur_fnt))and(c<>0) then
  1536.   begin
  1537.   if font_check(cur_fnt)<>0 then
  1538.     begin new_line; print_ln('---beware: check sums do not agree!   (',
  1539. @.beware: check sums do not agree@>
  1540. @.check sums do not agree@>
  1541.       c:1,' vs. ',font_check(cur_fnt):1,')');
  1542.     mark_error;
  1543.     end;
  1544.   if u then font_check(cur_fnt):=c;
  1545.   end;
  1546. procedure check_design_size(@!d:int_32);
  1547.   {compare |font_design(cur_fnt)| with |d|}
  1548. begin if abs(d-font_design(cur_fnt))>2 then
  1549.   begin new_line; print_ln('---beware: design sizes do not agree!   (',
  1550. @.beware: design sizes do not agree@>
  1551. @.design sizes do not agree@>
  1552.     d:1,' vs. ',font_design(cur_fnt):1,')');
  1553.   mark_error;
  1554.   end;
  1555. function check_width(w:int_32):width_pointer;
  1556.   {compare |widths[font_width(cur_fnt)(cur_res)]| with |w|}
  1557. var wp:width_pointer; {pointer to \.{TFM} width value}
  1558. begin if (cur_res>=font_bc(cur_fnt))and(cur_res<=font_ec(cur_fnt)) then
  1559.   wp:=font_width(cur_fnt)(cur_res)
  1560. else wp:=invalid_width;
  1561. if wp=invalid_width then
  1562.   begin print_nl('Bad char ',cur_res:1);
  1563. @.Bad char c@>
  1564.   if cur_ext<>0 then print('.',cur_ext:1);
  1565.   print(' font ',cur_fnt:1); print_font(cur_fnt);
  1566.   abort(' (compare TFM file)');
  1567.   end;
  1568. if w<>widths[wp] then
  1569.   begin new_line; print_ln('---beware: char widths do not agree!   (',
  1570. @.beware: char widths do not agree@>
  1571. @.char widths do not agree@>
  1572.     w:1,' vs. ',widths[wp]:1,')');
  1573.   mark_error;
  1574.   end;
  1575. check_width:=wp;
  1576. @ When processing a font definition we put the data extracted from the
  1577. \.{DVI} or \.{VF} file into the fields of |font_data[nf]| and call
  1578. |make_font| to obtain the internal font number for this font.
  1579. The |make_font| function determines if this font is already defined and,
  1580. if this is not the case, reads the \.{TFM} file.
  1581. @p function make_font:font_number;
  1582. var l:int_16; {loop index}
  1583. @!p:char_pointer; {index into |char_widths|}
  1584. @!q:width_pointer; {index into |widths|}
  1585. @!bc,@!ec:int_15; {first and last character in this font}
  1586. @!lh:int_15; {length of header in four byte words}
  1587. @!nw:int_15; {number of words in width table}
  1588. @!w:int_32; {a four byte integer}
  1589. @!save_fnt:font_number; {used to save |cur_fnt|}
  1590. @<Variables for scaling computation@>@;
  1591. begin save_fnt:=cur_fnt; {save}
  1592. cur_fnt:=0;
  1593. while (font_name(cur_fnt)<>font_name(nf))or@|
  1594.   (font_scaled(cur_fnt)<>font_scaled(nf)) do incr(cur_fnt);
  1595. d_print(' => ',cur_fnt:1); print_font(cur_fnt);
  1596. if cur_fnt<nf then  begin check_check_sum(font_check(nf),true);
  1597.   check_design_size(font_design(nf));
  1598.   d_print(' loaded previously');
  1599.   end
  1600. else @<Define a new font@>;
  1601. new_line;
  1602. make_font:=cur_fnt;
  1603. cur_fnt:=save_fnt; {restore}
  1604. @ @<Define a new font@>=
  1605. begin if nf=max_fonts then overflow(str_fonts,max_fonts);
  1606.   font_type(cur_fnt):=new_font_type; font_font(cur_fnt):=invalid_font;
  1607.   @<TFM: Open |tfm_file|@>;
  1608.   @<TFM: Read past the header data@>;
  1609.   @<TFM: Store character-width indices@>;
  1610.   @<TFM: Read and convert the width values@>;
  1611.   @<TFM: Convert character-width indices to character-width pointers@>;
  1612.   close_in(tfm_file);
  1613.   d_print(' loaded at ',font_scaled(cur_fnt):1,' DVI units');
  1614.   incr(nf);
  1615.   end
  1616. @ @<Glob...@>=
  1617. @!tfm_conv:real; {\.{DVI} units per absolute \.{TFM} unit}
  1618. @ We will use the following \.{WEB} macros to construct integers from
  1619. two or four of the four bytes read by |read_tfm_word|.
  1620. @^system dependencies@>
  1621. @d tfm_b01(#)== {|tfm_b0..tfm_b1| as non-negative integer}
  1622. if tfm_b0>127 then bad_font
  1623. else #:=tfm_b0*256+tfm_b1
  1624. @d tfm_b23(#)== {|tfm_b2..tfm_b3| as non-negative integer}
  1625. if tfm_b2>127 then bad_font
  1626. else #:=tfm_b2*256+tfm_b3
  1627. @d tfm_squad(#)== {|tfm_b0..tfm_b3| as signed integer}
  1628. if tfm_b0<128 then #:=((tfm_b0*256+tfm_b1)*256+tfm_b2)*256+tfm_b3
  1629. else #:=(((tfm_b0-256)*256+tfm_b1)*256+tfm_b2)*256+tfm_b3
  1630. @d tfm_uquad== {|tfm_b0..tfm_b3| as unsigned integer}
  1631. (((tfm_b0*256+tfm_b1)*256+tfm_b2)*256+tfm_b3)
  1632. @<TFM: Read past the header data@>=
  1633. read_tfm_word; tfm_b23(lh);
  1634. read_tfm_word; tfm_b01(bc); tfm_b23(ec);
  1635. if ec<bc then
  1636.   begin bc:=1; ec:=0;
  1637.   end
  1638. else if ec>255 then bad_font;
  1639. read_tfm_word; tfm_b01(nw);
  1640. if (nw=0)or(nw>256) then bad_font;
  1641. for l:=-2 to lh do
  1642.   begin read_tfm_word;
  1643.   if l=1 then  begin tfm_squad(w); check_check_sum(w,true);
  1644.     end
  1645.   else if l=2 then  begin if tfm_b0>127 then bad_font;
  1646.     check_design_size(round(tfm_conv*tfm_uquad));
  1647.     end;
  1648.   end
  1649. @ The width indices for the characters are stored in positions |n_chars|
  1650. through |n_chars-bc+ec| of the |char_widths| array; if characters on
  1651. either end of the range |bc..ec| do not exist, they are ignored and the
  1652. range is adjusted accordingly.
  1653. @<TFM: Store character-width indices@>=
  1654. read_tfm_word;
  1655. while (tfm_b0=0)and(bc<=ec) do
  1656.   begin incr(bc); read_tfm_word;
  1657.   end;
  1658. font_bc(cur_fnt):=bc; font_chars(cur_fnt):=n_chars-bc;
  1659. if ec>=max_chars-font_chars(cur_fnt) then overflow(str_chars,max_chars);
  1660. for l:=bc to ec do
  1661.   begin char_widths[n_chars]:=tfm_b0; incr(n_chars); read_tfm_word;
  1662.   end;
  1663. while (char_widths[n_chars-1]=0)and(ec>=bc) do
  1664.   begin decr(n_chars); decr(ec);
  1665.   end;
  1666. font_ec(cur_fnt):=ec
  1667. @ The most important part of |make_font| is the width computation, which
  1668. involves multiplying the relative widths in the \.{TFM} file by the
  1669. scaling factor in the \.{DVI} file. A similar computation is used for
  1670. dimensions read from \.{VF} files. This fixed-point multiplication must
  1671. be done with precisely the same accuracy by all \.{DVI}-reading programs,
  1672. in order to validate the assumptions made by \.{DVI}-writing programs
  1673. like \TeX82.
  1674. Let us therefore summarize what needs to be done. Each width in a \.{TFM}
  1675. file appears as a four-byte quantity called a |fix_word|.  A |fix_word|
  1676. whose respective bytes are $(a,b,c,d)$ represents the number
  1677. $$x=\left\{\vcenter{\halign{$#$,\hfil\qquad&if $#$\hfil\cr
  1678. b\cdot2^{-4}+c\cdot2^{-12}+d\cdot2^{-20}&a=0;\cr
  1679. -16+b\cdot2^{-4}+c\cdot2^{-12}+d\cdot2^{-20}&a=255.\cr}}\right.$$
  1680. (No other choices of $a$ are allowed, since the magnitude of a \.{TFM}
  1681. dimension must be less than 16.)  We want to multiply this quantity by the
  1682. integer~|z|, which is known to be less than $2^{27}$.
  1683. If $|z|<2^{23}$, the individual multiplications $b\cdot z$, $c\cdot z$,
  1684. $d\cdot z$ cannot overflow; otherwise we will divide |z| by 2, 4, 8, or
  1685. 16, to obtain a multiplier less than $2^{23}$, and we can compensate for
  1686. this later. If |z| has thereby been replaced by $|z|^\prime=|z|/2^e$, let
  1687. $\beta=2^{4-e}$; we shall compute
  1688. $$\lfloor(b+c\cdot2^{-8}+d\cdot2^{-16})\,z^\prime/\beta\rfloor$$ if $a=0$,
  1689. or the same quantity minus $\alpha=2^{4+e}z^\prime$ if $a=255$.
  1690. This calculation must be done exactly, for the reasons stated above; the
  1691. following program does the job in a system-independent way, assuming
  1692. that arithmetic is exact on numbers less than $2^{31}$ in magnitude. We
  1693. use \.{WEB} macros for various versions of this computation.
  1694. @^system dependencies@>
  1695. @^optimization@>
  1696. @d tfm_fix3u== {convert |tfm_b1..tfm_b3| to an unsigned scaled dimension}
  1697. (((((tfm_b3*z)div@'400)+(tfm_b2*z))div@'400)+(tfm_b1*z))div beta
  1698. @d tfm_fix4(#)== {convert |tfm_b0..tfm_b3| to a scaled dimension}
  1699.   #:=tfm_fix3u;
  1700.   if tfm_b0>0 then if tfm_b0=255 then Decr(#)(alpha) else bad_font
  1701. @d tfm_fix3(#)== {convert |tfm_b1..tfm_b3| to a scaled dimension}
  1702.   #:=tfm_fix3u; @+ if tfm_b1>127 then Decr(#)(alpha)
  1703. @d tfm_fix2== {convert |tfm_b2..tfm_b3| to a scaled dimension}
  1704.   if tfm_b2>127 then tfm_b1:=255 else tfm_b1:=0;
  1705.   tfm_fix3
  1706. @d tfm_fix1== {convert |tfm_b3| to a scaled dimension}
  1707.   if tfm_b3>127 then tfm_b1:=255 else tfm_b1:=0;
  1708.   tfm_b2:=tfm_b1; tfm_fix3
  1709. @<Variables for scaling computation@>=
  1710. @!z:int_32; {multiplier}
  1711. @!alpha:int_32; {correction for negative values}
  1712. @!beta:int_15; {divisor}
  1713. @ @<Replace |z| by $|z|^\prime$ and compute $\alpha,\beta$@>=
  1714. alpha:=16;
  1715. while z>=@'40000000 do
  1716.   begin z:=z div 2; alpha:=alpha+alpha;
  1717.   end;
  1718. beta:=256 div alpha; alpha:=alpha*z
  1719. @ The first width value, which indicates that a character does not exist
  1720. and which must vanish, is converted to |invalid_width|; the other width
  1721. values are scaled by |font_scaled(cur_fnt)| and converted to width
  1722. pointers by |make_width|. The resulting width pointers are stored
  1723. temporarily in the |char_widths| array, following the with indices.
  1724. @<TFM: Read and convert the width values@>=
  1725. if nw-1>max_chars-n_chars then overflow(str_chars,max_chars);
  1726. if (tfm_b0<>0)or(tfm_b1<>0)or(tfm_b2<>0)or(tfm_b3<>0) then bad_font
  1727.   else char_widths[n_chars]:=invalid_width;
  1728. z:=font_scaled(cur_fnt);
  1729. @!device font_space(cur_fnt):=z div 6; {this is a 3-unit ``thin space''}
  1730. ecived @;
  1731. @<Replace |z|...@>;
  1732. for p:=n_chars+1 to n_chars+nw-1 do
  1733.   begin read_tfm_word; tfm_fix4(w);
  1734.   char_widths[p]:=make_width(w);
  1735.   end
  1736. @ We simply translate the width indices into width pointers. In addition
  1737. we initialize the character packets with the invalid packet.
  1738. @<TFM: Convert character-width indices to character-width pointers@>=
  1739. for p:=font_chars(cur_fnt)+bc to n_chars-1 do
  1740.   begin q:=char_widths[n_chars+char_widths[p]]; char_widths[p]:=q;
  1741.   @!device char_pixels[p]:=pix_widths[q]; @+ ecived @; @/
  1742.   char_packets[p]:=invalid_packet;
  1743.   end
  1744. @* Low-level DVI input routines.
  1745. The program uses the binary file variable |dvi_file| for its main input
  1746. file; |dvi_loc| is the number of the byte about to be read next from
  1747. |dvi_file|.
  1748. @<Glob...@>=
  1749. @!dvi_file:byte_file; {the stuff we are \.{\title}ing}
  1750. @!dvi_loc:int_32; {where we are about to look, in |dvi_file|}
  1751. @ If the \.{DVI} file is badly malformed, we say |bad_dvi|; this
  1752. procedure gives an error message which refers the user to \.{DVItype},
  1753. and terminates \.{\title}.
  1754. @<Error handling...@>=
  1755. procedure bad_dvi;
  1756. begin new_line; print_ln('Bad DVI file: loc=',dvi_loc:1,'!');
  1757. @.Bad DVI file@>
  1758. print(' Use DVItype with output level');
  1759. @.Use DVItype@>
  1760. if random_reading then print('=4') @+ else print('<4');
  1761. abort('to diagnose the problem');
  1762. @ To prepare |dvi_file| for input, we |reset| it.
  1763. @<Open input file(s)@>=
  1764. reset(dvi_file); {prepares to read packed bytes from |dvi_file|}
  1765. dvi_loc:=0;
  1766. @ Reading the \.{DVI} file should be done as efficient as possible for a
  1767. particular system; on many systems this means that a large number of
  1768. bytes from |dvi_file| is read into a buffer and will then be extracted
  1769. from that buffer. In order to simplify such system dependent changes
  1770. we use a pair of \.{WEB} macros: |dvi_byte| extracts the next \.{DVI}
  1771. byte and |dvi_eof| is |true| if we have reached the end of the \.{DVI}
  1772. file. Here we give simple minded definitions for these macros in terms
  1773. of standard \PASCAL.
  1774. @^system dependencies@>
  1775. @^optimization@>
  1776. @d dvi_eof == eof(dvi_file) {has the \.{DVI} file been exhausted?}
  1777. @d dvi_byte(#) ==
  1778.   if dvi_eof then bad_dvi
  1779.   else read(dvi_file,#) {obtain next \.{DVI} byte}
  1780. @ Next we come to the routines that are used only if |random_reading|    is
  1781. |true|. The driver program below needs two such routines: |dvi_length| should
  1782. compute the total number of bytes in |dvi_file|, possibly also
  1783. causing |eof(dvi_file)| to be true; and |dvi_move(n)| should position
  1784. |dvi_file| so that the next |dvi_byte| will read byte |n|, starting with
  1785. |n=0| for the first byte in the file.
  1786. @^system dependencies@>
  1787. Such routines are, of course, highly system dependent. They are implemented
  1788. here in terms of two assumed system routines called |set_pos| and |cur_pos|.
  1789. The call |set_pos(f,n)| moves to item |n| in file |f|, unless |n| is
  1790. negative or larger than the total number of items in |f|; in the latter
  1791. case, |set_pos(f,n)| moves to the end of file |f|.
  1792. The call |cur_pos(f)| gives the total number of items in |f|, if
  1793. |eof(f)| is true; we use |cur_pos| only in such a situation.
  1794. @p function dvi_length:int_32;
  1795. begin set_pos(dvi_file,-1); dvi_length:=cur_pos(dvi_file);
  1796. procedure dvi_move(@!n:int_32);
  1797. begin set_pos(dvi_file,n); dvi_loc:=n;
  1798. @ We need seven simple functions to read the next byte or bytes
  1799. from |dvi_file|.
  1800. @p function dvi_sbyte:int_8; {returns the next byte, signed}
  1801. @!begin_byte(dvi_byte); incr(dvi_loc); comp_sbyte(dvi_sbyte);
  1802. function dvi_ubyte:int_8u; {returns the next byte, unsigned}
  1803. @!begin_byte(dvi_byte); incr(dvi_loc); comp_ubyte(dvi_ubyte);
  1804. function dvi_spair:int_16; {returns the next two bytes, signed}
  1805. @!begin_pair(dvi_byte); Incr(dvi_loc)(2); comp_spair(dvi_spair);
  1806. function dvi_upair:int_16u; {returns the next two bytes, unsigned}
  1807. @!begin_pair(dvi_byte); Incr(dvi_loc)(2); comp_upair(dvi_upair);
  1808. function dvi_strio:int_24; {returns the next three bytes, signed}
  1809. @!begin_trio(dvi_byte); Incr(dvi_loc)(3); comp_strio(dvi_strio);
  1810. function dvi_utrio:int_24u; {returns the next three bytes, unsigned}
  1811. @!begin_trio(dvi_byte); Incr(dvi_loc)(3); comp_utrio(dvi_utrio);
  1812. function dvi_squad:int_32; {returns the next four bytes, signed}
  1813. @!begin_quad(dvi_byte); Incr(dvi_loc)(4); comp_squad(dvi_squad);
  1814. @ Three other functions are used in cases where a four byte integer
  1815. (which is always signed) must have a non-negative value, a positive
  1816. value, or is a pointer which must be either positive or |=-1|.
  1817. @p function dvi_uquad:int_31; {result must be non-negative}
  1818. var x:int_32;
  1819. begin x:=dvi_squad; if x<0 then bad_dvi
  1820. else dvi_uquad:=x;
  1821. function dvi_pquad:int_31; {result must be positive}
  1822. var x:int_32;
  1823. begin x:=dvi_squad; if x<=0 then bad_dvi
  1824. else dvi_pquad:=x;
  1825. function dvi_pointer:int_32; {result must be positive or |=-1|}
  1826. var x:int_32;
  1827. begin x:=dvi_squad; if (x<=0)and(x<>-1) then bad_dvi
  1828. else dvi_pointer:=x;
  1829. @ Given the structure of the \.{DVI} commands it is fairly obvious
  1830. that their interpretation consists of two steps: First zero to four
  1831. bytes are read in order to obtain the value of the first parameter
  1832. (e.g., zero bytes for |set_char_0|, four bytes for |set4|); then,
  1833. depending on the command class, a specific action is performed (e.g.,
  1834. typeset a character but don't move the reference point for |put1..put4|).
  1835. The \.{DVItype} program uses large case statements for both steps;
  1836. unfortunately some \PASCAL\ compilers fail to implement large case
  1837. statements efficiently -- in particular those as the one used in the
  1838. |first_par| function of \.{DVItype}. Here we use a pair of look up tables:
  1839. |dvi_par| determines how to obtain the value of the first parameter, and
  1840. |dvi_cl| determines the command class.
  1841. A slight complication arises from the fact that we want to decompose the
  1842. character code of each character to be typset into a residue
  1843. |0<=char_res<256| and extension: |char_code=char_res+256*char_ext|;
  1844. the \.{TFM} widths as well as the pixel widths for a given resolution
  1845. are the same for all characters in a font with the same residue.
  1846. @d two_cases(#)==#,#+1
  1847. @d three_cases(#)==#,#+1,#+2
  1848. @d five_cases(#)==#,#+1,#+2,#+3,#+4
  1849. @ First we define the values used as array elements of |dvi_par|; we
  1850. distinguish between pure numbers and dimensions because dimensions read
  1851. from a \.{VF} file must be scaled.
  1852. @d char_par=0 {character for \\{set} and |put|}
  1853. @d no_par=1 {no parameter}
  1854. @d dim1_par=2 {one-byte signed dimension}
  1855. @d num1_par=3 {one-byte unsigned number}
  1856. @d dim2_par=4 {two-byte signed dimension}
  1857. @d num2_par=5 {two-byte unsigned number}
  1858. @d dim3_par=6 {three-byte signed dimension}
  1859. @d num3_par=7 {three-byte unsigned number}
  1860. @d dim4_par=8 {four-byte signed dimension}
  1861. @d num4_par=9 {four-byte signed number}
  1862. @d numu_par=10 {four-byte non-negative number}
  1863. @d rule_par=11 {dimensions for |set_rule| and |put_rule|}
  1864. @d fnt_par=12 {font for |fnt_num| commands}
  1865. @d max_par=12 {largest possible value}
  1866. @<Types...@>=
  1867. @!cmd_par=char_par..max_par;
  1868. @ Here we declare the array |dvi_par|.
  1869. @<Globals...@>=
  1870. @!dvi_par:packed array [eight_bits] of cmd_par;
  1871. @ And here we initialize it.
  1872. @<Set init...@>=
  1873. for i:=0 to put1+3 do dvi_par[i]:=char_par;@/
  1874. for i:=nop to 255 do dvi_par[i]:=no_par;@/
  1875. dvi_par[set_rule]:=rule_par; dvi_par[put_rule]:=rule_par;@/
  1876. dvi_par[right1]:=dim1_par; dvi_par[right1+1]:=dim2_par;
  1877. dvi_par[right1+2]:=dim3_par; dvi_par[right1+3]:=dim4_par;@/
  1878. for i:=fnt_num_0 to fnt_num_0+63 do dvi_par[i]:=fnt_par;@/
  1879. dvi_par[fnt1]:=num1_par; dvi_par[fnt1+1]:=num2_par;
  1880. dvi_par[fnt1+2]:=num3_par; dvi_par[fnt1+3]:=num4_par;@/
  1881. dvi_par[xxx1]:=num1_par; dvi_par[xxx1+1]:=num2_par;
  1882. dvi_par[xxx1+2]:=num3_par; dvi_par[xxx1+3]:=numu_par;@/
  1883. for i:=0 to 3 do
  1884.   begin dvi_par[i+w1]:=dvi_par[i+right1];
  1885.   dvi_par[i+x1]:=dvi_par[i+right1];
  1886.   dvi_par[i+down1]:=dvi_par[i+right1];
  1887.   dvi_par[i+y1]:=dvi_par[i+right1];
  1888.   dvi_par[i+z1]:=dvi_par[i+right1];
  1889.   dvi_par[i+fnt_def1]:=dvi_par[i+fnt1];
  1890.   end;
  1891. @ Next we define the values used as array elements of |dvi_cl|;
  1892. several \.{DVI} commands (e.g., |nop|, |bop|, |eop|, |pre|, |post|) will
  1893. allways be treated separately and are therfore assigned to the invalid
  1894. class here.
  1895. @d char_cl=0
  1896. @d rule_cl=char_cl+1
  1897. @d xxx_cl=char_cl+2
  1898. @d push_cl=3
  1899. @d pop_cl=4
  1900. @d w0_cl=5
  1901. @d x0_cl=w0_cl+1
  1902. @d right_cl=w0_cl+2
  1903. @d w_cl=w0_cl+3
  1904. @d x_cl=w0_cl+4
  1905. @d y0_cl=10
  1906. @d z0_cl=y0_cl+1
  1907. @d down_cl=y0_cl+2
  1908. @d y_cl=y0_cl+3
  1909. @d z_cl=y0_cl+4
  1910. @d fnt_cl=15
  1911. @d fnt_def_cl=16
  1912. @d invalid_cl=17
  1913. @d max_cl=invalid_cl {largest possible value}
  1914. @<Types...@>=
  1915. @!cmd_cl=char_cl..max_cl;
  1916. @ Here we declare the array |dvi_cl|.
  1917. @<Globals...@>=
  1918. @!dvi_cl:packed array [eight_bits] of cmd_cl;
  1919. @ And here we initialize it.
  1920. @<Set init...@>=
  1921. for i:=set_char_0 to put1+3 do dvi_cl[i]:=char_cl;
  1922. dvi_cl[set_rule]:=rule_cl; dvi_cl[put_rule]:=rule_cl;@/
  1923. dvi_cl[nop]:=invalid_cl;
  1924. dvi_cl[bop]:=invalid_cl; dvi_cl[eop]:=invalid_cl;@/
  1925. dvi_cl[push]:=push_cl; dvi_cl[pop]:=pop_cl;@/
  1926. dvi_cl[w0]:=w0_cl; dvi_cl[x0]:=x0_cl;@/
  1927. dvi_cl[y0]:=y0_cl; dvi_cl[z0]:=z0_cl;@/
  1928. for i:=0 to 3 do
  1929.   begin dvi_cl[i+right1]:=right_cl;
  1930.   dvi_cl[i+w1]:=w_cl;
  1931.   dvi_cl[i+x1]:=x_cl;@/
  1932.   dvi_cl[i+down1]:=down_cl;
  1933.   dvi_cl[i+y1]:=y_cl;
  1934.   dvi_cl[i+z1]:=z_cl;@/
  1935.   dvi_cl[i+xxx1]:=xxx_cl;
  1936.   dvi_cl[i+fnt_def1]:=fnt_def_cl;
  1937.   end;
  1938. for i:=fnt_num_0 to fnt1+3 do dvi_cl[i]:=fnt_cl;
  1939. for i:=pre to 255 do dvi_cl[i]:=invalid_cl;
  1940. @ A few small arrays are used to generate \.{DVI} commands.
  1941. @<Glob...@>=
  1942. @!dvi_char_cmd:array[boolean] of eight_bits; {|put1| and |set1|}
  1943. @!dvi_rule_cmd:array[boolean] of eight_bits; {|put_rule| and |set_rule|}
  1944. @!dvi_right_cmd:array[right_cl..x_cl] of eight_bits; {|right1|, |w1|, and |x1|}
  1945. @!dvi_down_cmd:array[down_cl..z_cl] of eight_bits; {|down1|, |y1|, and |z1|}
  1946. @ @<Set init...@>=
  1947. dvi_char_cmd[false]:=put1;
  1948. dvi_char_cmd[true]:=set1;@/
  1949. dvi_rule_cmd[false]:=put_rule;
  1950. dvi_rule_cmd[true]:=set_rule;@/
  1951. dvi_right_cmd[right_cl]:=right1;
  1952. dvi_right_cmd[w_cl]:=w1;
  1953. dvi_right_cmd[x_cl]:=x1;@/
  1954. dvi_down_cmd[down_cl]:=down1;
  1955. dvi_down_cmd[y_cl]:=y1;
  1956. dvi_down_cmd[z_cl]:=z1;
  1957. @ The global variables |cur_cmd|, |cur_parm| and |cur_class| are used
  1958. for the current \.{DVI} command, its first parameter (if any), and its
  1959. command class respectively.
  1960. @<Glob...@>=
  1961. @!cur_cmd:eight_bits; {current \.{DVI} command byte}
  1962. @!cur_parm:int_32; {its first parameter (if any)}
  1963. @!cur_class:cmd_cl; {its class}
  1964. @ When typesetting a character or rule, the boolean variable |cur_upd|
  1965. is |true| for \\{set} commands, |false| for |put| commands.
  1966. @<Glob...@>=
  1967. @!cur_wp:width_pointer; {width pointer of the current character}
  1968. @!cur_upd:boolean; {is this a \\{set} or |set_rule| command ?}
  1969. @!cur_v_dimen:int_32; {a vertical dimension}
  1970. @!cur_h_dimen:int_32; {a horizontal dimension}
  1971. @ The |dvi_first_par| procedure first reads \.{DVI} command bytes into
  1972. |cur_cmd| until |cur_cmd<>nop|; then |cur_parm| is set to the value of
  1973. the first parameter (if any) and |cur_class| to the command class.
  1974. @d set_cur_char(#)== {set up |cur_res|, |cur_ext|, and |cur_upd|}
  1975. begin cur_ext:=0;
  1976. if cur_cmd<set1 then
  1977.   begin cur_res:=cur_cmd; cur_upd:=true
  1978.   end
  1979. else  begin cur_res:=#; cur_upd:=(cur_cmd<put1);
  1980.   Decr(cur_cmd)(dvi_char_cmd[cur_upd]);
  1981.   while cur_cmd>0 do
  1982.     begin if cur_cmd=3 then if cur_res>127 then cur_ext:=-1;
  1983.     cur_ext:=cur_ext*256+cur_res; cur_res:=#; decr(cur_cmd);
  1984.     end;
  1985.   end;
  1986. @p procedure dvi_first_par;
  1987. begin repeat cur_cmd:=dvi_ubyte;
  1988. until cur_cmd<>nop; {skip over |nop|s}
  1989. case dvi_par[cur_cmd] of
  1990. char_par: set_cur_char(dvi_ubyte);
  1991. no_par: do_nothing;
  1992. dim1_par: cur_parm:=dvi_sbyte;
  1993. num1_par: cur_parm:=dvi_ubyte;
  1994. dim2_par: cur_parm:=dvi_spair;
  1995. num2_par: cur_parm:=dvi_upair;
  1996. dim3_par: cur_parm:=dvi_strio;
  1997. num3_par: cur_parm:=dvi_utrio;
  1998. two_cases(dim4_par): cur_parm:=dvi_squad; {|dim4_par| and |num4_par|}
  1999. numu_par: cur_parm:=dvi_uquad;
  2000. rule_par:
  2001.   begin cur_v_dimen:=dvi_squad; cur_h_dimen:=dvi_squad;
  2002.   cur_upd:=(cur_cmd=set_rule);
  2003.   end;
  2004. fnt_par:cur_parm:=cur_cmd-fnt_num_0;
  2005. end; {there are no other cases}
  2006. cur_class:=dvi_cl[cur_cmd];
  2007. @ The global variable |dvi_nf| is used for the number of different
  2008. \.{DVI} fonts defined so far; their external font numbers (as extracted
  2009. from the \.{DVI} file) are stored in the array |dvi_e_fnts|, the
  2010. corresponding internal font numbers used internally by \.{\title} are
  2011. stored in the array |dvi_i_fnts|.
  2012. @<Glob...@>=
  2013. @!dvi_e_fnts:array[font_number] of int_32; {external font numbers}
  2014. @!dvi_i_fnts:array[font_number] of font_number; {corresponding
  2015.   internal font numbers}
  2016. @!dvi_nf:font_number; {number of \.{DVI} fonts defined so far}
  2017. @ @<Set ini...@>=
  2018. dvi_nf:=0;
  2019. @ The |dvi_font| procedure sets |cur_fnt| to the internal font number
  2020. corresponding to the external font number |cur_parm| (or aborts the
  2021. program if such a font was never defined).
  2022. @p procedure dvi_font; {computes |cur_fnt| corresponding to |cur_parm|}
  2023. var f:font_number; {where the font is sought}
  2024. begin @<DVI: Locate font |cur_parm|@>;
  2025. if f=dvi_nf then bad_dvi;
  2026. cur_fnt:=dvi_i_fnts[f];
  2027. @ @<DVI: Locate font |cur_parm|@>=
  2028. f:=0; dvi_e_fnts[dvi_nf]:=cur_parm;
  2029. while cur_parm<>dvi_e_fnts[f] do incr(f)
  2030. @ Finally the |dvi_do_font| procedure is called when one of the command
  2031. |fnt_def1..fnt_def4| and its first parameter have been read from the
  2032. \.{DVI} file; the argument indicates whether this should be the second
  2033. definition of the font (|true|) or not (|false|).
  2034. @p procedure dvi_do_font(@!second:boolean);
  2035. var f:font_number; {where the font is sought}
  2036. @!k:int_15; {general purpose variable}
  2037. begin print('DVI: font ',cur_parm:1);
  2038. @<DVI: Locate font |cur_parm|@>;
  2039. if (f=dvi_nf)=second then bad_dvi;
  2040. font_check(nf):=dvi_squad;
  2041. font_scaled(nf):=dvi_pquad;
  2042. font_design(nf):=dvi_pquad;
  2043. k:=dvi_ubyte; pckt_room(1); append_byte(k);
  2044. Incr(k)(dvi_ubyte); pckt_room(k);
  2045. while k>0 do  begin append_byte(dvi_ubyte); decr(k);
  2046.   end;
  2047. font_name(nf):=make_packet; {the font area plus name}
  2048. dvi_i_fnts[dvi_nf]:=make_font;
  2049. if not second then
  2050.   begin if dvi_nf=max_fonts then overflow(str_fonts,max_fonts);
  2051.   incr(dvi_nf);
  2052.   end
  2053. else if dvi_i_fnts[f]<>dvi_i_fnts[dvi_nf] then bad_dvi;
  2054. @* Low-level VF input routines.
  2055. A detailed description of the \.{VF} file format can be found in the
  2056. documentation of \.{VFtoVP}; here we just define symbolic names for
  2057. some of the \.{VF} command bytes.
  2058. @d long_char=242 {\.{VF} command for general character packet}
  2059. @d vf_id=202 {identifies \.{VF} files}
  2060. @ The program uses the binary file variable |vf_file| for input from
  2061. \.{VF} files; |vf_loc| is the number of the byte about to be read next
  2062. from |vf_file|.
  2063. @<Glob...@>=
  2064. @!vf_file:byte_file; {a \.{VF} file}
  2065. @!vf_loc:int_32; {where we are about to look, in |vf_file|}
  2066. @!vf_limit:int_32; {value of |vf_loc| at end of a character packet}
  2067. @!vf_ext:pckt_pointer; {extension for \.{VF} files}
  2068. @!vf_cur_fnt:font_number; {current font number in a \.{VF} file}
  2069. @ @<Initialize predefined strings@>=
  2070. id3(".")("V")("F")(vf_ext); {file name extension for \.{VF} files}
  2071. @ If a \.{VF} file is badly malformed, we say |bad_font|; this procedure
  2072. gives an error message which refers the user to \.{VFtoVP} and \.{VPtoVF},
  2073. and terminates \.{\title}.
  2074. @<Cases for |bad_font|@>=
  2075. vf_font_type: begin print('Bad VF file'); print_font(cur_fnt);
  2076. @.Bad VF file@>
  2077.   print_ln(' loc=',vf_loc:1);
  2078.   abort('Use VFtoVP/VPtoVF to diagnose and correct the problem');
  2079. @.Use VFtoVP/VPtoVF@>
  2080.   end;
  2081. @ If no font directory has been specified, \.{\title} is supposed to use
  2082. the default \.{VF} directory, which is a system-dependent place where
  2083. the \.{VF} files for standard fonts are kept.
  2084. The string variable |VF_default_area| contains the name of this area.
  2085. @^system dependencies@>
  2086. @d VF_default_area_name=='TeXvfonts:' {change this to the correct name}
  2087. @d VF_default_area_name_length=10 {change this to the correct length}
  2088. @<Glob...@>=
  2089. @!VF_default_area:packed array[1..VF_default_area_name_length] of char;
  2090. @ @<Set init...@>=
  2091. VF_default_area:=VF_default_area_name;
  2092. @ To prepare |vf_file| for input we |reset| it.
  2093. @<VF: Open |vf_file| or |goto not_found|@>=
  2094. make_font_name(VF_default_area_name_length)(VF_default_area)(vf_ext);
  2095. reset(vf_file,cur_name);
  2096. if eof(vf_file) then
  2097. @^system dependencies@>
  2098.   goto not_found;
  2099. vf_loc:=0
  2100. @ Reading a \.{VF} file should be done as efficient as possible for a
  2101. particular system; on many systems this means that a large number of
  2102. bytes from |vf_file| is read into a buffer and will then be extracted
  2103. from that buffer. In order to simplify such system dependent changes
  2104. we use a pair of \.{WEB} macros: |vf_byte| extracts the next \.{VF}
  2105. byte and |vf_eof| is |true| if we have reached the end of the \.{VF}
  2106. file. Here we give simple minded definitions for these macros in terms
  2107. of standard \PASCAL.
  2108. @^system dependencies@>
  2109. @^optimization@>
  2110. @d vf_eof == eof(vf_file) {has the \.{VF} file been exhausted?}
  2111. @d vf_byte(#) ==
  2112.   if vf_eof then bad_font
  2113.   else read(vf_file,#) {obtain next \.{VF} byte}
  2114. @ We need several simple functions to read the next byte or bytes
  2115. from |vf_file|.
  2116. @p function vf_ubyte:int_8u; {returns the next byte, unsigned}
  2117. @!begin_byte(vf_byte); incr(vf_loc); comp_ubyte(vf_ubyte);
  2118. function vf_upair:int_16u; {returns the next two bytes, unsigned}
  2119. @!begin_pair(vf_byte); Incr(vf_loc)(2); comp_upair(vf_upair);
  2120. function vf_strio:int_24; {returns the next three bytes, signed}
  2121. @!begin_trio(vf_byte); Incr(vf_loc)(3); comp_strio(vf_strio);
  2122. function vf_utrio:int_24u; {returns the next three bytes, unsigned}
  2123. @!begin_trio(vf_byte); Incr(vf_loc)(3); comp_utrio(vf_utrio);
  2124. function vf_squad:int_32; {returns the next four bytes, signed}
  2125. @!begin_quad(vf_byte); Incr(vf_loc)(4); comp_squad(vf_squad);
  2126. @ All dimensions in a \.{VF} file, except the design sizes of a virtual
  2127. font and its local fonts, are |fix_word|s that must be scaled in exactly
  2128. the same way as the character widths from a \.{TFM} file; we can use the
  2129. same code, but this time |z|, |alpha|, and |beta| are global variables.
  2130. @<Glob...@>=
  2131. @<Variables for scaling computation@>@;
  2132. @ We need five functions to read the next byte or bytes and convert a
  2133. |fix_word| to a scaled dimension.
  2134. @p function vf_fix1:int_32; {returns the next byte as scaled value}
  2135. var x:int_32; {accumulator}
  2136. begin vf_byte(tfm_b3); incr(vf_loc);
  2137. tfm_fix1(x); vf_fix1:=x;
  2138. function vf_fix2:int_32; {returns the next two bytes as scaled value}
  2139. var x:int_32; {accumulator}
  2140. begin vf_byte(tfm_b2); vf_byte(tfm_b3); Incr(vf_loc)(2);
  2141. tfm_fix2(x); vf_fix2:=x;
  2142. function vf_fix3:int_32; {returns the next three bytes as scaled value}
  2143. var x:int_32; {accumulator}
  2144. begin vf_byte(tfm_b1); vf_byte(tfm_b2); vf_byte(tfm_b3);
  2145. Incr(vf_loc)(3);@/
  2146. tfm_fix3(x); vf_fix3:=x;
  2147. function vf_fix3u:int_32; {returns the next three bytes as scaled value}
  2148. begin vf_byte(tfm_b1); vf_byte(tfm_b2); vf_byte(tfm_b3);
  2149. Incr(vf_loc)(3);@/
  2150. vf_fix3u:=tfm_fix3u;
  2151. function vf_fix4:int_32; {returns the next four bytes as scaled value}
  2152. var x:int_32; {accumulator}
  2153. begin vf_byte(tfm_b0); vf_byte(tfm_b1); vf_byte(tfm_b2); vf_byte(tfm_b3);
  2154. Incr(vf_loc)(4);@/
  2155. tfm_fix4(x); vf_fix4:=x;
  2156. @ Three other functions are used in cases where the result must have a
  2157. non-negative value or a positive value.
  2158. @p function vf_uquad:int_31; {result must be non-negative}
  2159. var x:int_32;
  2160. begin x:=vf_squad; if x<0 then bad_font @+ else vf_uquad:=x;
  2161. function vf_pquad:int_31; {result must be positive}
  2162. var x:int_32;
  2163. begin x:=vf_squad; if x<=0 then bad_font @+ else vf_pquad:=x;
  2164. function vf_fixp:int_31; {result must be positive}
  2165. var x:int_32; {accumulator}
  2166. begin vf_byte(tfm_b0); vf_byte(tfm_b1); vf_byte(tfm_b2); vf_byte(tfm_b3);
  2167. Incr(vf_loc)(4);@/
  2168. if tfm_b0>0 then bad_font;
  2169. vf_fixp:=tfm_fix3u;
  2170. @ The |vf_first_par| procedure first reads a \.{VF} command byte into
  2171. |cur_cmd|; then |cur_parm| is set to the value of the first parameter
  2172. (if any) and |cur_class| to the command class.
  2173. @d set_cur_wp_end(#)== if cur_wp=invalid_width then #
  2174. @d set_cur_wp(#)== {set |cur_wp| to the char's width pointer}
  2175. cur_wp:=invalid_width;
  2176. if #<>invalid_font then
  2177.   if (cur_res>=font_bc(#))and(cur_res<=font_ec(#)) then
  2178.     cur_wp:=font_width(#)(cur_res);
  2179. set_cur_wp_end
  2180. @p procedure vf_first_par;
  2181. begin cur_cmd:=vf_ubyte;
  2182. case dvi_par[cur_cmd] of
  2183. char_par:
  2184.   begin set_cur_char(vf_ubyte); set_cur_wp(vf_cur_fnt)(bad_font);
  2185.   end;
  2186. no_par: do_nothing;
  2187. dim1_par: cur_parm:=vf_fix1;
  2188. num1_par: cur_parm:=vf_ubyte;
  2189. dim2_par: cur_parm:=vf_fix2;
  2190. num2_par: cur_parm:=vf_upair;
  2191. dim3_par: cur_parm:=vf_fix3;
  2192. num3_par: cur_parm:=vf_utrio;
  2193. dim4_par: cur_parm:=vf_fix4;
  2194. num4_par: cur_parm:=vf_squad;
  2195. numu_par: cur_parm:=vf_uquad;
  2196. rule_par:
  2197.   begin cur_v_dimen:=vf_fix4; cur_h_dimen:=vf_fix4;
  2198.   cur_upd:=(cur_cmd=set_rule);
  2199.   end;
  2200. fnt_par:cur_parm:=cur_cmd-fnt_num_0;
  2201. end; {there are no other cases}
  2202. cur_class:=dvi_cl[cur_cmd];
  2203. @ For a virtual font we set |font_type(f):=vf_font_type|; in this case
  2204. |font_font(f)| is the default font for character packets from virtual
  2205. font~|f|.
  2206. @^font types@>
  2207. The global variable |vf_nf| is used for the number of different local
  2208. fonts defined in a \.{VF} file so far; their external font numbers (as
  2209. extracted from the \.{VF} file) are stored in the array |vf_e_fnts|, the
  2210. corresponding internal font numbers used internally by \.{\title} are
  2211. stored in the array |vf_i_fnts|.
  2212. @<Glob...@>=
  2213. @!vf_e_fnts:array[font_number] of int_32; {external font numbers}
  2214. @!vf_i_fnts:array[font_number] of font_number; {corresponding
  2215.   internal font numbers}
  2216. @!vf_nf:font_number; {number of local fonts defined so far}
  2217. @!lcl_nf:font_number; {largest |vf_nf| value for any \.{VF} file}
  2218. @ @<Set init...@>=
  2219. lcl_nf:=0;
  2220. @ The |vf_font| procedure sets |vf_cur_fnt| to the internal font number
  2221. corresponding to the external font number |cur_parm| (or aborts the
  2222. program if such a font was never defined).
  2223. @p procedure vf_font; {computes |vf_cur_fnt| corresponding to |cur_parm|}
  2224. var f:font_number; {where the font is sought}
  2225. begin @<VF: Locate font |cur_parm|@>;
  2226. if f=vf_nf then bad_font;
  2227. vf_cur_fnt:=vf_i_fnts[f];
  2228. @ @<VF: Locate font |cur_parm|@>=
  2229. f:=0; vf_e_fnts[vf_nf]:=cur_parm;
  2230. while cur_parm<>vf_e_fnts[f] do incr(f)
  2231. @ Finally the |vf_do_font| procedure is called when one of the command
  2232. |fnt_def1..fnt_def4| and its first parameter have been read from the
  2233. \.{VF} file.
  2234. @p procedure vf_do_font;
  2235. var f:font_number; {where the font is sought}
  2236. @!k:int_15; {general purpose variable}
  2237. begin print('VF: font ',cur_parm:1);@/
  2238. @<VF: Locate font |cur_parm|@>;
  2239. if f<>vf_nf then bad_font;
  2240. font_check(nf):=vf_squad;
  2241. font_scaled(nf):=vf_fixp;
  2242. font_design(nf):=round(tfm_conv*vf_pquad);
  2243. k:=vf_ubyte; pckt_room(1); append_byte(k);
  2244. Incr(k)(vf_ubyte); pckt_room(k);
  2245. while k>0 do  begin append_byte(vf_ubyte); decr(k);
  2246.   end;
  2247. font_name(nf):=make_packet; {the font area plus name}
  2248. vf_i_fnts[vf_nf]:=make_font;
  2249. if vf_nf=lcl_nf then
  2250.   if lcl_nf=max_fonts then overflow(str_fonts,max_fonts)
  2251.   else incr(lcl_nf);
  2252. incr(vf_nf);
  2253. @* Reading VF files.
  2254. The |do_vf| function attempts to read the \.{VF} file for a font and
  2255. returns |false| if the \.{VF} file could not be found; when the \.{VF}
  2256. file has been read, the font type is changed to |vf_font_type|.
  2257. @p function do_vf:boolean; {read a \.{VF} file}
  2258. label reswitch,done,not_found,exit;
  2259. var temp_int:int_32; {integer for temporary variables}
  2260. @!temp_byte:int_8u; {byte for temporary variables}
  2261. @!k:byte_pointer; {index into |byte_mem|}
  2262. @!l:int_15; {general purpose variable}
  2263. @!save_ext:int_24; {used to save |cur_ext|}
  2264. @!save_res:int_8u; {used to save |cur_res|}
  2265. @!save_wp:width_pointer; {used to save |cur_wp|}
  2266. @!save_upd:boolean; {used to save |cur_upd|}
  2267. @!vf_wp:width_pointer; {width pointer for the current character packet}
  2268. @!vf_fnt:font_number; {current font in the current character packet}
  2269. @!move_zero:boolean; {|true| if rule 1 is used}
  2270. @!last_pop:boolean; {|true| if final |pop| has been manufactured}
  2271. begin @<VF: Open |vf_file| or |goto not_found|@>;
  2272. save_ext:=cur_ext; save_res:=cur_res; save_wp:=cur_wp;
  2273. save_upd:=cur_upd; {save}
  2274. font_type(cur_fnt):=vf_font_type;@/
  2275. @<VF: Process the preamble@>;@/
  2276. @<VF: Process the font definitions@>;@/
  2277. while cur_cmd<=long_char do @<VF: Build a character packet@>;
  2278. if cur_cmd<>post then bad_font;
  2279. @!debug print('VF file for font ',cur_fnt:1); print_font(cur_fnt);
  2280. print_ln(' loaded.');
  2281. gubed @;@/
  2282. close_in(vf_file);
  2283. cur_ext:=save_ext; cur_res:=save_res; cur_wp:=save_wp;
  2284. cur_upd:=save_upd; {restore}
  2285. do_vf:=true; return;
  2286. not_found:do_vf:=false;
  2287. exit:end;
  2288. @ @<VF: Process the preamble@>=
  2289. if vf_ubyte<>pre then bad_font;
  2290. if vf_ubyte<>vf_id then bad_font;
  2291. temp_byte:=vf_ubyte; pckt_room(temp_byte);
  2292. for l:=1 to temp_byte do append_byte(vf_ubyte);
  2293. print('VF file: '''); print_packet(new_packet); print(''',');
  2294. flush_packet;@/
  2295. check_check_sum(vf_squad,false);
  2296. check_design_size(round(tfm_conv*vf_pquad));@/
  2297. z:=font_scaled(cur_fnt);
  2298. @<Replace |z|...@>;@/
  2299. print_nl('   for font ',cur_fnt:1); print_font(cur_fnt); print_ln('.')
  2300. @ @<VF: Process the font definitions@>=
  2301. vf_i_fnts[0]:=invalid_font; vf_nf:=0;@/
  2302. cur_cmd:=vf_ubyte;
  2303. while (cur_cmd>=fnt_def1)and(cur_cmd<=fnt_def1+3) do
  2304.   begin case cur_cmd-fnt_def1 of
  2305.   0: cur_parm:=vf_ubyte;
  2306.   1: cur_parm:=vf_upair;
  2307.   2: cur_parm:=vf_utrio;
  2308.   3: cur_parm:=vf_squad;
  2309.   end; {there are no other cases}
  2310.   vf_do_font;
  2311.   cur_cmd:=vf_ubyte;
  2312.   end;
  2313. font_font(cur_fnt):=vf_i_fnts[0]
  2314. @ The \.{VF} format specifies that the interpretation of each packet
  2315. begins with |w=x=y=z=0|; any |w0|, |x0|, |y0|, or |z0| command using
  2316. these initial values will be ignored.
  2317. @<Types...@>=
  2318. @!vf_state=array[0..1,0..1] of boolean; {state of |w|, |x|, |y|, and |z|}
  2319. @ As implied by the \.{VF} format the \.{DVI} commands read from the
  2320. \.{VF} file are enclosed by |push| and |pop|; as we read \.{DVI}
  2321. commands and append them to |byte_mem|, we perform a set of
  2322. transformations in order to simplify the resulting packet: Let |zero| be
  2323. any of the commands |put|, |put_rule|, |fnt_num|, |fnt|, or |xxx| which
  2324. all leave the current position on the page unchanged, let |move| be any
  2325. of the horizontal or vertical movement commands |right1..z4|, and let
  2326. |any| be any sequence of commands containing |push| and |pop| in
  2327. properly nested pairs; whenever possible we apply one of the following
  2328. transformation rules: $$\def\n#1:{\hbox to 3cm{\hfil#1:}}
  2329. \leqalignno{
  2330. \hbox{|push| |zero|}&\RA\hbox{|zero| |push|}&\n1:\cr
  2331. \hbox{|move| |pop|}&\RA\hbox{|pop|}&\n2:\cr
  2332. \hbox{|push| |pop|}&\RA{}&\n3:\cr
  2333. \hbox{|push| |set_char| |pop|}&\RA\hbox{|put|}&\n4a:\cr
  2334. \hbox{|push| \\{set} |pop|}&\RA\hbox{|put|}&\n4b:\cr
  2335. \hbox{|push| |set_rule| |pop|}&\RA\hbox{|put_rule|}&\n4c:\cr
  2336. \hbox{|push| |push| |any| |pop|}&\RA\hbox{|push| |any| |pop| |push|}&\n5:\cr
  2337. \hbox{|push| |any| |pop| |pop|}&\RA\hbox{|any| |pop|}&\n6:\cr
  2338. @ In order to perform these transformations we need a stack which is
  2339. indexed by |vf_ptr|, the number of |push| commands without corresponding
  2340. |pop| in the packet we are building; the |vf_push_loc| array contains
  2341. the locations in |byte_mem| following such |push| commands.
  2342. In view of rule~5 consecutive |push| commands are never stored, the
  2343. |vf_push_num| array is used to count them.
  2344. The |vf_last| array indicates the type of the last non-discardable item:
  2345. a character, a rule, or a group enclosed by |push| and |pop|;
  2346. the |vf_last_end| array points to the ending locations and, if
  2347. |vf_last<>vf_other|, the |vf_last_loc| array points to the starting
  2348. locations of these items.
  2349. @d vf_set=0 {|vf_set=char_cl|, last item is a |set_char| or \\{set}}
  2350. @d vf_rule=1 {|vf_rule=rule_cl|, last item is a |set_rule|}
  2351. @d vf_group=2 {last item is a group enclosed by |push| and |pop|}
  2352. @d vf_put=3 {last item is a |put|}
  2353. @d vf_other=4 {last item (if any) is none of the above}
  2354. @<Types...@>=
  2355. @!vf_type=vf_set..vf_other;
  2356. @ @<Glob...@>=
  2357. @!vf_move: array[stack_pointer] of vf_state; {state of |w|, |x|, |y|, and |z|}
  2358. @!vf_push_loc: array[stack_pointer] of byte_pointer; {end of a |push|}
  2359. @!vf_last_loc: array[stack_pointer] of byte_pointer; {start of an item}
  2360. @!vf_last_end: array[stack_pointer] of byte_pointer; {end of an item}
  2361. @!vf_push_num: array[stack_pointer] of eight_bits; {|push| count}
  2362. @!vf_last: array[stack_pointer] of vf_type; {type of last item}
  2363. @!vf_ptr:stack_pointer; {current number of unfinished groups}
  2364. @!stack_used:stack_pointer; {largest |vf_ptr| or |stack_ptr| value}
  2365. @ We use two small arrays to determine the item type of a character or a
  2366. rule.
  2367. @<Glob...@>=
  2368. @!vf_char_type:array[boolean] of vf_type;
  2369. @!vf_rule_type:array[boolean] of vf_type;
  2370. @ @<Set init...@>=
  2371. vf_move[0][0][0]:=false; vf_move[0][0][1]:=false;
  2372. vf_move[0][1][0]:=false; vf_move[0][1][1]:=false;@/
  2373. stack_used:=0;@/
  2374. vf_char_type[false]:=vf_put; vf_char_type[true]:=vf_set;@/
  2375. vf_rule_type[false]:=vf_other; vf_rule_type[true]:=vf_rule;
  2376. @ Here we read the first bytes of a character packet from the \.{VF}
  2377. file and initialize the packet being built in |byte_mem|; the start of
  2378. the whole packet is stored in |vf_push_loc[0]|. When the character
  2379. packet is finished, a type is be assigned to it: |vf_simple| if the
  2380. packet ends with a character of the correct width, or |vf_complex|
  2381. otherwise. Moreover, if such a packet for a character with
  2382. extension zero consists of just one character with extension zero and
  2383. the same residue, and if there is no previous packet, the whole packet
  2384. is replaced by the empty packet.
  2385. @d vf_simple=0 {the packet ends with a character of the correct width}
  2386. @d vf_complex=vf_simple+1 {otherwise}
  2387. @<VF: Build a character packet@>=
  2388. begin if cur_cmd<long_char then
  2389.   begin vf_limit:=cur_cmd;
  2390.   cur_ext:=0; cur_res:=vf_ubyte; vf_wp:=check_width(vf_fix3u);
  2391.   end
  2392. else  begin vf_limit:=vf_uquad;
  2393.   cur_ext:=vf_strio; cur_res:=vf_ubyte; vf_wp:=check_width(vf_fix4);
  2394.   end;
  2395. Incr(vf_limit)(vf_loc);
  2396. vf_push_loc[0]:=byte_ptr; vf_last_end[0]:=byte_ptr;
  2397. vf_last[0]:=vf_other; vf_ptr:=0;@/
  2398. start_packet(vf_complex);
  2399. @<VF: Append \.{DVI} commands to the character packet@>;@/
  2400. k:=pckt_start[pckt_ptr];
  2401. if vf_last[0]=vf_put then if cur_wp=vf_wp then
  2402.   begin decr(byte_mem[k]); {change |vf_complex| into |vf_simple|}
  2403.   if (byte_mem[k]=bi(0))and@|(vf_push_loc[0]=vf_last_loc[0])and@|
  2404.     (cur_ext=0)and@|(cur_res=pckt_res) then byte_ptr:=k;
  2405.   end;
  2406. build_packet;
  2407. cur_cmd:=vf_ubyte;
  2408. @ For every \.{DVI} command read from the \.{VF} file some action is
  2409. performed; in addition the initial |push| and the final |pop| are
  2410. manufactured here.
  2411. @<VF: Append \.{DVI} commands to the character packet@>=
  2412. vf_cur_fnt:=font_font(cur_fnt); vf_fnt:=vf_cur_fnt;@/
  2413. last_pop:=false; cur_class:=push_cl; {initial |push|}
  2414. loop  begin
  2415. reswitch:case cur_class of
  2416.   three_cases(char_cl): @<VF: Do a |char|, |rule|, or |xxx|@>;
  2417.   push_cl: @<VF: Do a |push|@>;
  2418.   pop_cl: @<VF: Do a |pop|@>;
  2419.   two_cases(w0_cl):
  2420.     if vf_move[vf_ptr][0][cur_class-w0_cl] then append_one(cur_cmd);
  2421.   three_cases(right_cl):
  2422.     begin pckt_signed(dvi_right_cmd[cur_class],cur_parm);
  2423.     if cur_class>=w_cl then vf_move[vf_ptr][0][cur_class-w_cl]:=true;
  2424.     end;
  2425.   two_cases(y0_cl):
  2426.     if vf_move[vf_ptr][1][cur_class-y0_cl] then append_one(cur_cmd);
  2427.   three_cases(down_cl):
  2428.     begin pckt_signed(dvi_down_cmd[cur_class],cur_parm);
  2429.     if cur_class>=y_cl then vf_move[vf_ptr][1][cur_class-y_cl]:=true;
  2430.     end;
  2431.   fnt_cl: vf_font;
  2432.   fnt_def_cl: bad_font;
  2433.   invalid_cl: if cur_cmd<>nop then bad_font;
  2434.   end; {there are no other cases}
  2435.   if vf_loc<vf_limit then vf_first_par
  2436.   else if last_pop then goto done
  2437.   else  begin cur_class:=pop_cl; last_pop:=true; {final |pop|}
  2438.     end;
  2439.   end;
  2440. done:if (vf_ptr<>0)or(vf_loc<>vf_limit) then bad_font
  2441. @ For a |push| we either increase |vf_push_num| or start a new level and
  2442. append a |push|.
  2443. @d incr_stack(#)==
  2444. if #=stack_used then
  2445.   if stack_used=stack_size then overflow(str_stack,stack_size)
  2446.   else incr(stack_used);
  2447. incr(#)
  2448. @<VF: Do a |push|@>=
  2449. if (vf_ptr>0)and(vf_push_loc[vf_ptr]=byte_ptr) then
  2450.   begin if vf_push_num[vf_ptr]=255 then overflow(str_stack,255);
  2451.   incr(vf_push_num[vf_ptr]);
  2452.   end
  2453. else  begin incr_stack(vf_ptr);
  2454.   @<VF: Start a new level@>;
  2455.   vf_push_num[vf_ptr]:=0;
  2456.   end
  2457. @ @<VF: Start a new level@>=
  2458. append_one(push);
  2459. vf_move[vf_ptr]:=vf_move[vf_ptr-1];
  2460. vf_push_loc[vf_ptr]:=byte_ptr;
  2461. vf_last_end[vf_ptr]:=byte_ptr;
  2462. vf_last[vf_ptr]:=vf_other
  2463. @ When a character, a rule, or an |xxx| is appended, transformation
  2464. rule~1 might be applicable.
  2465. @<VF: Do a |char|, |rule|, or |xxx|@>=
  2466. begin if (vf_ptr=0)or(byte_ptr>vf_push_loc[vf_ptr]) then move_zero:=false
  2467. else case cur_class of
  2468. char_cl: move_zero:=(not cur_upd)or(vf_cur_fnt<>vf_fnt);
  2469. rule_cl: move_zero:=not cur_upd;
  2470. xxx_cl: move_zero:=true;
  2471. end; {there are no other cases}
  2472. if move_zero then  begin decr(byte_ptr); decr(vf_ptr);
  2473.   end;
  2474. case cur_class of
  2475. char_cl: @<VF: Do a |fnt|, a |char|, or both@>;
  2476. rule_cl: @<VF: Do a |rule|@>;
  2477. xxx_cl: @<VF: Do an |xxx|@>;
  2478. end; {there are no other cases}
  2479. vf_last_end[vf_ptr]:=byte_ptr;
  2480. if move_zero then
  2481.   begin incr(vf_ptr); append_one(push); vf_push_loc[vf_ptr]:=byte_ptr;
  2482.   vf_last_end[vf_ptr]:=byte_ptr;
  2483.   if cur_class=char_cl then if cur_upd then goto reswitch;
  2484.   end;
  2485. @ A special situation arises if transformation rule~1 is applied to a
  2486. |fnt_num| of |fnt| command, but not to the |set_char| or \\{set} command
  2487. following it; in this case |cur_upd| and |move_zero| are both |true| and
  2488. the |set_char| or \\{set} command will be appended later.
  2489. @<VF: Do a |fnt|, a |char|, or both@>=
  2490. begin if vf_cur_fnt<>vf_fnt then
  2491.   begin vf_last[vf_ptr]:=vf_other;
  2492.   pckt_unsigned(fnt1,vf_cur_fnt); vf_fnt:=vf_cur_fnt;
  2493.   end;
  2494. if (not move_zero)or(not cur_upd) then
  2495.   begin vf_last[vf_ptr]:=vf_char_type[cur_upd];
  2496.   vf_last_loc[vf_ptr]:=byte_ptr;
  2497.   pckt_char(cur_upd,cur_ext,cur_res);
  2498.   end;
  2499. @ @<VF: Do a |rule|@>=
  2500. begin vf_last[vf_ptr]:=vf_rule_type[cur_upd];
  2501. vf_last_loc[vf_ptr]:=byte_ptr;
  2502. append_one(dvi_rule_cmd[cur_upd]);
  2503. pckt_four(cur_v_dimen); pckt_four(cur_h_dimen);
  2504. @ @<VF: Do an |xxx|@>=
  2505. begin vf_last[vf_ptr]:=vf_other;
  2506. pckt_unsigned(xxx1,cur_parm); pckt_room(cur_parm);
  2507. while cur_parm>0 do
  2508.   begin append_byte(vf_ubyte); decr(cur_parm);
  2509.   end;
  2510. @ Transformation rules 2--6 are triggered by a |pop|, either read from
  2511. the \.{VF} file or manufactured at the end of the packet.
  2512. @<VF: Do a |pop|@>=
  2513. begin if vf_ptr<1 then bad_font;
  2514. byte_ptr:=vf_last_end[vf_ptr]; {this is rule 2}
  2515. if vf_last[vf_ptr]<=vf_rule then
  2516.  if vf_last_loc[vf_ptr]=vf_push_loc[vf_ptr] then
  2517.   @<VF: Prepare for rule 4@>;
  2518. if byte_ptr=vf_push_loc[vf_ptr] then @<VF: Apply rule 3 or 4@>
  2519. else  begin if vf_last[vf_ptr]=vf_group then @<VF: Apply rule 6@>;
  2520.   append_one(pop); decr(vf_ptr); vf_last[vf_ptr]:=vf_group;
  2521.   vf_last_loc[vf_ptr]:=vf_push_loc[vf_ptr+1]-1;
  2522.   vf_last_end[vf_ptr]:=byte_ptr;
  2523.   if vf_push_num[vf_ptr+1]>0 then @<VF: Apply rule 5@>;
  2524.   end;
  2525. @ In order to implement transformation rule~4, we cancel the |set_char|,
  2526. \\{set}, or |set_rule|, append a |pop|, and insert a |put| or |put_rule|
  2527. with the old parameters.
  2528. @<VF: Prepare for rule 4@>=
  2529. begin cur_class:=vf_last[vf_ptr]; cur_upd:=false;
  2530. byte_ptr:=vf_push_loc[vf_ptr];
  2531. @ @<VF: Apply rule 3 or 4@>=
  2532. begin if vf_push_num[vf_ptr]>0 then
  2533.   begin decr(vf_push_num[vf_ptr]);
  2534.   vf_move[vf_ptr]:=vf_move[vf_ptr-1];
  2535.   end
  2536. else  begin decr(byte_ptr); decr(vf_ptr);
  2537.   end;
  2538. if cur_class<>pop_cl then goto reswitch; {this is rule 4}
  2539. @ @<VF: Apply rule 6@>=
  2540. begin Decr(byte_ptr)(2);
  2541. for k:=vf_last_loc[vf_ptr]+1 to byte_ptr do byte_mem[k-1]:=byte_mem[k];
  2542. vf_last[vf_ptr]:=vf_other; vf_last_end[vf_ptr]:=byte_ptr;
  2543. @ @<VF: Apply rule 5@>=
  2544. begin incr(vf_ptr);
  2545. @<VF: Start a new level@>;
  2546. decr(vf_push_num[vf_ptr]);
  2547. @ The \.{VF} format specifies that after a character packet invoked by a
  2548. |set_char| or \\{set} command, ``|h|~is increased by the \.{TFM} width
  2549. (properly scaled)---just as if a simple character had been typeset'';
  2550. for |vf_simple| packets this is achieved by changing the final |put|
  2551. command into |set_char| or \\{set}, but for |vf_complex| packets an
  2552. explicit movement must be done. This poses a problem for programs,
  2553. such as \.{DVIcopy}, which write a new \.{DVI} file with all references
  2554. to characters from virtual fonts replaced by their character packets:
  2555. The \.{DVItype} program specifies that the horizontal movements after a
  2556. |set_char| or \\{set} command, after a |set_rule| command, and after one
  2557. of the commands |right1..x4|, are all treated differently when \.{DVI}
  2558. units are converted to pixels.
  2559. Thus we introduce a slight extension of \.{DVItype}'s pixel rounding
  2560. algorithm and hope that this extension will become part of the standard
  2561. \.{DVItype} program in the near future: If a \.{DVI} file contains a
  2562. |set_rule| command for a rule with the negative height |width_dimen|,
  2563. then this rule shall be treated in exactly the same way as a ficticious
  2564. character whose width is the width of that rule; as value of |width_dimen|
  2565. we choose $-2^{31}$, the smallest signed 32-bit integer.
  2566. @<Glob...@>=
  2567. @!width_dimen:int_32; {vertical dimension of special rules}
  2568. @ When initializing |width_dimen| we are careful to avoid arithmetic
  2569. overflow.
  2570. @<Set init...@>=
  2571. width_dimen:=-@"40000000; Decr(width_dimen)(@"40000000);
  2572. @* Terminal communication.
  2573. When \.{\title} begins, it engages the user in a brief dialog so that
  2574. various options may be specified. This part of \.{\title} requires
  2575. nonstandard \PASCAL\ constructions to handle the online interaction; so
  2576. it may be preferable in some cases to omit the dialog and simply to
  2577. stick to the default options. On other hand, the system-dependent
  2578. routines that are needed are not complicated, so it will not be terribly
  2579. difficult to introduce them; furthermore they are similar to those in
  2580. \.{DVItype}.
  2581. The |input_ln| routine waits for the user to type a line at his or her
  2582. terminal; then it puts ASCII-code equivalents for the characters on that
  2583. line into the |byte_mem| array as a temporary string. \PASCAL's
  2584. standard |input| file is used for terminal input, as |output| is used
  2585. for terminal output.
  2586. Since the terminal is being used for both input and output, some systems
  2587. need a special routine to make sure that the user can see a prompt message
  2588. before waiting for input based on that message. (Otherwise the message
  2589. may just be sitting in a hidden buffer somewhere, and the user will have
  2590. no idea what the program is waiting for.) We shall invoke a system-dependent
  2591. subroutine |update_terminal| in order to avoid this problem.
  2592. @^system dependencies@>
  2593. @d update_terminal == break(output) {empty the terminal output buffer}
  2594. @d scan_skip== {skip blanks}
  2595.   while (byte_mem[scan_ptr]=bi(" "))and(scan_ptr<byte_ptr) do incr(scan_ptr)
  2596. @d scan_init== {initialize |scan_ptr|}
  2597.   byte_mem[byte_ptr]:=bi(" "); scan_ptr:=pckt_start[pckt_ptr-1]; scan_skip
  2598. @<Action procedures for |dialog|@>=
  2599. procedure input_ln; {inputs a line from the terminal}
  2600. var k:0..terminal_line_length;
  2601. begin print('Enter option: '); update_terminal; reset(input);
  2602. if eoln(input) then read_ln(input);
  2603. k:=0; pckt_room(terminal_line_length);
  2604. while (k<terminal_line_length)and not eoln(input) do
  2605.   begin append_byte(xord[input^]); incr(k); get(input);
  2606.   end;
  2607. @ The global variable |scan_ptr| is used while scanning the temporary
  2608. packet; it points to the next byte in |byte_mem| to be examined.
  2609. @<Glob...@>=
  2610. @!scan_ptr:byte_pointer; {pointer to next byte to be examined}
  2611. @ The |scan_keyword| function is used to test for keywords in a character
  2612. string stored as temporary packet in |byte_mem|; the result is |true|
  2613. (and |scan_ptr| is updated) if the characters starting at position
  2614. |scan_ptr| are an abbreviation of a given keyword followed by at least
  2615. one blank.
  2616. @<Action procedures for |dialog|@>=
  2617. function scan_keyword(@!p:pckt_pointer;@!l:int_7):boolean;
  2618. var i,@!j,@!k:byte_pointer; {indices into |byte_mem|}
  2619. begin i:=pckt_start[p]; j:=pckt_start[p+1]; k:=scan_ptr;
  2620. while (i<j)and((byte_mem[k]=byte_mem[i])or(byte_mem[k]=byte_mem[i]-"a"+"A")) do
  2621.   begin incr(i); incr(k);
  2622.   end;
  2623. if (byte_mem[k]=bi(" "))and(i-pckt_start[p]>=l) then
  2624.   begin scan_ptr:=k; scan_skip; scan_keyword:=true;
  2625.   end
  2626. else scan_keyword:=false;
  2627. @ Here is a routine that scans a (possibly signed) integer and computes
  2628. the decimal value. If no decimal integer starts at |scan_ptr|, the
  2629. value~0 is returned. The integer should be less than $2^{31}$ in
  2630. absolute value.
  2631. @<Action procedures for |dialog|@>=
  2632. function scan_int:int_32;
  2633. var x:int_32; {accumulates the value}
  2634. @!negative:boolean; {should the value be negated?}
  2635. begin if byte_mem[scan_ptr]="-" then
  2636.   begin negative:=true; incr(scan_ptr);
  2637.   end
  2638. else negative:=false;
  2639. x:=0;
  2640. while (byte_mem[scan_ptr]>="0")and(byte_mem[scan_ptr]<="9") do
  2641.   begin x:=10*x+byte_mem[scan_ptr]-"0"; incr(scan_ptr);
  2642.   end;
  2643. scan_skip;
  2644. if negative then scan_int:=-x @+ else scan_int:=x;
  2645. @ The selected options are put into global variables by the |dialog|
  2646. procedure, which is called just as \.{\title} begins.
  2647. @^system dependencies@>
  2648. @p @<Action procedures for |dialog|@>@;
  2649. procedure dialog;
  2650. label exit;
  2651. var p:pckt_pointer; {packet being created}
  2652. begin @<Initialize options@>@;
  2653. loop  begin input_ln; p:=new_packet; scan_init;
  2654.   if scan_ptr=byte_ptr then
  2655.     begin flush_packet; return;
  2656.     end@;@/
  2657.   @<Cases for options@>@;@/
  2658.   else  begin print_ln('Valid options are:'); @<Print valid options@>@;
  2659.     end;
  2660.   flush_packet;
  2661.   end;
  2662. exit:end;
  2663. @* Subroutines for typesetting commands.
  2664. This is the central part of the whole \.{\title} program:
  2665. When a typesetting command from the \.{DVI} file or from a \.{VF} packet
  2666. has been decoded, one of the typesetting routines defined below is
  2667. invoked to execute the command; apart from the necessary book keeping,
  2668. these routines invoke device dependent code defined later.
  2669. @p @<Declare typesetting procedures@>
  2670. @ These typesetting routines communicate with the rest of the program
  2671. through global variables.
  2672. @<Glob...@>=
  2673. @!type_setting:boolean; {|true| while typesetting a page}
  2674. @!device
  2675. @!h_conv:real; {converts \.{DVI} units to horizontal pixels}
  2676. @!v_conv:real; {converts \.{DVI} units to vertical pixels}
  2677. @!h_pixels:pix_value; {a horizontal dimension in pixels}
  2678. @!v_pixels:pix_value; {a vertical dimension in pixels}
  2679. @!temp_pix:pix_value; {temporary value for pixel rounding}
  2680. ecived
  2681. @ @<Set init...@>=
  2682. type_setting:=false;
  2683. @ A stack is used to keep track of the current horizonal and vertical
  2684. position, |h| and |v|, and the four registers |w|, |x|, |y|, and |z|;
  2685. the register pairs |(w,x)| and |(y,z)| are maintained as arrays.
  2686. @<Types...@>=
  2687. @!stack_pointer=0..stack_size;@/
  2688. @!stack_index=1..stack_size;@/
  2689. @!pair_32=array[0..1] of int_32; {a pair of |int_32| variables}
  2690. @!stack_record=record@;@/
  2691.   @!h_field:int_32; {horizontal position |h|}
  2692.   @!v_field:int_32; {vertical position |v|}
  2693.     @!device
  2694.     @!hh_field:pix_value; {horizontal pixel position |hh|}
  2695.     @!vv_field:pix_value; {vertical pixel position |vv|}
  2696.     ecived @; @/
  2697.   @!w_x_field:pair_32; {|w| and |x| register for horizontal movements}
  2698.   @!y_z_field:pair_32; {|y| and |z| register for vertical movements}
  2699.   end;
  2700. @ The current values are kept in |cur_stack|; they are pushed onto and
  2701. popped from |stack|. We use \.{WEB} macros to access the current values.
  2702. @d cur_h==cur_stack.h_field {the current |@!h| value}
  2703. @d cur_v==cur_stack.v_field {the current |@!v| value}
  2704. @d cur_hh==cur_stack.hh_field {the current |@!hh| value}
  2705. @d cur_vv==cur_stack.vv_field {the current |@!vv| value}
  2706. @d cur_w_x==cur_stack.w_x_field {the current |@!w| and |@!x| value}
  2707. @d cur_y_z==cur_stack.y_z_field {the current |@!y| and |@!z| value}
  2708. @<Glob...@>=
  2709. @!stack:array[stack_index] of stack_record; {the pushed values}
  2710. @!cur_stack:stack_record; {the current values}
  2711. @!zero_stack:stack_record; {initial values}
  2712. @!stack_ptr:stack_pointer; {last used position in |stack|}
  2713. @ @<Set init...@>=
  2714. zero_stack.h_field:=0; zero_stack.v_field:=0;
  2715. @!device zero_stack.hh_field:=0; zero_stack.vv_field:=0; @+ ecived @; @/
  2716. for i:=0 to 1 do
  2717.   begin zero_stack.w_x_field[i]:=0; zero_stack.y_z_field[i]:=0;
  2718.   end;
  2719. @ A sequence of consecutive rules, or consecutive characters in a fixed-width
  2720. font whose width is not an integer number of pixels, can cause |hh| to drift
  2721. far away from a correctly rounded value. \.{\title} ensures that the
  2722. amount of drift will never exceed |max_h_drift| pixels; similarly |vv|
  2723. shall never drift away from the correctly rounded value by more than
  2724. |max_v_drift| pixels.
  2725. @d max_h_drift=2 {we insist that abs|(hh-h_pixel_round(h))<=max_drift|}
  2726. @d max_v_drift=2 {we insist that abs|(vv-v_pixel_round(v))<=max_drift|}
  2727. @ The user may select up to |max_select| ranges of consecutive pages to
  2728. be processed. Each starting page specification is recorded in two global
  2729. arrays called |start_count| and |start_there|. For example, `\.{1.*.-5}'
  2730. is represented by |start_there[0]=true|, |start_count[0]=1|,
  2731. |start_there[1]=false|, |start_there[2]=true|, |start_count[2]=-5|. We
  2732. also set |start_vals=2|, to indicate that count 2 was the last one
  2733. mentioned. The other values of |start_count| and |start_there| are not
  2734. important, in this example. The number of pages is recorded in
  2735. |max_pages|; a non positive value indicates that there is no limit.
  2736. @d start_count==select_count[cur_select] {count values to select
  2737.   starting page}
  2738. @d start_there==select_there[cur_select] {is the |start_count| value
  2739.   relevant?}
  2740. @d start_vals==select_vals[cur_select] {the last count considered
  2741.   significant}
  2742. @d max_pages==select_max[cur_select] {at most this many |bop..eop| pages
  2743.   will be printed}
  2744. @<Glob...@>=
  2745. @!select_count:array[0..max_select-1,0..9] of int_32;
  2746. @!select_there:array[0..max_select-1,0..9] of boolean;
  2747. @!select_vals:array[0..max_select-1] of 0..9;
  2748. @!select_max:array[0..max_select-1] of int_32;
  2749. @!out_mag:int_32; {output maginfication}
  2750. @!count:array[0..9] of int_32; {the count values on the current page}
  2751. @!num_select:0..max_select; {number of page selection ranges specified}
  2752. @!cur_select:0..max_select; {current page selection range}
  2753. @!selected:boolean; {has starting page been found?}
  2754. @!all_done:boolean; {have all selected pages been processed?}
  2755. @!str_mag,@!str_select:pckt_pointer;
  2756. @ Here is a simple subroutine that tests if the current page might be the
  2757. starting page.
  2758. @p function start_match:boolean; {does |count| match the starting spec?}
  2759. var k:0..9;  {loop index}
  2760. @!match:boolean; {does everything match so far?}
  2761. begin match:=true;
  2762. for k:=0 to start_vals do
  2763.   if start_there[k]and(start_count[k]<>count[k]) then match:=false;
  2764. start_match:=match;
  2765. @ @<Initialize options@>=
  2766. out_mag:=0; cur_select:=0; max_pages:=0; selected:=true;
  2767. @ @<Print valid options@>=
  2768. print_ln('  mag <mag>');
  2769. print_ln('  select <first page> [<num pages>]');
  2770. @ @<Action procedures for |dialog|@>=
  2771. procedure scan_count; {scan a |start_count| value}
  2772. begin if byte_mem[scan_ptr]=bi("*") then
  2773.   begin start_there[start_vals]:=false; incr(scan_ptr); scan_skip;
  2774.   end
  2775. else  begin start_there[start_vals]:=true;
  2776.   start_count[start_vals]:=scan_int;
  2777.   if cur_select=0 then selected:=false; {don't start at first page}
  2778.   end;
  2779. @ @<Cases for options@>=
  2780. else if scan_keyword(str_mag,3) then out_mag:=scan_int
  2781. else if scan_keyword(str_select,3) then
  2782.   if cur_select=max_select then print_ln('Too many page selections')
  2783.   else  begin start_vals:=0; scan_count;
  2784.     while (start_vals<9)and(byte_mem[scan_ptr]=bi(".")) do
  2785.       begin incr(start_vals); incr(scan_ptr); scan_count;
  2786.       end;
  2787.     max_pages:=scan_int; incr(cur_select);
  2788.     end
  2789. @ @<Initialize predefined strings@>=
  2790. id3("m")("a")("g")(str_mag);
  2791. id6("s")("e")("l")("e")("c")("t")(str_select);
  2792. @ The routines defined below use sections named `Declare local variables
  2793. (if any) for \dots' or `Declare additional local variables for \dots';
  2794. the former may declare variables (including the keyword \&{var}), whereas
  2795. the later must at least contain the keyword \&{var}. In general, both may
  2796. start with the declaration of labels, constants, and\slash or types.
  2797. Let us start with the simple cases:
  2798. The |do_pre| procedure is called when the preamble has been read from
  2799. the \.{DVI} file; the preamble comment has just been converted into a
  2800. temporary packet with the |new_packet| procedure.
  2801. @p procedure do_pre;@/
  2802. @<OUT: Declare local variables (if any) for |do_pre|@>@;
  2803. begin all_done:=false; num_select:=cur_select; cur_select:=0;
  2804. if num_select=0 then max_pages:=0;
  2805. @<OUT: Process the |pre|@>@;@/
  2806. @!device
  2807. h_conv:=(dvi_num/254000.0)*(h_resolution/dvi_den)*(out_mag/1000.0);
  2808. v_conv:=(dvi_num/254000.0)*(v_resolution/dvi_den)*(out_mag/1000.0);
  2809. ecived @; @/
  2810. @ The |do_bop| procedure is called when a |bop| has been read. This
  2811. routine determines whether a page shall be processed or skipped and sets
  2812. the variable |type_setting| accordingly.
  2813. @p procedure do_bop;@/
  2814. @<OUT: Declare additional local variables |do_bop|@>@;
  2815. @!i,@!j:0..9; {indices into |count|}
  2816. begin @<Determine whether this page should be processed or skipped@>;
  2817. print('DVI: ');
  2818. if type_setting then print('process') @+ else print('skipp');
  2819. print('ing page ',count[0]:1); j:=9;
  2820. while (j>0)and(count[j]=0) do decr(j);
  2821. for i:=1 to j do print('.',count[i]:1);
  2822. d_print(' at ',dvi_loc-45:1);
  2823. print_ln('.');
  2824. if type_setting then
  2825.   begin stack_ptr:=0; cur_stack:=zero_stack; cur_fnt:=invalid_font;@/
  2826.   @<OUT: Process a |bop|@>@;@/
  2827.   end;
  2828. @ @<Determine whether this page...@>=
  2829. if not selected then selected:=start_match;
  2830. type_setting:=selected
  2831. @ The |do_eop| procedure is called in order to process an |eop|;
  2832. the stack should be empty.
  2833. @p procedure do_eop;@/
  2834. @<OUT: Declare local variables (if any) for |do_eop|@>@;
  2835. begin if stack_ptr<>0 then bad_dvi;
  2836. @<OUT: Process an |eop|@>@;
  2837. if max_pages>0 then
  2838.   begin decr(max_pages);
  2839.   if max_pages=0 then
  2840.     begin selected:=false; incr(cur_select);
  2841.    if cur_select=num_select then all_done:=true;
  2842.     end;
  2843.   end;
  2844. type_setting:=false;
  2845. @ The procedures |do_push| and |do_pop| are called in order to process
  2846. |push| and |pop| commands; |do_push| must check for stack overflow,
  2847. |do_pop| should never be called when the stack is empty.
  2848. @p procedure do_push; {push onto stack}
  2849. @<OUT: Declare local variables (if any) for |do_push|@>@;
  2850. begin incr_stack(stack_ptr); stack[stack_ptr]:=cur_stack;@/
  2851. @<OUT: Process a |push|@>@;
  2852. procedure do_pop; {pop from stack}
  2853. @<OUT: Declare local variables (if any) for |do_pop|@>@;
  2854. begin if stack_ptr=0 then bad_dvi;
  2855. @<OUT: Process a |pop|@>@;@/
  2856. cur_stack:=stack[stack_ptr]; decr(stack_ptr);
  2857. @ The |do_xxx| procedure is called in order to process a special command.
  2858. The bytes of the special string have been put into |byte_mem| as the
  2859. current string. They are converted to a temporary packet and discarded
  2860. again.
  2861. @p procedure do_xxx;@/
  2862. @<OUT: Declare additional local variables for |do_xxx|@>@;
  2863. @!p:pckt_pointer; {temporary packet}
  2864. begin p:=new_packet;@/
  2865. @<OUT: Process an |xxx|@>@;@/
  2866. flush_packet;
  2867. @ Next are the movement commands:
  2868. The |do_right| procedure is called in order to process the horizontal
  2869. movement commands |right|, |w|, and |x|.
  2870. @d do_h_pixels(#)== {check for proper horizontal pixel rounding}
  2871. begin Incr(cur_hh)(#); temp_pix:=h_pixel_round(cur_h);
  2872. if abs(temp_pix-cur_hh)>max_h_drift then
  2873.   if temp_pix>cur_hh then cur_hh:=temp_pix-max_h_drift
  2874.   else cur_hh:=temp_pix+max_h_drift;
  2875. @p procedure do_right;@/
  2876. @<OUT: Declare local variables (if any) for |do_right|@>@;
  2877. begin if cur_class>=w_cl then cur_w_x[cur_class-w_cl]:=cur_parm
  2878. else if cur_class<right_cl then cur_parm:=cur_w_x[cur_class-w0_cl];
  2879. @<OUT: Process a |right| or |w| or |x|@>@;@/
  2880. Incr(cur_h)(cur_parm);
  2881. @!device
  2882. if (cur_parm>=font_space(cur_fnt))or(cur_parm<=-4*font_space(cur_fnt)) then
  2883.  cur_hh:=h_pixel_round(cur_h)
  2884. else do_h_pixels(h_pixel_round(cur_parm));
  2885. ecived @; @/
  2886. @<OUT: Move right@>@;
  2887. @ The |do_down| procedure is called in order to process the vertical
  2888. movement commands |down|, |y|, and |z|.
  2889. @d do_v_pixels(#)== {check for proper vertical pixel rounding}
  2890. begin Incr(cur_vv)(#); temp_pix:=v_pixel_round(cur_v);
  2891. if abs(temp_pix-cur_vv)>max_v_drift then
  2892.   if temp_pix>cur_vv then cur_vv:=temp_pix-max_v_drift
  2893.   else cur_vv:=temp_pix+max_v_drift;
  2894. @p procedure do_down;@/
  2895. @<OUT: Declare local variables (if any) for |do_down|@>@;
  2896. begin if cur_class>=y_cl then cur_y_z[cur_class-y_cl]:=cur_parm
  2897. else if cur_class<down_cl then cur_parm:=cur_y_z[cur_class-y0_cl];
  2898. @<OUT: Process a |down| or |y| or |z|@>@;@/
  2899. Incr(cur_v)(cur_parm);
  2900. @!device
  2901. if abs(cur_parm)>=5*font_space(cur_fnt) then cur_vv:=v_pixel_round(cur_v)
  2902. else do_v_pixels(v_pixel_round(cur_parm));
  2903. ecived @; @/
  2904. @<OUT: Move down@>@;
  2905. @ The |do_width| procedure is called in order to increase the current
  2906. horizontal position |cur_h| by |cur_h_dimen| in exactly the same way
  2907. as if a character of width |cur_h_dimen| had been typeset.
  2908. @p procedure do_width;@/
  2909. @<OUT: Declare local variables (if any) for |do_width|@>@;
  2910. begin @<OUT: Typeset a |width|@>@;@/
  2911. Incr(cur_h)(cur_h_dimen);
  2912. @!device do_h_pixels(h_pixels); @+ ecived @/ @;
  2913. @<OUT: Move right@>@;
  2914. @ Finally we have the commands for the typesetting of rules and characters;
  2915. the global variable |cur_upd| is |true| if the horizontal position shall
  2916. be updated (\\{set} commands).
  2917. Here are two other subroutine that we need: They computes the number of
  2918. pixels in the height or width of a rule. Characters and rules will line up
  2919. properly if the sizes are computed precisely as specified here.  (Since
  2920. |h_conv| and |v_conv| are computed with some floating-point roundoff error,
  2921. in a machine-dependent way, format designers who are tailoring something for
  2922. a particular resolution should not plan their measurements to come out to an
  2923. exact integer number of pixels; they should compute things so that the
  2924. rule dimensions are a little less than an integer number of pixels, e.g.,
  2925. 4.99 instead of 5.00.)
  2926. @p @!device
  2927. function h_rule_pixels(x:int_32):pix_value;
  2928.   {computes $\lceil|h_conv|\cdot x\rceil$}
  2929. var n:int_32;
  2930. begin n:=trunc(h_conv*x);
  2931. if n<h_conv*x then h_rule_pixels:=n+1 @+ else h_rule_pixels:=n;
  2932. function v_rule_pixels(x:int_32):pix_value;
  2933.   {computes $\lceil|v_conv|\cdot x\rceil$}
  2934. var n:int_32;
  2935. begin n:=trunc(v_conv*x);
  2936. if n<v_conv*x then v_rule_pixels:=n+1 @+ else v_rule_pixels:=n;
  2937. ecived
  2938. @ The |do_rule| procedure is called in order to typeset a rule.
  2939. @p procedure do_rule;@/
  2940. @<OUT: Declare additional local variables |do_rule|@>@;
  2941. @!visible:boolean;
  2942. begin if (cur_h_dimen>0)and(cur_v_dimen>0) then
  2943.   begin visible:=true;
  2944.   @!device
  2945.   h_pixels:=h_rule_pixels(cur_h_dimen);
  2946.   v_pixels:=v_rule_pixels(cur_v_dimen);
  2947.   ecived @; @/
  2948.   @<OUT: Typeset a visible |rule|@>@;
  2949.   end
  2950. else  begin visible:=false;
  2951.   @<OUT: Typeset an invisible |rule|@>@;
  2952.   end;
  2953. if cur_upd then
  2954.   begin Incr(cur_h)(cur_h_dimen);
  2955.   @!device if not visible then h_pixels:=h_rule_pixels(cur_h_dimen);
  2956.   do_h_pixels(h_pixels);
  2957.   ecived @; @/
  2958.   @<OUT: Move right@>@;
  2959.   end;
  2960. @ Last not least the |do_char| procedure is called in order to typeset
  2961. character~|cur_res| with extension~|cur_ext| from the real font~|cur_fnt|.
  2962. @p procedure do_char;@/
  2963. @<OUT: Declare local variables (if any) for |do_char|@>@;
  2964. begin @<OUT: Typeset a |char|@>@;
  2965. if cur_upd then
  2966.   begin Incr(cur_h)(widths[cur_wp]);
  2967.   @!device do_h_pixels(font_pixel(cur_fnt)(cur_res)); @+ ecived @; @/
  2968.   @<OUT: Move right@>@;
  2969.   end;
  2970. @ If the program terminates abnormally, the following code may be
  2971. invoked in the middle of a page.
  2972. @<Finish output file(s)@>=
  2973. begin if type_setting then @<OUT: Finish incomplete page@>;
  2974. @<OUT: Finish output file(s)@>@;
  2975. @ When the first character of font~|cur_fnt| is about to be typeset,
  2976. the |do_font| procedure is called in order to decide whether this is
  2977. a virtual font or a real font.
  2978. One step in this decision is the attempt to find and read the \.{VF}
  2979. file for this font; other attempts to locate a font file may be performed
  2980. before and after that, depending on the nature of the output device and
  2981. on the structure of the file system at a particular installation.
  2982. In any case |do_font| must change |font_type(cur_fnt)| from |new_font_type|
  2983. to anything else; as a last resort one might use the \.{TFM} width data
  2984. and leave blank spaces in the output.
  2985. @p procedure do_font;@/
  2986. label done;@/
  2987. @<OUT: Declare local variables (if any) for |do_font|@>@;
  2988. begin @<OUT: Look for a font file before trying to read the \.{VF} file;
  2989.   if found |goto done|@>@;@/
  2990. if do_vf then goto done; {try to read the \.{VF} file}
  2991. @<OUT: Look for a font file after trying to read the \.{VF} file@>@;@/
  2992. done:
  2993. @!debug if font_type(cur_fnt)=new_font_type then confusion(str_fonts);
  2994. gubed@;
  2995. @* Interpreting VF packets.
  2996. The |pckt_first_par| procedure first reads a \.{DVI} command byte from
  2997. the packet into |cur_cmd|; then |cur_parm| is set to the value of the
  2998. first parameter (if any) and |cur_class| to the command class.
  2999. @p procedure pckt_first_par;
  3000. begin cur_cmd:=pckt_ubyte;
  3001. case dvi_par[cur_cmd] of
  3002. char_par: set_cur_char(pckt_ubyte);
  3003. no_par: do_nothing;
  3004. dim1_par: cur_parm:=pckt_sbyte;
  3005. num1_par: cur_parm:=pckt_ubyte;
  3006. dim2_par: cur_parm:=pckt_spair;
  3007. num2_par: cur_parm:=pckt_upair;
  3008. dim3_par: cur_parm:=pckt_strio;
  3009. num3_par: cur_parm:=pckt_utrio;
  3010. three_cases(dim4_par): cur_parm:=pckt_squad; {|dim4|, |num4|, or |numu|}
  3011. rule_par:
  3012.   begin cur_v_dimen:=pckt_squad; cur_h_dimen:=pckt_squad;
  3013.   cur_upd:=(cur_cmd=set_rule);
  3014.   end;
  3015. fnt_par:cur_parm:=cur_cmd-fnt_num_0;
  3016. end; {there are no other cases}
  3017. cur_class:=dvi_cl[cur_cmd];
  3018. @ The |do_vf_packet| procedure is called in order to interpret the
  3019. character packet for a virtual character. Such a packet may contain the
  3020. instruction to typeset a character from the same or an other virtual
  3021. font; in such cases |do_vf_packet| calls itself recursively. The
  3022. recursion level, i.e., the number of times this has happened, is kept
  3023. in the global variable |n_recur| and should not exceed |max_recursion|.
  3024. @^recursion@>
  3025. @<Types...@>=
  3026. @!recur_pointer=0..max_recursion;
  3027. @ The \.{\title} processor should detect an infinite recursion caused by
  3028. bad \.{VF} files; thus a new recursion level is entered even in cases
  3029. where this could be avoided without difficulty.
  3030. If the recursion level exceeds the allowed maximum, we want to give
  3031. a traceback how this has happened; thus some of the global variables
  3032. used in different invocations of |do_vf_packet| are saved in a stack,
  3033. others are saved as local variables of |do_vf_packet|.
  3034. @<Glob...@>=
  3035. @!recur_fnt:array[recur_pointer] of font_number; {this packet's font}
  3036. @!recur_ext:array[recur_pointer] of int_24; {this packet's extension}
  3037. @!recur_res:array[recur_pointer] of eight_bits; {this packet's residue}
  3038. @!recur_pckt:array[recur_pointer] of pckt_pointer; {the packet}
  3039. @!recur_loc:array[recur_pointer] of byte_pointer; {next byte of packet}
  3040. @!n_recur:recur_pointer; {current recursion level}
  3041. @!recur_used:recur_pointer; {highest recursion level used so far}
  3042. @ @<Set init...@>=
  3043. n_recur:=0; recur_used:=0;
  3044. @ Here now is the |do_vf_packet| procedure.
  3045. @p procedure do_vf_packet;
  3046. label continue,found,done;
  3047. var k:recur_pointer; {loop index}
  3048. @!f:int_8u; {packet type flag}
  3049. @!save_upd:boolean; {used to save |cur_upd|}
  3050. @!save_wp:width_pointer; {used to save |cur_wp|}
  3051. @!save_limit:byte_pointer; {used to save |cur_limit|}
  3052. begin @<VF: Save values on entry to |do_vf_packet|@>;@/
  3053. @<VF: Interpret the \.{DVI} commands in the packet@>@;@/
  3054. if save_upd then
  3055.   begin cur_h_dimen:=widths[save_wp];
  3056.   @!device h_pixels:=pix_widths[save_wp]; @+ ecived @; @/
  3057.   do_width;
  3058.   end;
  3059. @<VF: Restore values on exit from |do_vf_packet|@>;@/
  3060. @ On entry to |do_vf_packet| several values must be saved.
  3061. @<VF: Save values on entry to |do_vf_packet|@>=
  3062. save_upd:=cur_upd;
  3063. save_wp:=cur_wp;@/
  3064. recur_fnt[n_recur]:=cur_fnt;
  3065. recur_ext[n_recur]:=cur_ext;
  3066. recur_res[n_recur]:=cur_res
  3067. @ Some of these values must be restored on exit from |do_vf_packet|.
  3068. @<VF: Restore values on exit from |do_vf_packet|@>=
  3069. cur_fnt:=recur_fnt[n_recur]
  3070. @ If |cur_pckt| is the empty packet, we manufacture a |put| command;
  3071. otherwise we read and interpret \.{DVI} commands from the packet.
  3072. @<VF: Interpret the \.{DVI} commands in the packet@>=
  3073. if find_packet then f:=cur_type @+ else goto done;
  3074. recur_pckt[n_recur]:=cur_pckt;
  3075. save_limit:=cur_limit;
  3076. cur_fnt:=font_font(cur_fnt);
  3077. if cur_pckt=empty_packet then
  3078.   begin cur_class:=char_cl; goto found;
  3079.   end;
  3080. if cur_loc>=cur_limit then goto done;
  3081. continue: pckt_first_par;
  3082. found: case cur_class of
  3083. char_cl: @<VF: Typeset a |char|@>;
  3084. rule_cl: do_rule;
  3085. xxx_cl:
  3086.   begin pckt_room(cur_parm);
  3087.   while cur_parm>0 do
  3088.     begin append_byte(pckt_ubyte); decr(cur_parm);
  3089.     end;
  3090.   do_xxx;
  3091.   end;
  3092. push_cl: do_push;
  3093. pop_cl: do_pop;
  3094. five_cases(w0_cl): do_right; {|right|, |w|, or |x|}
  3095. five_cases(y0_cl): do_down; {|down|, |y|, or |z|}
  3096. fnt_cl: cur_fnt:=cur_parm;
  3097. othercases confusion(str_packets); {font definition or invalid}
  3098. endcases;
  3099. if cur_loc<cur_limit then goto continue;
  3100. done:
  3101. @ When a font is used for the first time, the |do_font| procedure is
  3102. called to decide whether this is a virtual font or not.
  3103. The final |put| of a simple packet may be changed into |set_char| or
  3104. \\{set}.
  3105. @<VF: Typeset a |char|@>=
  3106. begin cur_wp:=font_width(cur_fnt)(cur_res);
  3107. if font_type(cur_fnt)=new_font_type then do_font; {|cur_fnt| was not yet used}
  3108. if (cur_loc=cur_limit)and(f=vf_simple) and save_upd then
  3109.   begin save_upd:=false; cur_upd:=true;
  3110.   end;
  3111. if font_type(cur_fnt)=vf_font_type then
  3112.   @<VF: Enter a new recursion level@>
  3113. else do_char;
  3114. @ Before entering a new recursion level we must test for overflow; in
  3115. addition a few variables must be saved and restored.
  3116. A |set_char| or \\{set} followed by |pop| is changed into |put|.
  3117. @<VF: Enter a new recursion level@>=
  3118. begin recur_loc[n_recur]:=cur_loc; {save}
  3119. if cur_loc<cur_limit then
  3120.   if byte_mem[cur_loc]=bi(pop) then cur_upd:=false;
  3121. if n_recur=recur_used then
  3122.   if recur_used=max_recursion then
  3123.     @<VF: Display the recursion traceback and terminate@>
  3124.   else incr(recur_used);@/
  3125. incr(n_recur);
  3126. do_vf_packet;
  3127. decr(n_recur); {recurse}
  3128. cur_loc:=recur_loc[n_recur];
  3129. cur_limit:=save_limit; {restore}
  3130. @ @<VF: Display the recursion traceback and terminate@>=
  3131. begin print_ln(' !Infinite VF recursion?');
  3132. @.Infinite VF recursion?@>
  3133. for k:=max_recursion downto 0 do
  3134.   begin print('level=',k:1,' font');
  3135.   d_print('=',recur_fnt[k]:1);
  3136.   print_font(recur_fnt[k]);
  3137.   print(' char=',recur_res[k]:1);
  3138.   if recur_ext[k]<>0 then print('.',recur_ext[k]:1);
  3139.   new_line;
  3140.   @!debug hex_packet(recur_pckt[k]); print_ln('loc=',recur_loc[k]:1);
  3141.   gubed@;
  3142.   end;
  3143. overflow(str_recursion,max_recursion);
  3144. @* Interpreting the DVI file.
  3145. The |do_dvi| procedure reads the entire \.{DVI} file and initiates
  3146. whatever actions may be necessary.
  3147. @p procedure do_dvi;
  3148. label done,exit;
  3149. var temp_byte:int_8u; {byte for temporary variables}
  3150. @!temp_int:int_32; {integer for temporary variables}
  3151. @!dvi_start:int_32; {starting location}
  3152. @!dvi_bop_post:int_32; {location of |bop| or |post|}
  3153. @!dvi_back:int_32; {a back pointer}
  3154. @!k:int_15; {general purpose variable}
  3155. begin @<DVI: Process the preamble@>;
  3156. if random_reading then @<DVI: Process the postamble@>;
  3157. repeat dvi_first_par;
  3158.   while cur_class=fnt_def_cl do
  3159.     begin dvi_do_font(random_reading); dvi_first_par;
  3160.     end;
  3161.   if cur_cmd=bop then @<DVI: Process one page@>;
  3162. until cur_cmd<>eop;
  3163. if cur_cmd<>post then bad_dvi;
  3164. exit:end;
  3165. @ @<DVI: Process the preamble@>=
  3166. if dvi_ubyte<>pre then bad_dvi;
  3167. if dvi_ubyte<>dvi_id then bad_dvi;
  3168. dvi_num:=dvi_pquad; dvi_den:=dvi_pquad; dvi_mag:=dvi_pquad;
  3169. tfm_conv:=(25400000.0/dvi_num)*(dvi_den/473628672)/16.0;
  3170. temp_byte:=dvi_ubyte; pckt_room(temp_byte);
  3171. for k:=1 to temp_byte do append_byte(dvi_ubyte);
  3172. print('DVI file: '''); print_packet(new_packet); print_ln(''',');
  3173. print('   num=',dvi_num:1,', den=',dvi_den:1,', mag=',dvi_mag:1);
  3174. if out_mag<=0 then out_mag:=dvi_mag @+ else print(' => ',out_mag:1);
  3175. print_ln('.');
  3176. do_pre; flush_packet
  3177. @ @<Glob...@>=
  3178. @!dvi_num:int_31; {numerator}
  3179. @!dvi_den:int_31; {denominator}
  3180. @!dvi_mag:int_31; {magnification}
  3181. @ @<DVI: Process the postamble@>=
  3182. begin dvi_start:=dvi_loc; {remember start of first page}
  3183. @<DVI: Find the postamble@>;
  3184. d_print_ln('DVI: postamble at ',dvi_bop_post:1);
  3185. dvi_back:=dvi_pointer;
  3186. if dvi_num<>dvi_pquad then bad_dvi;
  3187. if dvi_den<>dvi_pquad then bad_dvi;
  3188. if dvi_mag<>dvi_pquad then bad_dvi;
  3189. temp_int:=dvi_squad; temp_int:=dvi_squad;
  3190. if stack_size<dvi_upair then overflow(str_stack,stack_size);
  3191. temp_int:=dvi_upair;
  3192. dvi_first_par;
  3193. while cur_class=fnt_def_cl do
  3194.   begin dvi_do_font(false); dvi_first_par;
  3195.   end;
  3196. if cur_cmd<>post_post then bad_dvi;
  3197. if not selected then @<DVI: Find the starting page@>;
  3198. dvi_move(dvi_start); {go to first or starting page}
  3199. @ @<DVI: Find the postamble@>=
  3200. temp_int:=dvi_length-5;
  3201. repeat if temp_int<49 then bad_dvi;
  3202. dvi_move(temp_int); temp_byte:=dvi_ubyte; decr(temp_int);
  3203. until temp_byte<>223;
  3204. if temp_byte<>dvi_id then bad_dvi;
  3205. dvi_move(temp_int-4); if dvi_ubyte<>post_post then bad_dvi;
  3206. dvi_bop_post:=dvi_pointer;
  3207. if (dvi_bop_post<15)or(dvi_bop_post>dvi_loc-34) then bad_dvi;
  3208. dvi_move(dvi_bop_post); if dvi_ubyte<>post then bad_dvi
  3209. @ @<DVI: Find the starting page@>=
  3210. begin dvi_start:=dvi_bop_post; {just in case}
  3211. while dvi_back<>-1 do
  3212.   begin if (dvi_back<15)or(dvi_back>dvi_bop_post-46) then bad_dvi;
  3213.   dvi_bop_post:=dvi_back; dvi_move(dvi_back);
  3214.   if dvi_ubyte<>bop then bad_dvi;
  3215.   for k:=0 to 9 do count[k]:=dvi_squad;
  3216.   if start_match then dvi_start:=dvi_bop_post;
  3217.   dvi_back:=dvi_pointer;
  3218.   end;
  3219. @ When a |bop| has been read, the \.{DVI} commands for one page are
  3220. interpreted until an |eop| is found.
  3221. @<DVI: Process one page@>=
  3222. begin for k:=0 to 9 do count[k]:=dvi_squad;
  3223. temp_int:=dvi_pointer; do_bop;
  3224. dvi_first_par;
  3225. if type_setting then @<DVI: Process a page; then |goto done|@>
  3226. else @<DVI: Skip a page; then |goto done|@>;
  3227. done:if cur_cmd<>eop then bad_dvi;
  3228. if type_setting then
  3229.   begin do_eop;
  3230.   if all_done then return;
  3231.   end;
  3232. @ All \.{DVI} commands are processed, as long as |cur_class<>invalid_cl|;
  3233. then we should have found an |eop|.
  3234. @<DVI: Process a page; then |goto done|@>=
  3235. loop begin
  3236.   case cur_class of
  3237.   char_cl: @<DVI: Typeset a |char|@>;
  3238.   rule_cl:
  3239.     if cur_upd and(cur_v_dimen=width_dimen) then
  3240.       begin @!device h_pixels:=h_pixel_round(cur_h_dimen); @+ ecived @; @/
  3241.       do_width;
  3242.       end
  3243.     else do_rule;
  3244.   xxx_cl:
  3245.     begin pckt_room(cur_parm);
  3246.     while cur_parm>0 do
  3247.       begin append_byte(dvi_ubyte); decr(cur_parm);
  3248.       end;
  3249.     do_xxx;
  3250.     end;
  3251.   push_cl: do_push;
  3252.   pop_cl: do_pop;
  3253.   five_cases(w0_cl): do_right; {|right|, |w|, or |x|}
  3254.   five_cases(y0_cl): do_down; {|down|, |y|, or |z|}
  3255.   fnt_cl: dvi_font;
  3256.   fnt_def_cl: dvi_do_font(random_reading);
  3257.   invalid_cl: goto done;
  3258.   end; {there are no other cases}
  3259. dvi_first_par; {get the next command}
  3260. @ While skipping a page all commands other than font definitions are
  3261. ignored.
  3262. @<DVI: Skip a page; then |goto done|@>=
  3263. loop begin
  3264.   case cur_class of
  3265.   xxx_cl: while cur_parm>0 do
  3266.     begin temp_byte:=dvi_ubyte; decr(cur_parm);
  3267.     end;
  3268.   fnt_def_cl: dvi_do_font(random_reading);
  3269.   invalid_cl: goto done;
  3270.   othercases do_nothing;
  3271.   endcases;
  3272. dvi_first_par; {get the next command}
  3273. @ When a font is used for the first time, the |do_font| procedure is
  3274. called to decide whether this is a virtual font or not.
  3275. @<DVI: Typeset a |char|@>=
  3276. begin set_cur_wp(cur_fnt)(bad_dvi);
  3277. if font_type(cur_fnt)=new_font_type then do_font; {|cur_fnt| was not yet used}
  3278. if font_type(cur_fnt)=vf_font_type then do_vf_packet @+ else do_char;
  3279. @* The main program.
  3280. The code for real devices is still rather incomplete.
  3281. Moreover several branches of the program have not been tested because
  3282. they are never used with \.{DVI} files made by \TeX\ and \.{VF} files
  3283. made by \.{VPtoVF}.
  3284. @ At the end of the program the output file(s) have to be finished and
  3285. on some systems it may be necessary to close input and\slash or output
  3286. files.
  3287. @^system dependencies@>
  3288. @p procedure close_files_and_terminate;
  3289. var k:@!int_15; {general purpose index}
  3290. begin close_in(dvi_file);
  3291. if history<fatal_message then @<Finish output file(s)@>;
  3292. stat @<Print memory usage statistics@>;@+tats@;@/
  3293. @<Close output file(s)@>@;
  3294. @<Print the job |history|@>;
  3295. @ Now we are ready to put it all together.
  3296. Here is where \.{\title} starts, and where it ends.
  3297. @^system dependencies@>
  3298. @p begin initialize; {get all variables initialized}
  3299. @<Initialize predefined strings@>@;
  3300. dialog; {get options}
  3301. @<Open input file(s)@>@;
  3302. @<Open output file(s)@>@;
  3303. do_dvi; {process the entire \.{DVI} file}
  3304. close_files_and_terminate;
  3305. final_end:end.
  3306. @ @<Print memory usage statistics@>=
  3307. print_ln('Memory usage statistics:');
  3308. print(dvi_nf:1,' dvi, ',lcl_nf:1,' local, ');
  3309. @<Print more font usage statistics@>@;@/
  3310. print_ln('and ',nf:1,' internal fonts of ',max_fonts:1);
  3311. print_ln(n_widths:1,' widths of ',max_widths:1,' for ',
  3312.   n_chars:1,' characters of ',max_chars:1);
  3313. print_ln(pckt_ptr:1,' byte packets of ',max_packets:1,' with ',
  3314.   byte_ptr:1,' bytes of ',max_bytes:1);
  3315. @<Print more memory usage statistics@>@;@/
  3316. print_ln(stack_used:1,' of ',stack_size:1,' stack and ',
  3317.   recur_used:1,' of ',max_recursion:1,' recursion levels.')
  3318. @ Some implementations may wish to pass the |history| value to the
  3319. operating system so that it can be used to govern whether or not other
  3320. programs are started. Here we simply report the history to the user.
  3321. @^system dependencies@>
  3322. @<Print the job |history|@>=
  3323. case history of
  3324. spotless: print_ln('(No errors were found.)');
  3325. harmless_message: print_ln('(Did you see the warning message above?)');
  3326. error_message: print_ln('(Pardon me, but I think I spotted something wrong.)');
  3327. fatal_message: print_ln('(That was a fatal error, my friend.)');
  3328. end {there are no other cases}
  3329. @* Low-level output routines.
  3330. The program uses the binary file variable |out_file| for its main output
  3331. file; |out_loc| is the number of the byte about to be written next on
  3332. |out_file|.
  3333. @<Glob...@>=
  3334. @!out_file:byte_file; {the \.{DVI} file we are writing}
  3335. @!out_loc:int_32; {where we are about to write, in |out_file|}
  3336. @!out_back:int_32; {a back pointer}
  3337. @!out_max_v:int_31; {maximum |v| value so far}
  3338. @!out_max_h:int_31; {maximum |h| value so far}
  3339. @!out_stack:int_16u; {maximum stack depth}
  3340. @!out_pages:int_16u; {total number of pages}
  3341. @ @<Set ini...@>=
  3342. out_loc:=0; out_back:=-1;
  3343. out_max_v:=0; out_max_h:=0;
  3344. out_stack:=0; out_pages:=0;
  3345. @ To prepare |out_file| for output, we |rewrite| it.
  3346. @<Open output file(s)@>=
  3347. rewrite(out_file); {prepares to write packed bytes to |out_file|}
  3348. @ For some operating systems it may be necessary to close |out_file|.
  3349. @<Close output file(s)@>=
  3350. @ Writing the |out_file| should be done as efficient as possible for a
  3351. particular system; on many systems this means that a large number of
  3352. bytes will be accumulated in a buffer and is then written from that
  3353. buffer to |out_file|. In order to simplify such system dependent changes
  3354. we use the \.{WEB} macro |out_byte| to write the next \.{DVI} byte. Here
  3355. we give a simple minded definition for this macro in terms of standard
  3356. \PASCAL.
  3357. @^system dependencies@>
  3358. @^optimization@>
  3359. @d out_byte(#) == write(out_file,#) {write next \.{DVI} byte}
  3360. @ The \.{WEB} macro |out_one| is used to write one byte and to update
  3361. |out_loc|.
  3362. @d out_one(#) == begin out_byte(#); incr(out_loc); @+ end
  3363. @ First the |out_packet| procedure copies a packet to |out_file|.
  3364. @<Declare typesetting procedures@>=
  3365. procedure out_packet(@!p:pckt_pointer);
  3366. var k:byte_pointer; {index into |byte_mem|}
  3367. begin Incr(out_loc)(pckt_length(p));
  3368. for k:=pckt_start[p] to pckt_start[p+1]-1 do out_byte(bo(byte_mem[k]));
  3369. @ Next are the procedures used to write integer numbers or even complete
  3370. \.{DVI} commands to |out_file|; they all keep |out_loc| up to date.
  3371. The |out_four| procedure outputs four bytes in two's complement notation,
  3372. without risking arithmetic overflow.
  3373. @<Declare typesetting procedures@>=
  3374. procedure out_four(@!x:int_32); {output four bytes}
  3375. @!begin_four; comp_four(out_byte); Incr(out_loc)(4);
  3376. @ The |out_char| procedure outputs a |set_char| or \\{set} command or, if
  3377. |upd=false|, a |put| command.
  3378. @<Declare typesetting procedures@>=
  3379. procedure out_char(@!upd:boolean;@!ext:int_32;@!res:eight_bits);
  3380.   {output \\{set} or |put|}
  3381. @!begin_char; comp_char(out_one);
  3382. @ The |out_unsigned| procedure outputs a |fnt|, |xxx|, or |fnt_def|
  3383. command with its first parameter (normally unsigned); a |fnt| command
  3384. is converted into |fnt_num| whenever this is possible.
  3385. @<Declare typesetting procedures@>=
  3386. procedure out_unsigned(@!o:eight_bits;@!x:int_32);
  3387.   {output |fnt_num|, |fnt|, |xxx|, or |fnt_def|}
  3388. @!begin_unsigned; comp_unsigned(out_one);
  3389. @ The |out_signed| procedure outputs a movement (|right|, |w|,
  3390. |x|, |down|, |y|, or |z|) command with its (signed) parameter.
  3391. @<Declare typesetting procedures@>=
  3392. procedure out_signed(@!o:eight_bits;@!x:int_32);
  3393.   {output |right|, |w|, |x|, |down|, |y|, or |z|}
  3394. @!begin_signed; comp_signed(out_one);
  3395. @ For an output font we set |font_type(f):=out_font_type|; in this case
  3396. |font_font(f)| is the font number used for font~|f| in |out_file|.
  3397. @^font types@>
  3398. The global variable |out_nf| is the number of fonts already used in
  3399. |out_file| and the array |out_fnts| contains their internal font numbers;
  3400. the current font in |out_file| is called |out_fnt|.
  3401. @<Glob...@>=
  3402. @!out_fnts:array[font_number] of font_number; {internal font numbers}
  3403. @!out_nf:font_number; {number of fonts used in |out_file|}
  3404. @!out_fnt:font_number; {internal font number of current output font}
  3405. @ @<Set init...@>=
  3406. out_nf:=0;
  3407. @ @<Print more font usage statistics@>=
  3408. print(out_nf:1,' out, ');
  3409. @ The |out_fnt_def| procedure outputs a complete font definition
  3410. command.
  3411. @<Declare typesetting procedures@>=
  3412. procedure out_fnt_def(@!f:font_number);
  3413. var p:pckt_pointer; {the font name packet}
  3414. @!k,@!l:byte_pointer; {indices into |byte_mem|}
  3415. @!a:eight_bits; {length of area part}
  3416. begin out_unsigned(fnt_def1,font_font(f)); out_four(font_check(f));
  3417. out_four(font_scaled(f)); out_four(font_design(f));@/
  3418. p:=font_name(f); k:=pckt_start[p]; l:=pckt_start[p+1]-1;
  3419. a:=bo(byte_mem[k]);@/
  3420. Incr(out_loc)(l-k+2); out_byte(a); out_byte(l-k-a);
  3421. while k<l do
  3422.   begin incr(k); out_byte(bo(byte_mem[k]));
  3423.   end;
  3424. @* Writing the output file.
  3425. Here we define the device dependent parts of the typesetting routines
  3426. described earlier in this program.
  3427. The device dependent code for a real output device must define a few constants;
  3428. here we demonstrate how they should be defined.
  3429. @d h_resolution==300 {horizontal resolution in pixels per inch (dpi)}
  3430. @d v_resolution==300 {vertical resolution in pixels per inch (dpi)}
  3431. @ These are the local variables (if any) needed for |do_pre|.
  3432. @<OUT: Declare local variables (if any) for |do_pre|@>=
  3433. var k:int_15; {general purpose variable}
  3434. @!p,@!q,@!r:byte_pointer; {indices into |byte_mem|}
  3435. @!comment:packed array[1..comm_length] of char; {preamble comment prefix}
  3436. @ And here is the device dependent code for |do_pre|; the \.{DVI} preamble
  3437. comment written to |out_file| is similar to the one produced by \.{GFtoPK},
  3438. but we want to apply our preamble comment prefix only once.
  3439. @<OUT: Process the |pre|@>=
  3440. out_one(pre); out_one(dvi_id);
  3441. out_four(dvi_num); out_four(dvi_den); out_four(out_mag);@/
  3442. p:=pckt_start[pckt_ptr-1]; q:=byte_ptr; {location of old \.{DVI} comment}
  3443. comment:=preamble_comment; pckt_room(comm_length);
  3444. for k:=1 to comm_length do append_byte(xord[comment[k]]);
  3445. while byte_mem[p]=bi(" ") do incr(p); {remove leading blanks}
  3446. if p=q then Decr(byte_ptr)(from_length)
  3447. else begin k:=0;
  3448.   while (k<comm_length)and(byte_mem[p+k]=byte_mem[q+k]) do incr(k);
  3449.   if k=comm_length then Incr(p)(comm_length);
  3450.   end;
  3451. k:=byte_ptr-p; {total length}
  3452. if k>255 then
  3453.   begin k:=255; q:=p+255-comm_length; {at most 255 bytes}
  3454.   end;
  3455. out_one(k); out_packet(new_packet); flush_packet;
  3456. for r:=p to q-1 do out_one(bo(byte_mem[r]));
  3457. @ These are the additional local variables (if any) needed for |do_bop|;
  3458. the variables |@!i| and |@!j| are already declared.
  3459. @<OUT: Declare additional local variables |do_bop|@>=
  3460. @ And here is the device dependent code for |do_bop|.
  3461. @<OUT: Process a |bop|@>=
  3462. out_one(bop); incr(out_pages);
  3463. for i:=0 to 9 do out_four(count[i]);
  3464. out_four(out_back); out_back:=out_loc-45;
  3465. out_fnt:=invalid_font;
  3466. @ These are the local variables (if any) needed for |do_eop|.
  3467. @<OUT: Declare local variables (if any) for |do_eop|@>=
  3468. @ And here is the device dependent code for |do_eop|.
  3469. @<OUT: Process an |eop|@>=
  3470. out_one(eop);
  3471. @ These are the local variables (if any) needed for |do_push|.
  3472. @<OUT: Declare local variables (if any) for |do_push|@>=
  3473. @ And here is the device dependent code for |do_push|.
  3474. @<OUT: Process a |push|@>=
  3475. if stack_ptr>out_stack then out_stack:=stack_ptr;
  3476. out_one(push);
  3477. @ These are the local variables (if any) needed for |do_pop|.
  3478. @<OUT: Declare local variables (if any) for |do_pop|@>=
  3479. @ And here is the device dependent code for |do_pop|.
  3480. @<OUT: Process a |pop|@>=
  3481. out_one(pop);
  3482. @ These are the additional local variables (if any) needed for |do_xxx|;
  3483. the variable |@!p|, the pointer to the packet containing the special
  3484. string, is already declared.
  3485. @<OUT: Declare additional local variables for |do_xxx|@>=
  3486. @ And here is the device dependent code for |do_xxx|.
  3487. @<OUT: Process an |xxx|@>=
  3488. out_unsigned(xxx1,pckt_length(p)); out_packet(p);
  3489. @ These are the local variables (if any) needed for |do_right|.
  3490. @<OUT: Declare local variables (if any) for |do_right|@>=
  3491. @ And here is the device dependent code for |do_right|.
  3492. @<OUT: Process a |right| or |w| or |x|@>=
  3493. if cur_class<right_cl then out_one(cur_cmd) {|w0| or |x0|}
  3494. else out_signed(dvi_right_cmd[cur_class],cur_parm); {|right|, |w|, or |x|}
  3495. @ Here we update the |out_max_h| value.
  3496. @<OUT: Move right@>=
  3497. if abs(cur_h)>out_max_h then out_max_h:=abs(cur_h);
  3498. @ These are the local variables (if any) needed for |do_down|.
  3499. @<OUT: Declare local variables (if any) for |do_down|@>=
  3500. @ And here is the device dependent code for |do_down|.
  3501. @<OUT: Process a |down| or |y| or |z|@>=
  3502. if cur_class<down_cl then out_one(cur_cmd) {|y0| or |z0|}
  3503. else out_signed(dvi_down_cmd[cur_class],cur_parm); {|down|, |y|, or |z|}
  3504. @ Here we update the |out_max_v| value.
  3505. @<OUT: Move down@>=
  3506. if abs(cur_v)>out_max_v then out_max_v:=abs(cur_v);
  3507. @ These are the local variables (if any) needed for |do_width|.
  3508. @<OUT: Declare local variables (if any) for |do_width|@>=
  3509. @ And here is the device dependent code for |do_width|.
  3510. @<OUT: Typeset a |width|@>=
  3511. out_one(set_rule);
  3512. out_four(width_dimen); out_four(cur_h_dimen);
  3513. @ These are the additional local variables (if any) needed for |do_rule|;
  3514. the variable |@!visible| is already declared.
  3515. @<OUT: Declare additional local variables |do_rule|@>=
  3516. @ And here is the device dependent code for |do_rule|.
  3517. @<OUT: Typeset a visible |rule|@>=
  3518. out_one(dvi_rule_cmd[cur_upd]);
  3519. out_four(cur_v_dimen); out_four(cur_h_dimen);
  3520. @ @<OUT: Typeset an invisible |rule|@>=
  3521. @<OUT: Typeset a visible |rule|@>
  3522. @ These are the local variables (if any) needed for |do_font|.
  3523. @<OUT: Declare local variables (if any) for |do_font|@>=
  3524. @ And here is the device dependent code for |do_font|; if the \.{VF} file
  3525. for a font could not be found, we simply assume this must be a real font.
  3526. @<OUT: Look for a font file before trying to read the \.{VF} file;
  3527.   if found |goto done|@>=
  3528. @ @<OUT: Look for a font file after trying to read the \.{VF} file@>=
  3529. if(out_nf>=max_fonts) then overflow(str_fonts,max_fonts);
  3530. print('OUT: font ',cur_fnt:1); d_print(' => ',out_nf:1);
  3531. print_font(cur_fnt);
  3532. d_print(' at ',font_scaled(cur_fnt):1,' DVI units'); print_ln('.');
  3533. font_type(cur_fnt):=out_font_type; font_font(cur_fnt):=out_nf;
  3534. out_fnts[out_nf]:=cur_fnt; incr(out_nf);
  3535. out_fnt_def(cur_fnt);
  3536. @ These are the local variables (if any) needed for |do_char|.
  3537. @<OUT: Declare local variables (if any) for |do_char|@>=
  3538. @ And here is the device dependent code for |do_char|.
  3539. @<OUT: Typeset a |char|@>=
  3540. @!debug if font_type(cur_fnt)<>out_font_type then confusion(str_fonts);
  3541. gubed @;
  3542. if cur_fnt<>out_fnt then
  3543.   begin out_unsigned(fnt1,font_font(cur_fnt)); out_fnt:=cur_fnt;
  3544.   end;
  3545. out_char(cur_upd,cur_ext,cur_res);
  3546. @ If the program terminates in the middle of a page, we write as many
  3547. |pop|s as necessary and one |eop|.
  3548. @<OUT: Finish incomplete page@>=
  3549. begin while stack_ptr>0 do
  3550.   begin out_one(pop); decr(stack_ptr);
  3551.   end;
  3552.   out_one(eop);
  3553. @ If the output file has been started, we write the postamble; in
  3554. addition we print the number of bytes and pages written to |out_file|.
  3555. @<OUT: Finish output file(s)@>=
  3556. if out_loc>0 then
  3557.   begin @<OUT: Write the postamble@>;
  3558.   k:=7-((out_loc-1) mod 4); {the number of 223's}
  3559.   while k>0 do
  3560.     begin out_one(223); decr(k);
  3561.     end;
  3562.   print('OUT file: ',out_loc:1,' bytes, ',out_pages:1,' page');
  3563.   if out_pages<>1 then print('s');
  3564.   end
  3565. else print('OUT file: no output');
  3566. print_ln(' written.');
  3567. if out_pages=0 then mark_harmless;
  3568. @ Here we simply write the values accumulated during the \.{DVI} output.
  3569. @<OUT: Write the postamble@>=
  3570. out_one(post); out_four(out_back); out_back:=out_loc-5;@/
  3571. out_four(dvi_num); out_four(dvi_den); out_four(out_mag);@/
  3572. out_four(out_max_v); out_four(out_max_h);@/
  3573. out_one(out_stack div @"100); out_one(out_stack mod @"100);@/
  3574. out_one(out_pages div @"100); out_one(out_pages mod @"100);@/
  3575. k:=out_nf;
  3576. while k>0 do
  3577.   begin decr(k); out_fnt_def(out_fnts[k]);
  3578.   end;
  3579. out_one(post_post); out_four(out_back);@/
  3580. out_one(dvi_id)
  3581. @ Here we could print more memory usage statistics; this possibility is,
  3582. however, not used for \.{DVIcopy}.
  3583. @<Print more memory usage statistics@>=
  3584. @* System-dependent changes.
  3585. This section should be replaced, if necessary, by changes to the program
  3586. that are necessary to make \.{DVIcopy} work at a particular installation.
  3587. It is usually best to design your change file so that all changes to
  3588. previous sections preserve the section numbering; then everybody's version
  3589. will be consistent with the printed program. More extensive changes,
  3590. which introduce new sections, can be inserted here; then only the index
  3591. itself will get a new section number.
  3592. @^system dependencies@>
  3593. @* Index.
  3594. Pointers to error messages appear here together with the section numbers
  3595. where each ident\-i\-fier is used.
  3596.