2 * Copyright (c) 1995 - 2006 Kungliga Tekniska Högskolan
3 * (Royal Institute of Technology, Stockholm, Sweden).
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
17 * 3. Neither the name of the Institute nor the names of its contributors
18 * may be used to endorse or promote products derived from this software
19 * without specific prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36 __RCSID("$Heimdal: sl.c 21160 2007-06-18 22:58:21Z lha $"
44 mandoc_template(SL_cmd
*cmds
,
45 const char *extra_string
)
48 char timestr
[64], cmd
[64];
52 printf(".\\\" Things to fix:\n");
53 printf(".\\\" * correct section, and operating system\n");
54 printf(".\\\" * remove Op from mandatory flags\n");
55 printf(".\\\" * use better macros for arguments (like .Pa for files)\n");
58 strftime(timestr
, sizeof(timestr
), "%b %d, %Y", localtime(&t
));
59 printf(".Dd %s\n", timestr
);
60 p
= strrchr(getprogname(), '/');
61 if(p
) p
++; else p
= getprogname();
62 strncpy(cmd
, p
, sizeof(cmd
));
63 cmd
[sizeof(cmd
)-1] = '\0';
66 printf(".Dt %s SECTION\n", cmd
);
67 printf(".Os OPERATING_SYSTEM\n");
69 printf(".Nm %s\n", p
);
71 printf("in search of a description\n");
72 printf(".Sh SYNOPSIS\n");
74 for(c
= cmds
; c
->name
; ++c
) {
75 /* if (c->func == NULL)
77 printf(".Op Fl %s", c
->name
);
81 if (extra_string
&& *extra_string
)
82 printf (".Ar %s\n", extra_string
);
83 printf(".Sh DESCRIPTION\n");
84 printf("Supported options:\n");
85 printf(".Bl -tag -width Ds\n");
87 for(c
= cmds
; c
->name
; ++c
) {
90 printf ("\n%s\n", prev
->usage
);
92 printf (".It Fl %s", c
->name
);
95 printf (", %s\n", c
->name
);
98 printf ("\n%s\n", prev
->usage
);
101 printf(".\\\".Sh ENVIRONMENT\n");
102 printf(".\\\".Sh FILES\n");
103 printf(".\\\".Sh EXAMPLES\n");
104 printf(".\\\".Sh DIAGNOSTICS\n");
105 printf(".\\\".Sh SEE ALSO\n");
106 printf(".\\\".Sh STANDARDS\n");
107 printf(".\\\".Sh HISTORY\n");
108 printf(".\\\".Sh AUTHORS\n");
109 printf(".\\\".Sh BUGS\n");
113 sl_match (SL_cmd
*cmds
, char *cmd
, int exactp
)
115 SL_cmd
*c
, *current
= NULL
, *partial_cmd
= NULL
;
116 int partial_match
= 0;
118 for (c
= cmds
; c
->name
; ++c
) {
121 if (strcmp (cmd
, c
->name
) == 0)
123 else if (strncmp (cmd
, c
->name
, strlen(cmd
)) == 0 &&
124 partial_cmd
!= current
) {
126 partial_cmd
= current
;
129 if (partial_match
== 1 && !exactp
)
136 sl_help (SL_cmd
*cmds
, int argc
, char **argv
)
140 if (getenv("SLMANDOC")) {
141 mandoc_template(cmds
, NULL
);
147 for (c
= cmds
; c
->name
; ++c
) {
150 printf ("\n\t%s%s", prev_c
->usage
? prev_c
->usage
: "",
151 prev_c
->usage
? "\n" : "");
153 printf ("%s", c
->name
);
155 printf (", %s", c
->name
);
158 printf ("\n\t%s%s", prev_c
->usage
? prev_c
->usage
: "",
159 prev_c
->usage
? "\n" : "");
161 c
= sl_match (cmds
, argv
[1], 0);
163 printf ("No such command: %s. "
164 "Try \"help\" for a list of all commands\n",
167 printf ("%s\t%s\n", c
->name
, c
->usage
);
168 if(c
->help
&& *c
->help
)
169 printf ("%s\n", c
->help
);
170 if((++c
)->name
&& c
->func
== NULL
) {
171 printf ("Synonyms:");
172 while (c
->name
&& c
->func
== NULL
)
173 printf ("\t%s", (c
++)->name
);
182 char *readline(char *prompt
);
183 void add_history(char *p
);
188 readline(char *prompt
)
191 printf ("%s", prompt
);
193 if(fgets(buf
, sizeof(buf
), stdin
) == NULL
)
195 buf
[strcspn(buf
, "\r\n")] = '\0';
207 sl_command(SL_cmd
*cmds
, int argc
, char **argv
)
210 c
= sl_match (cmds
, argv
[0], 0);
213 return (*c
->func
)(argc
, argv
);
222 sl_make_argv(char *line
, int *ret_argc
, char ***ret_argv
)
230 argv
= malloc(nargv
* sizeof(*argv
));
237 while(isspace((unsigned char)*p
))
244 } else if (*p
== '"') {
246 memmove(&p
[0], &p
[1], strlen(&p
[1]) + 1);
248 } else if (*p
== '\\') {
251 memmove(&p
[0], &p
[1], strlen(&p
[1]) + 1);
254 } else if (quote
|| !isspace((unsigned char)*p
)) {
261 if(argc
== nargv
- 1) {
264 tmp
= realloc (argv
, nargv
* sizeof(*argv
));
271 argv
[argc
++] = begining
;
272 while(isspace((unsigned char)*p
))
287 static jmp_buf sl_jmp
;
289 static void sl_sigint(int sig
)
294 static char *sl_readline(const char *prompt
)
298 old
= signal(SIGINT
, sl_sigint
);
301 s
= readline(rk_UNCONST(prompt
));
310 * return value of command */
312 sl_command_loop(SL_cmd
*cmds
, const char *prompt
, void **data
)
320 buf
= sl_readline(prompt
);
326 ret
= sl_make_argv(buf
, &argc
, &argv
);
328 fprintf(stderr
, "sl_loop: out of memory\n");
333 ret
= sl_command(cmds
, argc
, argv
);
335 printf ("Unrecognized command: %s\n", argv
[0]);
345 sl_loop(SL_cmd
*cmds
, const char *prompt
)
349 while((ret
= sl_command_loop(cmds
, prompt
, &data
)) >= 0)
355 sl_apropos (SL_cmd
*cmd
, const char *topic
)
357 for (; cmd
->name
!= NULL
; ++cmd
)
358 if (cmd
->usage
!= NULL
&& strstr(cmd
->usage
, topic
) != NULL
)
359 printf ("%-20s%s\n", cmd
->name
, cmd
->usage
);
363 * Help to be used with slc.
367 sl_slc_help (SL_cmd
*cmds
, int argc
, char **argv
)
370 sl_help(cmds
, 1, argv
- 1 /* XXX */);
372 SL_cmd
*c
= sl_match (cmds
, argv
[0], 0);
374 fprintf (stderr
, "No such command: %s. "
375 "Try \"help\" for a list of commands\n",
379 char *fake
[] = { NULL
, "--help", NULL
};
382 fprintf(stderr
, "\n");
384 if(c
->help
&& *c
->help
)
385 fprintf (stderr
, "%s\n", c
->help
);
386 if((++c
)->name
&& c
->func
== NULL
) {
388 fprintf (stderr
, "Synonyms:");
389 while (c
->name
&& c
->func
== NULL
) {
390 fprintf (stderr
, "%s%s", f
? ", " : " ", (c
++)->name
);
393 fprintf (stderr
, "\n");