Deleted Added
full compact
option.c (221715) option.c (237613)
1/*
1/*
2 * Copyright (C) 1984-2011 Mark Nudelman
2 * Copyright (C) 1984-2012 Mark Nudelman
3 *
4 * You may distribute under the terms of either the GNU General Public
5 * License or the Less License, as specified in the README file.
6 *
3 *
4 * You may distribute under the terms of either the GNU General Public
5 * License or the Less License, as specified in the README file.
6 *
7 * For more information about less, or for information on how to
8 * contact the author, see the README file.
7 * For more information, see the README file.
9 */
10
11
12/*
13 * Process command line options.
14 *
15 * Each option is a single letter which controls a program variable.
16 * The options have defaults which may be changed via
17 * the command line option, toggled via the "-" command,
18 * or queried via the "_" command.
19 */
20
21#include "less.h"
22#include "option.h"
23
24static struct loption *pendopt;
25public int plusoption = FALSE;
26
27static char *optstring();
28static int flip_triple();
29
30extern int screen_trashed;
31extern int less_is_more;
32extern int quit_at_eof;
33extern char *every_first_cmd;
34
35/*
36 * Return a printable description of an option.
37 */
38 static char *
39opt_desc(o)
40 struct loption *o;
41{
42 static char buf[OPTNAME_MAX + 10];
43 if (o->oletter == OLETTER_NONE)
44 SNPRINTF1(buf, sizeof(buf), "--%s", o->onames->oname);
45 else
46 SNPRINTF2(buf, sizeof(buf), "-%c (--%s)", o->oletter, o->onames->oname);
47 return (buf);
48}
49
50/*
51 * Return a string suitable for printing as the "name" of an option.
52 * For example, if the option letter is 'x', just return "-x".
53 */
54 public char *
55propt(c)
56 int c;
57{
58 static char buf[8];
59
60 sprintf(buf, "-%s", prchar(c));
61 return (buf);
62}
63
64/*
65 * Scan an argument (either from the command line or from the
66 * LESS environment variable) and process it.
67 */
68 public void
69scan_option(s)
70 char *s;
71{
72 register struct loption *o;
73 register int optc;
74 char *optname;
75 char *printopt;
76 char *str;
77 int set_default;
78 int lc;
79 int err;
80 PARG parg;
81
82 if (s == NULL)
83 return;
84
85 /*
86 * If we have a pending option which requires an argument,
87 * handle it now.
88 * This happens if the previous option was, for example, "-P"
89 * without a following string. In that case, the current
90 * option is simply the argument for the previous option.
91 */
92 if (pendopt != NULL)
93 {
94 switch (pendopt->otype & OTYPE)
95 {
96 case STRING:
97 (*pendopt->ofunc)(INIT, s);
98 break;
99 case NUMBER:
100 printopt = opt_desc(pendopt);
101 *(pendopt->ovar) = getnum(&s, printopt, (int*)NULL);
102 break;
103 }
104 pendopt = NULL;
105 return;
106 }
107
108 set_default = FALSE;
109 optname = NULL;
110
111 while (*s != '\0')
112 {
113 /*
114 * Check some special cases first.
115 */
116 switch (optc = *s++)
117 {
118 case ' ':
119 case '\t':
120 case END_OPTION_STRING:
121 continue;
122 case '-':
123 /*
124 * "--" indicates an option name instead of a letter.
125 */
126 if (*s == '-')
127 {
128 optname = ++s;
129 break;
130 }
131 /*
132 * "-+" means set these options back to their defaults.
133 * (They may have been set otherwise by previous
134 * options.)
135 */
136 set_default = (*s == '+');
137 if (set_default)
138 s++;
139 continue;
140 case '+':
141 /*
142 * An option prefixed by a "+" is ungotten, so
143 * that it is interpreted as less commands
144 * processed at the start of the first input file.
145 * "++" means process the commands at the start of
146 * EVERY input file.
147 */
148 plusoption = TRUE;
149 s = optstring(s, &str, propt('+'), NULL);
150 if (*str == '+')
151 every_first_cmd = save(++str);
152 else
153 ungetsc(str);
154 continue;
155 case '0': case '1': case '2': case '3': case '4':
156 case '5': case '6': case '7': case '8': case '9':
157 /*
158 * Special "more" compatibility form "-<number>"
159 * instead of -z<number> to set the scrolling
160 * window size.
161 */
162 s--;
163 optc = 'z';
164 break;
165 case 'n':
166 if (less_is_more)
167 optc = 'z';
168 break;
169 }
170
171 /*
172 * Not a special case.
173 * Look up the option letter in the option table.
174 */
175 err = 0;
176 if (optname == NULL)
177 {
178 printopt = propt(optc);
179 lc = ASCII_IS_LOWER(optc);
180 o = findopt(optc);
181 } else
182 {
183 printopt = optname;
184 lc = ASCII_IS_LOWER(optname[0]);
185 o = findopt_name(&optname, NULL, &err);
186 s = optname;
187 optname = NULL;
188 if (*s == '\0' || *s == ' ')
189 {
190 /*
191 * The option name matches exactly.
192 */
193 ;
194 } else if (*s == '=')
195 {
196 /*
197 * The option name is followed by "=value".
198 */
199 if (o != NULL &&
200 (o->otype & OTYPE) != STRING &&
201 (o->otype & OTYPE) != NUMBER)
202 {
203 parg.p_string = printopt;
204 error("The %s option should not be followed by =",
205 &parg);
206 quit(QUIT_ERROR);
207 }
208 s++;
209 } else
210 {
211 /*
212 * The specified name is longer than the
213 * real option name.
214 */
215 o = NULL;
216 }
217 }
218 if (o == NULL)
219 {
220 parg.p_string = printopt;
221 if (err == OPT_AMBIG)
222 error("%s is an ambiguous abbreviation (\"less --help\" for help)",
223 &parg);
224 else
225 error("There is no %s option (\"less --help\" for help)",
226 &parg);
227 quit(QUIT_ERROR);
228 }
229
230 str = NULL;
231 switch (o->otype & OTYPE)
232 {
233 case BOOL:
234 if (set_default)
235 *(o->ovar) = o->odefault;
236 else
237 *(o->ovar) = ! o->odefault;
238 break;
239 case TRIPLE:
240 if (set_default)
241 *(o->ovar) = o->odefault;
242 else
243 *(o->ovar) = flip_triple(o->odefault, lc);
244 break;
245 case STRING:
246 if (*s == '\0')
247 {
248 /*
249 * Set pendopt and return.
250 * We will get the string next time
251 * scan_option is called.
252 */
253 pendopt = o;
254 return;
255 }
256 /*
257 * Don't do anything here.
258 * All processing of STRING options is done by
259 * the handling function.
260 */
261 while (*s == ' ')
262 s++;
263 s = optstring(s, &str, printopt, o->odesc[1]);
264 break;
265 case NUMBER:
266 if (*s == '\0')
267 {
268 pendopt = o;
269 return;
270 }
271 *(o->ovar) = getnum(&s, printopt, (int*)NULL);
272 break;
273 }
274 /*
275 * If the option has a handling function, call it.
276 */
277 if (o->ofunc != NULL)
278 (*o->ofunc)(INIT, str);
279 }
280}
281
282/*
283 * Toggle command line flags from within the program.
284 * Used by the "-" and "_" commands.
285 * how_toggle may be:
286 * OPT_NO_TOGGLE just report the current setting, without changing it.
287 * OPT_TOGGLE invert the current setting
288 * OPT_UNSET set to the default value
289 * OPT_SET set to the inverse of the default value
290 */
291 public void
292toggle_option(o, lower, s, how_toggle)
293 struct loption *o;
294 int lower;
295 char *s;
296 int how_toggle;
297{
298 register int num;
299 int no_prompt;
300 int err;
301 PARG parg;
302
303 no_prompt = (how_toggle & OPT_NO_PROMPT);
304 how_toggle &= ~OPT_NO_PROMPT;
305
306 if (o == NULL)
307 {
308 error("No such option", NULL_PARG);
309 return;
310 }
311
312 if (how_toggle == OPT_TOGGLE && (o->otype & NO_TOGGLE))
313 {
314 parg.p_string = opt_desc(o);
315 error("Cannot change the %s option", &parg);
316 return;
317 }
318
319 if (how_toggle == OPT_NO_TOGGLE && (o->otype & NO_QUERY))
320 {
321 parg.p_string = opt_desc(o);
322 error("Cannot query the %s option", &parg);
323 return;
324 }
325
326 /*
327 * Check for something which appears to be a do_toggle
328 * (because the "-" command was used), but really is not.
329 * This could be a string option with no string, or
330 * a number option with no number.
331 */
332 switch (o->otype & OTYPE)
333 {
334 case STRING:
335 case NUMBER:
336 if (how_toggle == OPT_TOGGLE && *s == '\0')
337 how_toggle = OPT_NO_TOGGLE;
338 break;
339 }
340
341#if HILITE_SEARCH
342 if (how_toggle != OPT_NO_TOGGLE && (o->otype & HL_REPAINT))
343 repaint_hilite(0);
344#endif
345
346 /*
347 * Now actually toggle (change) the variable.
348 */
349 if (how_toggle != OPT_NO_TOGGLE)
350 {
351 switch (o->otype & OTYPE)
352 {
353 case BOOL:
354 /*
355 * Boolean.
356 */
357 switch (how_toggle)
358 {
359 case OPT_TOGGLE:
360 *(o->ovar) = ! *(o->ovar);
361 break;
362 case OPT_UNSET:
363 *(o->ovar) = o->odefault;
364 break;
365 case OPT_SET:
366 *(o->ovar) = ! o->odefault;
367 break;
368 }
369 break;
370 case TRIPLE:
371 /*
372 * Triple:
373 * If user gave the lower case letter, then switch
374 * to 1 unless already 1, in which case make it 0.
375 * If user gave the upper case letter, then switch
376 * to 2 unless already 2, in which case make it 0.
377 */
378 switch (how_toggle)
379 {
380 case OPT_TOGGLE:
381 *(o->ovar) = flip_triple(*(o->ovar), lower);
382 break;
383 case OPT_UNSET:
384 *(o->ovar) = o->odefault;
385 break;
386 case OPT_SET:
387 *(o->ovar) = flip_triple(o->odefault, lower);
388 break;
389 }
390 break;
391 case STRING:
392 /*
393 * String: don't do anything here.
394 * The handling function will do everything.
395 */
396 switch (how_toggle)
397 {
398 case OPT_SET:
399 case OPT_UNSET:
400 error("Cannot use \"-+\" or \"--\" for a string option",
401 NULL_PARG);
402 return;
403 }
404 break;
405 case NUMBER:
406 /*
407 * Number: set the variable to the given number.
408 */
409 switch (how_toggle)
410 {
411 case OPT_TOGGLE:
412 num = getnum(&s, NULL, &err);
413 if (!err)
414 *(o->ovar) = num;
415 break;
416 case OPT_UNSET:
417 *(o->ovar) = o->odefault;
418 break;
419 case OPT_SET:
420 error("Can't use \"-!\" for a numeric option",
421 NULL_PARG);
422 return;
423 }
424 break;
425 }
426 }
427
428 /*
429 * Call the handling function for any special action
430 * specific to this option.
431 */
432 if (o->ofunc != NULL)
433 (*o->ofunc)((how_toggle==OPT_NO_TOGGLE) ? QUERY : TOGGLE, s);
434
435#if HILITE_SEARCH
436 if (how_toggle != OPT_NO_TOGGLE && (o->otype & HL_REPAINT))
437 chg_hilite();
438#endif
439
440 if (!no_prompt)
441 {
442 /*
443 * Print a message describing the new setting.
444 */
445 switch (o->otype & OTYPE)
446 {
447 case BOOL:
448 case TRIPLE:
449 /*
450 * Print the odesc message.
451 */
452 error(o->odesc[*(o->ovar)], NULL_PARG);
453 break;
454 case NUMBER:
455 /*
456 * The message is in odesc[1] and has a %d for
457 * the value of the variable.
458 */
459 parg.p_int = *(o->ovar);
460 error(o->odesc[1], &parg);
461 break;
462 case STRING:
463 /*
464 * Message was already printed by the handling function.
465 */
466 break;
467 }
468 }
469
470 if (how_toggle != OPT_NO_TOGGLE && (o->otype & REPAINT))
471 screen_trashed = TRUE;
472}
473
474/*
475 * "Toggle" a triple-valued option.
476 */
477 static int
478flip_triple(val, lc)
479 int val;
480 int lc;
481{
482 if (lc)
483 return ((val == OPT_ON) ? OPT_OFF : OPT_ON);
484 else
485 return ((val == OPT_ONPLUS) ? OPT_OFF : OPT_ONPLUS);
486}
487
488/*
489 * Determine if an option takes a parameter.
490 */
491 public int
492opt_has_param(o)
493 struct loption *o;
494{
495 if (o == NULL)
496 return (0);
497 if (o->otype & (BOOL|TRIPLE|NOVAR|NO_TOGGLE))
498 return (0);
499 return (1);
500}
501
502/*
503 * Return the prompt to be used for a given option letter.
504 * Only string and number valued options have prompts.
505 */
506 public char *
507opt_prompt(o)
508 struct loption *o;
509{
510 if (o == NULL || (o->otype & (STRING|NUMBER)) == 0)
511 return ("?");
512 return (o->odesc[0]);
513}
514
515/*
516 * Return whether or not there is a string option pending;
517 * that is, if the previous option was a string-valued option letter
518 * (like -P) without a following string.
519 * In that case, the current option is taken to be the string for
520 * the previous option.
521 */
522 public int
523isoptpending()
524{
525 return (pendopt != NULL);
526}
527
528/*
529 * Print error message about missing string.
530 */
531 static void
532nostring(printopt)
533 char *printopt;
534{
535 PARG parg;
536 parg.p_string = printopt;
537 error("Value is required after %s", &parg);
538}
539
540/*
541 * Print error message if a STRING type option is not followed by a string.
542 */
543 public void
544nopendopt()
545{
546 nostring(opt_desc(pendopt));
547}
548
549/*
550 * Scan to end of string or to an END_OPTION_STRING character.
551 * In the latter case, replace the char with a null char.
552 * Return a pointer to the remainder of the string, if any.
553 */
554 static char *
555optstring(s, p_str, printopt, validchars)
556 char *s;
557 char **p_str;
558 char *printopt;
559 char *validchars;
560{
561 register char *p;
562
563 if (*s == '\0')
564 {
565 nostring(printopt);
566 quit(QUIT_ERROR);
567 }
568 *p_str = s;
569 for (p = s; *p != '\0'; p++)
570 {
571 if (*p == END_OPTION_STRING ||
572 (validchars != NULL && strchr(validchars, *p) == NULL))
573 {
574 switch (*p)
575 {
576 case END_OPTION_STRING:
577 case ' ': case '\t': case '-':
578 /* Replace the char with a null to terminate string. */
579 *p++ = '\0';
580 break;
581 default:
582 /* Cannot replace char; make a copy of the string. */
583 *p_str = (char *) ecalloc(p-s+1, sizeof(char));
584 strncpy(*p_str, s, p-s);
585 (*p_str)[p-s] = '\0';
586 break;
587 }
588 break;
589 }
590 }
591 return (p);
592}
593
594/*
595 */
596 static int
597num_error(printopt, errp)
598 char *printopt;
599 int *errp;
600{
601 PARG parg;
602
603 if (errp != NULL)
604 {
605 *errp = TRUE;
606 return (-1);
607 }
608 if (printopt != NULL)
609 {
610 parg.p_string = printopt;
611 error("Number is required after %s", &parg);
612 }
613 quit(QUIT_ERROR);
614 /* NOTREACHED */
615 return (-1);
616}
617
618/*
619 * Translate a string into a number.
620 * Like atoi(), but takes a pointer to a char *, and updates
621 * the char * to point after the translated number.
622 */
623 public int
624getnum(sp, printopt, errp)
625 char **sp;
626 char *printopt;
627 int *errp;
628{
629 register char *s;
630 register int n;
631 register int neg;
632
633 s = skipsp(*sp);
634 neg = FALSE;
635 if (*s == '-')
636 {
637 neg = TRUE;
638 s++;
639 }
640 if (*s < '0' || *s > '9')
641 return (num_error(printopt, errp));
642
643 n = 0;
644 while (*s >= '0' && *s <= '9')
645 n = 10 * n + *s++ - '0';
646 *sp = s;
647 if (errp != NULL)
648 *errp = FALSE;
649 if (neg)
650 n = -n;
651 return (n);
652}
653
654/*
655 * Translate a string into a fraction, represented by the part of a
656 * number which would follow a decimal point.
657 * The value of the fraction is returned as parts per NUM_FRAC_DENOM.
658 * That is, if "n" is returned, the fraction intended is n/NUM_FRAC_DENOM.
659 */
660 public long
661getfraction(sp, printopt, errp)
662 char **sp;
663 char *printopt;
664 int *errp;
665{
666 register char *s;
667 long frac = 0;
668 int fraclen = 0;
669
670 s = skipsp(*sp);
671 if (*s < '0' || *s > '9')
672 return (num_error(printopt, errp));
673
674 for ( ; *s >= '0' && *s <= '9'; s++)
675 {
676 frac = (frac * 10) + (*s - '0');
677 fraclen++;
678 }
679 if (fraclen > NUM_LOG_FRAC_DENOM)
680 while (fraclen-- > NUM_LOG_FRAC_DENOM)
681 frac /= 10;
682 else
683 while (fraclen++ < NUM_LOG_FRAC_DENOM)
684 frac *= 10;
685 *sp = s;
686 if (errp != NULL)
687 *errp = FALSE;
688 return (frac);
689}
690
691
692/*
693 * Get the value of the -e flag.
694 */
695 public int
696get_quit_at_eof()
697{
698 if (!less_is_more)
699 return quit_at_eof;
700 /* When less_is_more is set, the -e flag semantics are different. */
701 return quit_at_eof ? OPT_ON : OPT_ONPLUS;
702}
8 */
9
10
11/*
12 * Process command line options.
13 *
14 * Each option is a single letter which controls a program variable.
15 * The options have defaults which may be changed via
16 * the command line option, toggled via the "-" command,
17 * or queried via the "_" command.
18 */
19
20#include "less.h"
21#include "option.h"
22
23static struct loption *pendopt;
24public int plusoption = FALSE;
25
26static char *optstring();
27static int flip_triple();
28
29extern int screen_trashed;
30extern int less_is_more;
31extern int quit_at_eof;
32extern char *every_first_cmd;
33
34/*
35 * Return a printable description of an option.
36 */
37 static char *
38opt_desc(o)
39 struct loption *o;
40{
41 static char buf[OPTNAME_MAX + 10];
42 if (o->oletter == OLETTER_NONE)
43 SNPRINTF1(buf, sizeof(buf), "--%s", o->onames->oname);
44 else
45 SNPRINTF2(buf, sizeof(buf), "-%c (--%s)", o->oletter, o->onames->oname);
46 return (buf);
47}
48
49/*
50 * Return a string suitable for printing as the "name" of an option.
51 * For example, if the option letter is 'x', just return "-x".
52 */
53 public char *
54propt(c)
55 int c;
56{
57 static char buf[8];
58
59 sprintf(buf, "-%s", prchar(c));
60 return (buf);
61}
62
63/*
64 * Scan an argument (either from the command line or from the
65 * LESS environment variable) and process it.
66 */
67 public void
68scan_option(s)
69 char *s;
70{
71 register struct loption *o;
72 register int optc;
73 char *optname;
74 char *printopt;
75 char *str;
76 int set_default;
77 int lc;
78 int err;
79 PARG parg;
80
81 if (s == NULL)
82 return;
83
84 /*
85 * If we have a pending option which requires an argument,
86 * handle it now.
87 * This happens if the previous option was, for example, "-P"
88 * without a following string. In that case, the current
89 * option is simply the argument for the previous option.
90 */
91 if (pendopt != NULL)
92 {
93 switch (pendopt->otype & OTYPE)
94 {
95 case STRING:
96 (*pendopt->ofunc)(INIT, s);
97 break;
98 case NUMBER:
99 printopt = opt_desc(pendopt);
100 *(pendopt->ovar) = getnum(&s, printopt, (int*)NULL);
101 break;
102 }
103 pendopt = NULL;
104 return;
105 }
106
107 set_default = FALSE;
108 optname = NULL;
109
110 while (*s != '\0')
111 {
112 /*
113 * Check some special cases first.
114 */
115 switch (optc = *s++)
116 {
117 case ' ':
118 case '\t':
119 case END_OPTION_STRING:
120 continue;
121 case '-':
122 /*
123 * "--" indicates an option name instead of a letter.
124 */
125 if (*s == '-')
126 {
127 optname = ++s;
128 break;
129 }
130 /*
131 * "-+" means set these options back to their defaults.
132 * (They may have been set otherwise by previous
133 * options.)
134 */
135 set_default = (*s == '+');
136 if (set_default)
137 s++;
138 continue;
139 case '+':
140 /*
141 * An option prefixed by a "+" is ungotten, so
142 * that it is interpreted as less commands
143 * processed at the start of the first input file.
144 * "++" means process the commands at the start of
145 * EVERY input file.
146 */
147 plusoption = TRUE;
148 s = optstring(s, &str, propt('+'), NULL);
149 if (*str == '+')
150 every_first_cmd = save(++str);
151 else
152 ungetsc(str);
153 continue;
154 case '0': case '1': case '2': case '3': case '4':
155 case '5': case '6': case '7': case '8': case '9':
156 /*
157 * Special "more" compatibility form "-<number>"
158 * instead of -z<number> to set the scrolling
159 * window size.
160 */
161 s--;
162 optc = 'z';
163 break;
164 case 'n':
165 if (less_is_more)
166 optc = 'z';
167 break;
168 }
169
170 /*
171 * Not a special case.
172 * Look up the option letter in the option table.
173 */
174 err = 0;
175 if (optname == NULL)
176 {
177 printopt = propt(optc);
178 lc = ASCII_IS_LOWER(optc);
179 o = findopt(optc);
180 } else
181 {
182 printopt = optname;
183 lc = ASCII_IS_LOWER(optname[0]);
184 o = findopt_name(&optname, NULL, &err);
185 s = optname;
186 optname = NULL;
187 if (*s == '\0' || *s == ' ')
188 {
189 /*
190 * The option name matches exactly.
191 */
192 ;
193 } else if (*s == '=')
194 {
195 /*
196 * The option name is followed by "=value".
197 */
198 if (o != NULL &&
199 (o->otype & OTYPE) != STRING &&
200 (o->otype & OTYPE) != NUMBER)
201 {
202 parg.p_string = printopt;
203 error("The %s option should not be followed by =",
204 &parg);
205 quit(QUIT_ERROR);
206 }
207 s++;
208 } else
209 {
210 /*
211 * The specified name is longer than the
212 * real option name.
213 */
214 o = NULL;
215 }
216 }
217 if (o == NULL)
218 {
219 parg.p_string = printopt;
220 if (err == OPT_AMBIG)
221 error("%s is an ambiguous abbreviation (\"less --help\" for help)",
222 &parg);
223 else
224 error("There is no %s option (\"less --help\" for help)",
225 &parg);
226 quit(QUIT_ERROR);
227 }
228
229 str = NULL;
230 switch (o->otype & OTYPE)
231 {
232 case BOOL:
233 if (set_default)
234 *(o->ovar) = o->odefault;
235 else
236 *(o->ovar) = ! o->odefault;
237 break;
238 case TRIPLE:
239 if (set_default)
240 *(o->ovar) = o->odefault;
241 else
242 *(o->ovar) = flip_triple(o->odefault, lc);
243 break;
244 case STRING:
245 if (*s == '\0')
246 {
247 /*
248 * Set pendopt and return.
249 * We will get the string next time
250 * scan_option is called.
251 */
252 pendopt = o;
253 return;
254 }
255 /*
256 * Don't do anything here.
257 * All processing of STRING options is done by
258 * the handling function.
259 */
260 while (*s == ' ')
261 s++;
262 s = optstring(s, &str, printopt, o->odesc[1]);
263 break;
264 case NUMBER:
265 if (*s == '\0')
266 {
267 pendopt = o;
268 return;
269 }
270 *(o->ovar) = getnum(&s, printopt, (int*)NULL);
271 break;
272 }
273 /*
274 * If the option has a handling function, call it.
275 */
276 if (o->ofunc != NULL)
277 (*o->ofunc)(INIT, str);
278 }
279}
280
281/*
282 * Toggle command line flags from within the program.
283 * Used by the "-" and "_" commands.
284 * how_toggle may be:
285 * OPT_NO_TOGGLE just report the current setting, without changing it.
286 * OPT_TOGGLE invert the current setting
287 * OPT_UNSET set to the default value
288 * OPT_SET set to the inverse of the default value
289 */
290 public void
291toggle_option(o, lower, s, how_toggle)
292 struct loption *o;
293 int lower;
294 char *s;
295 int how_toggle;
296{
297 register int num;
298 int no_prompt;
299 int err;
300 PARG parg;
301
302 no_prompt = (how_toggle & OPT_NO_PROMPT);
303 how_toggle &= ~OPT_NO_PROMPT;
304
305 if (o == NULL)
306 {
307 error("No such option", NULL_PARG);
308 return;
309 }
310
311 if (how_toggle == OPT_TOGGLE && (o->otype & NO_TOGGLE))
312 {
313 parg.p_string = opt_desc(o);
314 error("Cannot change the %s option", &parg);
315 return;
316 }
317
318 if (how_toggle == OPT_NO_TOGGLE && (o->otype & NO_QUERY))
319 {
320 parg.p_string = opt_desc(o);
321 error("Cannot query the %s option", &parg);
322 return;
323 }
324
325 /*
326 * Check for something which appears to be a do_toggle
327 * (because the "-" command was used), but really is not.
328 * This could be a string option with no string, or
329 * a number option with no number.
330 */
331 switch (o->otype & OTYPE)
332 {
333 case STRING:
334 case NUMBER:
335 if (how_toggle == OPT_TOGGLE && *s == '\0')
336 how_toggle = OPT_NO_TOGGLE;
337 break;
338 }
339
340#if HILITE_SEARCH
341 if (how_toggle != OPT_NO_TOGGLE && (o->otype & HL_REPAINT))
342 repaint_hilite(0);
343#endif
344
345 /*
346 * Now actually toggle (change) the variable.
347 */
348 if (how_toggle != OPT_NO_TOGGLE)
349 {
350 switch (o->otype & OTYPE)
351 {
352 case BOOL:
353 /*
354 * Boolean.
355 */
356 switch (how_toggle)
357 {
358 case OPT_TOGGLE:
359 *(o->ovar) = ! *(o->ovar);
360 break;
361 case OPT_UNSET:
362 *(o->ovar) = o->odefault;
363 break;
364 case OPT_SET:
365 *(o->ovar) = ! o->odefault;
366 break;
367 }
368 break;
369 case TRIPLE:
370 /*
371 * Triple:
372 * If user gave the lower case letter, then switch
373 * to 1 unless already 1, in which case make it 0.
374 * If user gave the upper case letter, then switch
375 * to 2 unless already 2, in which case make it 0.
376 */
377 switch (how_toggle)
378 {
379 case OPT_TOGGLE:
380 *(o->ovar) = flip_triple(*(o->ovar), lower);
381 break;
382 case OPT_UNSET:
383 *(o->ovar) = o->odefault;
384 break;
385 case OPT_SET:
386 *(o->ovar) = flip_triple(o->odefault, lower);
387 break;
388 }
389 break;
390 case STRING:
391 /*
392 * String: don't do anything here.
393 * The handling function will do everything.
394 */
395 switch (how_toggle)
396 {
397 case OPT_SET:
398 case OPT_UNSET:
399 error("Cannot use \"-+\" or \"--\" for a string option",
400 NULL_PARG);
401 return;
402 }
403 break;
404 case NUMBER:
405 /*
406 * Number: set the variable to the given number.
407 */
408 switch (how_toggle)
409 {
410 case OPT_TOGGLE:
411 num = getnum(&s, NULL, &err);
412 if (!err)
413 *(o->ovar) = num;
414 break;
415 case OPT_UNSET:
416 *(o->ovar) = o->odefault;
417 break;
418 case OPT_SET:
419 error("Can't use \"-!\" for a numeric option",
420 NULL_PARG);
421 return;
422 }
423 break;
424 }
425 }
426
427 /*
428 * Call the handling function for any special action
429 * specific to this option.
430 */
431 if (o->ofunc != NULL)
432 (*o->ofunc)((how_toggle==OPT_NO_TOGGLE) ? QUERY : TOGGLE, s);
433
434#if HILITE_SEARCH
435 if (how_toggle != OPT_NO_TOGGLE && (o->otype & HL_REPAINT))
436 chg_hilite();
437#endif
438
439 if (!no_prompt)
440 {
441 /*
442 * Print a message describing the new setting.
443 */
444 switch (o->otype & OTYPE)
445 {
446 case BOOL:
447 case TRIPLE:
448 /*
449 * Print the odesc message.
450 */
451 error(o->odesc[*(o->ovar)], NULL_PARG);
452 break;
453 case NUMBER:
454 /*
455 * The message is in odesc[1] and has a %d for
456 * the value of the variable.
457 */
458 parg.p_int = *(o->ovar);
459 error(o->odesc[1], &parg);
460 break;
461 case STRING:
462 /*
463 * Message was already printed by the handling function.
464 */
465 break;
466 }
467 }
468
469 if (how_toggle != OPT_NO_TOGGLE && (o->otype & REPAINT))
470 screen_trashed = TRUE;
471}
472
473/*
474 * "Toggle" a triple-valued option.
475 */
476 static int
477flip_triple(val, lc)
478 int val;
479 int lc;
480{
481 if (lc)
482 return ((val == OPT_ON) ? OPT_OFF : OPT_ON);
483 else
484 return ((val == OPT_ONPLUS) ? OPT_OFF : OPT_ONPLUS);
485}
486
487/*
488 * Determine if an option takes a parameter.
489 */
490 public int
491opt_has_param(o)
492 struct loption *o;
493{
494 if (o == NULL)
495 return (0);
496 if (o->otype & (BOOL|TRIPLE|NOVAR|NO_TOGGLE))
497 return (0);
498 return (1);
499}
500
501/*
502 * Return the prompt to be used for a given option letter.
503 * Only string and number valued options have prompts.
504 */
505 public char *
506opt_prompt(o)
507 struct loption *o;
508{
509 if (o == NULL || (o->otype & (STRING|NUMBER)) == 0)
510 return ("?");
511 return (o->odesc[0]);
512}
513
514/*
515 * Return whether or not there is a string option pending;
516 * that is, if the previous option was a string-valued option letter
517 * (like -P) without a following string.
518 * In that case, the current option is taken to be the string for
519 * the previous option.
520 */
521 public int
522isoptpending()
523{
524 return (pendopt != NULL);
525}
526
527/*
528 * Print error message about missing string.
529 */
530 static void
531nostring(printopt)
532 char *printopt;
533{
534 PARG parg;
535 parg.p_string = printopt;
536 error("Value is required after %s", &parg);
537}
538
539/*
540 * Print error message if a STRING type option is not followed by a string.
541 */
542 public void
543nopendopt()
544{
545 nostring(opt_desc(pendopt));
546}
547
548/*
549 * Scan to end of string or to an END_OPTION_STRING character.
550 * In the latter case, replace the char with a null char.
551 * Return a pointer to the remainder of the string, if any.
552 */
553 static char *
554optstring(s, p_str, printopt, validchars)
555 char *s;
556 char **p_str;
557 char *printopt;
558 char *validchars;
559{
560 register char *p;
561
562 if (*s == '\0')
563 {
564 nostring(printopt);
565 quit(QUIT_ERROR);
566 }
567 *p_str = s;
568 for (p = s; *p != '\0'; p++)
569 {
570 if (*p == END_OPTION_STRING ||
571 (validchars != NULL && strchr(validchars, *p) == NULL))
572 {
573 switch (*p)
574 {
575 case END_OPTION_STRING:
576 case ' ': case '\t': case '-':
577 /* Replace the char with a null to terminate string. */
578 *p++ = '\0';
579 break;
580 default:
581 /* Cannot replace char; make a copy of the string. */
582 *p_str = (char *) ecalloc(p-s+1, sizeof(char));
583 strncpy(*p_str, s, p-s);
584 (*p_str)[p-s] = '\0';
585 break;
586 }
587 break;
588 }
589 }
590 return (p);
591}
592
593/*
594 */
595 static int
596num_error(printopt, errp)
597 char *printopt;
598 int *errp;
599{
600 PARG parg;
601
602 if (errp != NULL)
603 {
604 *errp = TRUE;
605 return (-1);
606 }
607 if (printopt != NULL)
608 {
609 parg.p_string = printopt;
610 error("Number is required after %s", &parg);
611 }
612 quit(QUIT_ERROR);
613 /* NOTREACHED */
614 return (-1);
615}
616
617/*
618 * Translate a string into a number.
619 * Like atoi(), but takes a pointer to a char *, and updates
620 * the char * to point after the translated number.
621 */
622 public int
623getnum(sp, printopt, errp)
624 char **sp;
625 char *printopt;
626 int *errp;
627{
628 register char *s;
629 register int n;
630 register int neg;
631
632 s = skipsp(*sp);
633 neg = FALSE;
634 if (*s == '-')
635 {
636 neg = TRUE;
637 s++;
638 }
639 if (*s < '0' || *s > '9')
640 return (num_error(printopt, errp));
641
642 n = 0;
643 while (*s >= '0' && *s <= '9')
644 n = 10 * n + *s++ - '0';
645 *sp = s;
646 if (errp != NULL)
647 *errp = FALSE;
648 if (neg)
649 n = -n;
650 return (n);
651}
652
653/*
654 * Translate a string into a fraction, represented by the part of a
655 * number which would follow a decimal point.
656 * The value of the fraction is returned as parts per NUM_FRAC_DENOM.
657 * That is, if "n" is returned, the fraction intended is n/NUM_FRAC_DENOM.
658 */
659 public long
660getfraction(sp, printopt, errp)
661 char **sp;
662 char *printopt;
663 int *errp;
664{
665 register char *s;
666 long frac = 0;
667 int fraclen = 0;
668
669 s = skipsp(*sp);
670 if (*s < '0' || *s > '9')
671 return (num_error(printopt, errp));
672
673 for ( ; *s >= '0' && *s <= '9'; s++)
674 {
675 frac = (frac * 10) + (*s - '0');
676 fraclen++;
677 }
678 if (fraclen > NUM_LOG_FRAC_DENOM)
679 while (fraclen-- > NUM_LOG_FRAC_DENOM)
680 frac /= 10;
681 else
682 while (fraclen++ < NUM_LOG_FRAC_DENOM)
683 frac *= 10;
684 *sp = s;
685 if (errp != NULL)
686 *errp = FALSE;
687 return (frac);
688}
689
690
691/*
692 * Get the value of the -e flag.
693 */
694 public int
695get_quit_at_eof()
696{
697 if (!less_is_more)
698 return quit_at_eof;
699 /* When less_is_more is set, the -e flag semantics are different. */
700 return quit_at_eof ? OPT_ON : OPT_ONPLUS;
701}