blob: 80e8d5ffa1fc05983fb384c7c0666330b11bc537 [file] [log] [blame]
Darren Tucker167bd9c2003-09-07 12:34:54 +10001#!/usr/bin/awk
2#
Darren Tucker6ac91a72009-10-24 11:52:42 +11003# $Id: mdoc2man.awk,v 1.9 2009/10/24 00:52:42 dtucker Exp $
Darren Tucker51e5ab02007-06-05 19:16:59 +10004#
Darren Tucker167bd9c2003-09-07 12:34:54 +10005# Version history:
Darren Tucker51e5ab02007-06-05 19:16:59 +10006# v4+ Adapted for OpenSSH Portable (see cvs Id and history)
Darren Tucker167bd9c2003-09-07 12:34:54 +10007# v3, I put the program under a proper license
8# Dan Nelson <dnelson@allantgroup.com> added .An, .Aq and fixed a typo
9# v2, fixed to work on GNU awk --posix and MacOS X
10# v1, first attempt, didn't work on MacOS X
11#
12# Copyright (c) 2003 Peter Stuge <stuge-mdoc2man@cdy.org>
13#
14# Permission to use, copy, modify, and distribute this software for any
15# purpose with or without fee is hereby granted, provided that the above
16# copyright notice and this permission notice appear in all copies.
17#
18# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
19# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
20# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
21# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
22# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
23# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
24# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
25
26
27BEGIN {
28 optlist=0
29 oldoptlist=0
30 nospace=0
31 synopsis=0
32 reference=0
33 block=0
34 ext=0
35 extopt=0
36 literal=0
37 prenl=0
Darren Tuckerd062da52004-07-02 18:43:09 +100038 breakw=0
Darren Tucker167bd9c2003-09-07 12:34:54 +100039 line=""
40}
41
42function wtail() {
43 retval=""
44 while(w<nwords) {
45 if(length(retval))
46 retval=retval OFS
47 retval=retval words[++w]
48 }
49 return retval
50}
51
52function add(str) {
53 for(;prenl;prenl--)
54 line=line "\n"
55 line=line str
56}
57
58! /^\./ {
59 for(;prenl;prenl--)
60 print ""
61 print
62 if(literal)
63 print ".br"
64 next
65}
66
67/^\.\\"/ { next }
68
69{
70 option=0
71 parens=0
72 angles=0
73 sub("^\\.","")
74 nwords=split($0,words)
75 for(w=1;w<=nwords;w++) {
76 skip=0
77 if(match(words[w],"^Li|Pf$")) {
78 skip=1
79 } else if(match(words[w],"^Xo$")) {
80 skip=1
81 ext=1
82 if(length(line)&&!(match(line," $")||prenl))
Damien Millera8e06ce2003-11-21 23:48:55 +110083 add(OFS)
Darren Tucker167bd9c2003-09-07 12:34:54 +100084 } else if(match(words[w],"^Xc$")) {
85 skip=1
86 ext=0
87 if(!extopt)
Damien Millera8e06ce2003-11-21 23:48:55 +110088 prenl++
Darren Tucker167bd9c2003-09-07 12:34:54 +100089 w=nwords
90 } else if(match(words[w],"^Bd$")) {
91 skip=1
92 if(match(words[w+1],"-literal")) {
Damien Millera8e06ce2003-11-21 23:48:55 +110093 literal=1
94 prenl++
95 w=nwords
Darren Tucker167bd9c2003-09-07 12:34:54 +100096 }
97 } else if(match(words[w],"^Ed$")) {
98 skip=1
99 literal=0
100 } else if(match(words[w],"^Ns$")) {
101 skip=1
102 if(!nospace)
Damien Millera8e06ce2003-11-21 23:48:55 +1100103 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000104 sub(" $","",line)
105 } else if(match(words[w],"^No$")) {
106 skip=1
107 sub(" $","",line)
108 add(words[++w])
109 } else if(match(words[w],"^Dq$")) {
110 skip=1
111 add("``")
112 add(words[++w])
113 while(w<nwords&&!match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100114 add(OFS words[++w])
Darren Tucker167bd9c2003-09-07 12:34:54 +1000115 add("''")
116 if(!nospace&&match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100117 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000118 } else if(match(words[w],"^Sq|Ql$")) {
119 skip=1
120 add("`" words[++w] "'")
121 if(!nospace&&match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100122 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000123 } else if(match(words[w],"^Oo$")) {
124 skip=1
125 extopt=1
126 if(!nospace)
Damien Millera8e06ce2003-11-21 23:48:55 +1100127 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000128 add("[")
129 } else if(match(words[w],"^Oc$")) {
130 skip=1
131 extopt=0
132 add("]")
133 }
134 if(!skip) {
135 if(!nospace&&length(line)&&!(match(line," $")||prenl))
Damien Millera8e06ce2003-11-21 23:48:55 +1100136 add(OFS)
Darren Tucker167bd9c2003-09-07 12:34:54 +1000137 if(nospace==1)
Damien Millera8e06ce2003-11-21 23:48:55 +1100138 nospace=0
Darren Tucker167bd9c2003-09-07 12:34:54 +1000139 }
140 if(match(words[w],"^Dd$")) {
Darren Tucker88bca062007-06-05 19:30:47 +1000141 if(match(words[w+1],"^\\$Mdocdate:")) {
Darren Tucker51e5ab02007-06-05 19:16:59 +1000142 w++;
143 if(match(words[w+4],"^\\$$")) {
144 words[w+4] = ""
145 }
146 }
Darren Tucker167bd9c2003-09-07 12:34:54 +1000147 date=wtail()
148 next
149 } else if(match(words[w],"^Dt$")) {
150 id=wtail()
151 next
Darren Tucker6ac91a72009-10-24 11:52:42 +1100152 } else if(match(words[w],"^Ux$")) {
153 add("UNIX")
154 skip=1
Darren Tuckerf5615962005-05-31 16:59:16 +1000155 } else if(match(words[w],"^Ox$")) {
156 add("OpenBSD")
157 skip=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000158 } else if(match(words[w],"^Os$")) {
159 add(".TH " id " \"" date "\" \"" wtail() "\"")
160 } else if(match(words[w],"^Sh$")) {
161 add(".SH")
162 synopsis=match(words[w+1],"SYNOPSIS")
163 } else if(match(words[w],"^Xr$")) {
164 add("\\fB" words[++w] "\\fP(" words[++w] ")" words[++w])
165 } else if(match(words[w],"^Rs$")) {
166 split("",refauthors)
167 nrefauthors=0
168 reftitle=""
169 refissue=""
170 refdate=""
171 refopt=""
Darren Tucker0c0dc492007-06-05 20:01:16 +1000172 refreport=""
Darren Tucker167bd9c2003-09-07 12:34:54 +1000173 reference=1
174 next
175 } else if(match(words[w],"^Re$")) {
176 prenl++
177 for(i=nrefauthors-1;i>0;i--) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100178 add(refauthors[i])
179 if(i>1)
180 add(", ")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000181 }
182 if(nrefauthors>1)
Damien Millera8e06ce2003-11-21 23:48:55 +1100183 add(" and ")
Darren Tucker0c0dc492007-06-05 20:01:16 +1000184 if(nrefauthors>0)
185 add(refauthors[0] ", ")
186 add("\\fI" reftitle "\\fP")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000187 if(length(refissue))
Damien Millera8e06ce2003-11-21 23:48:55 +1100188 add(", " refissue)
Darren Tucker0c0dc492007-06-05 20:01:16 +1000189 if(length(refreport)) {
190 add(", " refreport)
191 }
Darren Tucker167bd9c2003-09-07 12:34:54 +1000192 if(length(refdate))
Damien Millera8e06ce2003-11-21 23:48:55 +1100193 add(", " refdate)
Darren Tucker167bd9c2003-09-07 12:34:54 +1000194 if(length(refopt))
Damien Millera8e06ce2003-11-21 23:48:55 +1100195 add(", " refopt)
Darren Tucker167bd9c2003-09-07 12:34:54 +1000196 add(".")
197 reference=0
198 } else if(reference) {
199 if(match(words[w],"^%A$")) { refauthors[nrefauthors++]=wtail() }
200 if(match(words[w],"^%T$")) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100201 reftitle=wtail()
202 sub("^\"","",reftitle)
203 sub("\"$","",reftitle)
Darren Tucker167bd9c2003-09-07 12:34:54 +1000204 }
205 if(match(words[w],"^%N$")) { refissue=wtail() }
206 if(match(words[w],"^%D$")) { refdate=wtail() }
207 if(match(words[w],"^%O$")) { refopt=wtail() }
Darren Tucker0c0dc492007-06-05 20:01:16 +1000208 if(match(words[w],"^%R$")) { refreport=wtail() }
Darren Tucker167bd9c2003-09-07 12:34:54 +1000209 } else if(match(words[w],"^Nm$")) {
210 if(synopsis) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100211 add(".br")
212 prenl++
Darren Tucker167bd9c2003-09-07 12:34:54 +1000213 }
214 n=words[++w]
215 if(!length(name))
Damien Millera8e06ce2003-11-21 23:48:55 +1100216 name=n
Darren Tucker167bd9c2003-09-07 12:34:54 +1000217 if(!length(n))
Damien Millera8e06ce2003-11-21 23:48:55 +1100218 n=name
Darren Tucker167bd9c2003-09-07 12:34:54 +1000219 add("\\fB" n "\\fP")
220 if(!nospace&&match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100221 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000222 } else if(match(words[w],"^Nd$")) {
223 add("\\- " wtail())
224 } else if(match(words[w],"^Fl$")) {
225 add("\\fB\\-" words[++w] "\\fP")
226 if(!nospace&&match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100227 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000228 } else if(match(words[w],"^Ar$")) {
229 add("\\fI")
230 if(w==nwords)
Damien Millera8e06ce2003-11-21 23:48:55 +1100231 add("file ...\\fP")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000232 else {
Damien Millera8e06ce2003-11-21 23:48:55 +1100233 add(words[++w] "\\fP")
234 while(match(words[w+1],"^\\|$"))
235 add(OFS words[++w] " \\fI" words[++w] "\\fP")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000236 }
237 if(!nospace&&match(words[w+1],"^[\\.,]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100238 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000239 } else if(match(words[w],"^Cm$")) {
240 add("\\fB" words[++w] "\\fP")
241 while(w<nwords&&match(words[w+1],"^[\\.,:;)]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100242 add(words[++w])
Darren Tucker167bd9c2003-09-07 12:34:54 +1000243 } else if(match(words[w],"^Op$")) {
244 option=1
245 if(!nospace)
Damien Millera8e06ce2003-11-21 23:48:55 +1100246 nospace=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000247 add("[")
248 } else if(match(words[w],"^Pp$")) {
249 prenl++
250 } else if(match(words[w],"^An$")) {
251 prenl++
252 } else if(match(words[w],"^Ss$")) {
253 add(".SS")
254 } else if(match(words[w],"^Pa$")&&!option) {
255 add("\\fI")
256 w++
257 if(match(words[w],"^\\."))
Damien Millera8e06ce2003-11-21 23:48:55 +1100258 add("\\&")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000259 add(words[w] "\\fP")
260 while(w<nwords&&match(words[w+1],"^[\\.,:;)]"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100261 add(words[++w])
Darren Tucker167bd9c2003-09-07 12:34:54 +1000262 } else if(match(words[w],"^Dv$")) {
263 add(".BR")
264 } else if(match(words[w],"^Em|Ev$")) {
265 add(".IR")
266 } else if(match(words[w],"^Pq$")) {
267 add("(")
268 nospace=1
269 parens=1
270 } else if(match(words[w],"^Aq$")) {
271 add("<")
272 nospace=1
273 angles=1
274 } else if(match(words[w],"^S[xy]$")) {
275 add(".B " wtail())
276 } else if(match(words[w],"^Ic$")) {
277 plain=1
278 add("\\fB")
279 while(w<nwords) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100280 w++
281 if(match(words[w],"^Op$")) {
282 w++
283 add("[")
284 words[nwords]=words[nwords] "]"
285 }
286 if(match(words[w],"^Ar$")) {
287 add("\\fI" words[++w] "\\fP")
288 } else if(match(words[w],"^[\\.,]")) {
289 sub(" $","",line)
290 if(plain) {
291 add("\\fP")
292 plain=0
293 }
294 add(words[w])
295 } else {
296 if(!plain) {
297 add("\\fB")
298 plain=1
299 }
300 add(words[w])
301 }
302 if(!nospace)
303 add(OFS)
Darren Tucker167bd9c2003-09-07 12:34:54 +1000304 }
305 sub(" $","",line)
306 if(plain)
Damien Millera8e06ce2003-11-21 23:48:55 +1100307 add("\\fP")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000308 } else if(match(words[w],"^Bl$")) {
309 oldoptlist=optlist
310 if(match(words[w+1],"-bullet"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100311 optlist=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000312 else if(match(words[w+1],"-enum")) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100313 optlist=2
314 enum=0
Darren Tucker167bd9c2003-09-07 12:34:54 +1000315 } else if(match(words[w+1],"-tag"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100316 optlist=3
Darren Tucker167bd9c2003-09-07 12:34:54 +1000317 else if(match(words[w+1],"-item"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100318 optlist=4
Darren Tucker167bd9c2003-09-07 12:34:54 +1000319 else if(match(words[w+1],"-bullet"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100320 optlist=1
Darren Tucker167bd9c2003-09-07 12:34:54 +1000321 w=nwords
322 } else if(match(words[w],"^El$")) {
323 optlist=oldoptlist
Darren Tuckerd062da52004-07-02 18:43:09 +1000324 } else if(match(words[w],"^Bk$")) {
325 if(match(words[w+1],"-words")) {
326 w++
327 breakw=1
328 }
329 } else if(match(words[w],"^Ek$")) {
330 breakw=0
Darren Tucker167bd9c2003-09-07 12:34:54 +1000331 } else if(match(words[w],"^It$")&&optlist) {
332 if(optlist==1)
Damien Millera8e06ce2003-11-21 23:48:55 +1100333 add(".IP \\(bu")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000334 else if(optlist==2)
Damien Millera8e06ce2003-11-21 23:48:55 +1100335 add(".IP " ++enum ".")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000336 else if(optlist==3) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100337 add(".TP")
338 prenl++
Darren Tucker58cef1f2004-06-28 15:45:08 +1000339 if(match(words[w+1],"^Pa$|^Ev$")) {
Damien Millera8e06ce2003-11-21 23:48:55 +1100340 add(".B")
341 w++
342 }
Darren Tucker167bd9c2003-09-07 12:34:54 +1000343 } else if(optlist==4)
Damien Millera8e06ce2003-11-21 23:48:55 +1100344 add(".IP")
Darren Tucker167bd9c2003-09-07 12:34:54 +1000345 } else if(match(words[w],"^Sm$")) {
346 if(match(words[w+1],"off"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100347 nospace=2
Darren Tucker167bd9c2003-09-07 12:34:54 +1000348 else if(match(words[w+1],"on"))
Damien Millera8e06ce2003-11-21 23:48:55 +1100349 nospace=0
Darren Tucker167bd9c2003-09-07 12:34:54 +1000350 w++
351 } else if(!skip) {
352 add(words[w])
353 }
354 }
355 if(match(line,"^\\.[^a-zA-Z]"))
356 sub("^\\.","",line)
357 if(parens)
358 add(")")
359 if(angles)
360 add(">")
361 if(option)
362 add("]")
363 if(ext&&!extopt&&!match(line," $"))
364 add(OFS)
365 if(!ext&&!extopt&&length(line)) {
366 print line
367 prenl=0
368 line=""
369 }
370}