Hi folks, I have been dissatisfied with groff man(7)'s SY and YS macros for a long time. My primary grievance is one that has frustrated its uptake by documenters of libraries: the macros are designed for synopsizing Unix commands, not C library functions. After working on the ncurses man pages for a while it became clear to me how to modestly revise the way groff man(7)'s SY and YS macros work to serve both set of authors better. My proposal, as a diff to groff Git HEAD, is attached. Here are the highlights, starting with the most disruptive: 1. The `SY` macro no longer puts vertical space on the output. That's up to you now. You can use whatever paragraphing macro you please to separate multiple synopses. 2. The `SY` macro now initially breaks the output line _only_ if it is encountered repeatedly without a preceding `YS` call. Formerly, it _always_ initially broke the output line because it put vertical space on the output. This is largely a guardrail in case someone forgets to call `YS`. (Using `SY` multiple times before `YS` used to be idiomatic for synonymous command invocations like "foobar --help" and "foobar -h"; no longer. Now you bracket each with `SY` and `YS` and leave out a paragraphing macro if you don't want one.) 3. The computed indentation of synopsis lines after the first now also includes the width of anything on the line _before_ the synopsis. This is so that you can precede the synopsis keyword with other syntax. The most likely application of this is a return type for a C function prototype. 4. The `SY` macro now accepts an optional second argument. This second argument is typeset in bold and replaces the fixed-width space that is appended to the synopsis keyword in `SY`'s single-argument form. As with that fixed-width space, the width of this suffix argument informs the indentation used on subsequent lines of the synopsis if it needs to break.[*] 5. I tested the portability of these changes to DWB 3.3 troff, Heirloom Doctools troff, and Solaris 10 troff, and worked around what appears to be a glitch in their man(7) packages. That helped, but a small problem remains: the computed width marked with an asterisk[*] in the previous item is 1n too short. Concretely, instead of: int wborder(WINDOW *win, chtype ls, chtype rs, chtype ts, chtype bs, chtype tl, chtype tr, chtype bl, chtype br); you get the following on DWB/Heirloom/Solaris 10 nroff. int wborder(WINDOW *win, chtype ls, chtype rs, chtype ts, chtype bs, chtype tl, chtype tr, chtype bl, chtype br); I regarded this defect as too inconsequential to worry about, but if someone wants to research the innards of AT&T man(7) to see if a workaround can be found, I'll be receptive to suggestions. 6. I discovered that mandoc mishandles indentation; it does not honor the rules set forth in §6 of CSTR #54 (supporting evidence attached). The result is merely ugly, however, and no text is lost. Since mandoc is actively maintained I assume that eventually this will get fixed, perhaps after a strident exhortation to migrate one's documents to mdoc(7). ;-) 7. The first argument to `SY` remains the keyword. I avoided the complexity of a three-argument form partly because I didn't need it and partly because I wanted to preserve the invariant of the first argument being the name of the documented thing. A potential advantage to this practice is that we can, in future groff man(7) development, automatically generate hyperlink tags for these items. Need to know where a command or function is synopsized? With tags you can find out much more quickly than with a textual search. (The invariant is actually a bigger advantage for hypothetical external tools--or grep(1)--because the `SY` macro would "know" which item was the keyword based on the argument count.) The payoff is that, if adopted, you can write a function synopsis like this. .P .B int .SY wborder ( .BI WINDOW\~* win , .BI chtype\~ ls , .BI chtype\~ rs , .BI chtype\~ ts , .BI chtype\~ bs , .BI chtype\~ tl , .BI chtype\~ tr , .BI chtype\~ bl , .BI chtype\~ br ); .YS ...and not worry about the line length or line breaking or any of that stuff. Compare to the status quo for the foregoing function in ncurses. .SH SYNOPSIS .nf ... \fBint wborder(WINDOW *\fIwin\fB, chtype \fIls\fB, chtype \fIrs\fB,\fR \fBchtype \fIts\fB, chtype \fIbs\fB, chtype \fItl\fB, chtype \fItr\fB,\fR \fBchtype \fIbl\fB, chtype \fIbr\fB);\fR .fi The foregoing looks messier to me as input, and moreover it does not adapt to the width of the terminal, thanks to those `nf`/`fi` formatter requests. It will waste space on wide terminals and overrun the line on narrow ones. You may notice that I haven't changed `YS` at all. The implementation is in the permissively licensed "an-ext.tmac" file, so there is no _legal_ reason for those allergic to copyleft (or even Apache-style) licenses to have any compunctions about adopting it. The documentation is under the traditional GNU documentation license, known to SPDX as the LaTeX 2e license. (I don't know which is the earlier provenance, and I would like to.) My questions: A. Does anyone object to me committing this change to groff's master branch? It will of course require a NEWS item, which I will write. B. Does this look enticing enough to any documenters of C libraries for you to adopt it? Regards, Branden
diff --git a/tmac/an-ext.tmac b/tmac/an-ext.tmac index 5a2ede65c..2fb4190bb 100644 --- a/tmac/an-ext.tmac +++ b/tmac/an-ext.tmac @@ -1,6 +1,6 @@ .\" groff extension macros for man(7) package .\" -.\" Copyright (C) 2007-2023 Free Software Foundation, Inc. +.\" Copyright (C) 2007-2024 Free Software Foundation, Inc. .\" .\" Written by Eric S. Raymond <esr@xxxxxxxxxxx> .\" Werner Lemberg <wl@xxxxxxx> @@ -62,17 +62,31 @@ .de SY . nr mA \\n(.j . ad l . nr mI \\n(.i -. sp . \} -. el \{\ +. el \ . br -. ns -. \} . -. nr mT \w'\fB\\$1\fP\ ' -. HP \\n(mTu +. nr mT \\n(.k+\w'\fB\\$1\fP' +. if \\n(.$=1 \ +. nr mT +\w'\ ' +. if \\n(.$>1 \ +. nr mT +\w'\fB\\$2\fP' +. \" Ensure that a partially collected line exists so that the `in` +. \" request affects only _subsequent_ output lines. (CSTR #54 §6) +\&\c +' in +\\n(mTu +. if \\n(.$=1 \{\ +. do nr .X +0 \" Ensure this Heirloom register exists for testing. +. \" If the formatter is not groff, work around DWB/Heirloom/ +. \" Solaris 10 glitch. Something in their man(7) defeats the rules +. \" set forth in CSTR #54 §6. +. if \\n(.g=0:\\n(.X \ +' ti -\\n(mTu +. B \\$1\ \c +. \} +. if \\n(.$>1 \ +. B \\$1\\$2\c . rr mT -. B "\\$1" .. . . diff --git a/tmac/groff_man.7.man.in b/tmac/groff_man.7.man.in index f8488c6b0..d8f65334f 100644 --- a/tmac/groff_man.7.man.in +++ b/tmac/groff_man.7.man.in @@ -61,7 +61,7 @@ .SH Name .\" Legal Terms .\" ==================================================================== .\" -.\" Copyright (C) 1999-2023 Free Software Foundation, Inc. +.\" Copyright (C) 1999-2024 Free Software Foundation, Inc. .\" .\" Permission is granted to make and distribute verbatim copies of this .\" manual provided the copyright notice and this permission notice are @@ -106,6 +106,7 @@ .SH Synopsis .SY "groff \-man" .RI [ option\~ .\|.\|.\&] .RI [ file\~ .\|.\|.] +.YS . .SY "groff \-m man" .RI [ option\~ .\|.\|.\&] @@ -181,13 +182,13 @@ .SH Description \&.SH Section heading Document structure macros \&.SM Small Font style macros \&.SS Subsection heading Document structure macros -\&.SY Synopsis start Command synopsis macros +\&.SY Synopsis start Synopsis macros \&.TH Title heading Document structure macros \&.TP Tagged paragraph Paragraphing macros \&.TQ Supplemental paragraph tag Paragraphing macros \&.UE URI end Hyperlink macros \&.UR URI start Hyperlink macros -\&.YS Synopsis end Command synopsis macros +\&.YS Synopsis end Synopsis macros .TE . . @@ -994,7 +995,7 @@ .SS "Paragraphing macros" . _endif()dnl .\" ==================================================================== -.SS "Command synopsis macros" +.SS "Synopsis macros" .\" ==================================================================== . Use @@ -1003,8 +1004,6 @@ .SS "Command synopsis macros" .B .YS to summarize syntax using familiar Unix conventions. . -These macros break the output line. -. .\" TODO: Determine whether this (is still? was ever?) true. .\" Furthermore, .\" some tools are able to interpret these macros semantically and treat @@ -1034,54 +1033,61 @@ .SS "Command synopsis macros" . . .TP -.BI .SY\~ keyword +.BI .SY\~ keyword\~\c +.RI [ suffix ] Begin synopsis. . -A new paragraph begins as with -.BR .P , -unless +Adjustment and automatic hyphenation are disabled. +. +If .B .SY has already been called without a corresponding .BR .YS , -in which case only a break is performed. -. -Adjustment and automatic hyphenation are disabled. +a break is performed. . .I keyword -is set in bold. +and +.I suffix +(if any) +are set in bold. +. +If a break is required in subsequent text +(up to another paragraphing +sectioning, +or synopsis macro call), +lines after the first are indented. . -If a break is required, -lines after the first are indented by the width of +The indentation amount is the width of .I keyword -plus a space. +plus a space, +if that is the only argument, +and by the sum of the widths of +.I keyword +and +.I suffix +otherwise. . . .TP .B .YS -End synopsis. -. -Indentation, +End synopsis, +breaking the line and restoring +indentation, adjustment, and hyphenation -are restored to their previous states. +to their previous states. _ifstyle()dnl . . .P -Multiple +Interleave multiple .BR .SY / .YS -blocks can be specified, -for instance to distinguish differing modes of operation of a complex -command like -.MR tar 1 ; -each will be vertically separated as paragraphs are. +blocks with paragraphing macros to distinguish differing modes of +operation of a complex command like +.MR tar 1 . . -. -.P -.B .SY -can be repeated before -.B .YS -to indicate synonymous ways of invoking a particular mode of operation. +Omit the paragraphing macro to indicate synonymous ways of invoking a +particular mode of operation. . . .P @@ -1110,6 +1116,7 @@ .SS "Command synopsis macros" \&.YS \&. \&. +\&.P \&.SY groff \&.B \e\-h \&.YS @@ -1119,10 +1126,12 @@ .SS "Command synopsis macros" \&.YS \&. \&. +\&.P \&.SY groff \&.B \e\-v \&.RI [ option\e\(ti .\e|.\e|.\e&] \&.RI [ file\e\(ti .\e|.\e|.] +\&.YS \&. \&.SY groff \&.B \e\-\e\-version @@ -1136,6 +1145,7 @@ .SS "Command synopsis macros" produces the following output. . . +.P .RS .SY groff .RB [ \-abcCeEgGijklNpRsStUVXzZ ] @@ -1181,18 +1191,22 @@ .SS "Command synopsis macros" .YS . . +.P .SY groff .B \-h +.YS . .SY groff .B \-\-help .YS . . +.P .SY groff .B \-v .RI [ option\~ .\|.\|.\&] .RI [ file\~ .\|.\|.] +.YS . .SY groff .B \%\-\-version @@ -1275,6 +1289,44 @@ .SS "Command synopsis macros" and causing additional inter-sentence space to be placed after it. . See subsection \(lqPortability\(rq below. +. +. +.P +We might synopsize the standard C library function +.MR snprintf 3 +as follows. +. +. +.P +.RS +.EX +\&.P +\&.B int +\&.SY snprintf ( +\&.BI \(dqchar *\(dq str , +\&.BI size_t\(rs\(ti size , +\&.BI \(dqconst char *\(dq format , +\&.RB .\(rs|.\(rs|. ); +\&.YS +.EE +.RE +. +. +.P +.I man +produces the following result. +. +. +.RS +.P +.B int +.SY snprintf ( +.BI "char\~*" str , +.BI size_t\~ size , +.BI "const\~char\~*" format , +.RB .\|.\|. ); +.YS +.RE _endif()dnl . . @@ -4290,7 +4342,7 @@ .SH Notes . .IP Examples of ellipsis usage are shown above, -in subsection \[lq]Command synopsis macros\[rq]. +in subsection \[lq]Synopsis macros\[rq]. . The idiomatic .I roff
This is a normal line of reasonably generous length (about 60n). .sp The effect .ll 30n of \fBll\fP, \fBin\fP, or \fBti\fP is delayed, if a partially collected line exists, until after that line is output. .ll .sp The effect .in 30n of \fBll\fP, \fBin\fP, or \fBti\fP is delayed, if a partially collected line exists, until after that line is output. .in .sp The effect .ti 30n of \fBll\fP, \fBin\fP, or \fBti\fP is delayed, if a partially collected line exists, until after that line is output. .sp Now let's use the no-break control character with \fBin\fP and \fBti\fP. .sp The effect 'in 30n of \fBll\fP, \fBin\fP, or \fBti\fP is delayed, if a partially collected line exists, until after that line is output. .in .sp The effect 'ti 30n of \fBll\fP, \fBin\fP, or \fBti\fP is delayed, if a partially collected line exists, until after that line is output. .pl \n(nlu
Attachment:
signature.asc
Description: PGP signature