2 * Copyright (C) 1984-2023 Mark Nudelman
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.
7 * For more information, see the README file.
12 * Process command line options.
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.
23 static struct loption
*pendopt
;
24 public int plusoption
= FALSE
;
26 static char *optstring(char *s
, char **p_str
, char *printopt
, char *validchars
);
27 static int flip_triple(int val
, int lc
);
29 extern int screen_trashed
;
30 extern int less_is_more
;
31 extern int quit_at_eof
;
32 extern char *every_first_cmd
;
33 extern int opt_use_backslash
;
36 * Return a printable description of an option.
38 static char * opt_desc(struct loption
*o
)
40 static char buf
[OPTNAME_MAX
+ 10];
41 if (o
->oletter
== OLETTER_NONE
)
42 SNPRINTF1(buf
, sizeof(buf
), "--%s", o
->onames
->oname
);
44 SNPRINTF2(buf
, sizeof(buf
), "-%c (--%s)", o
->oletter
, o
->onames
->oname
);
49 * Return a string suitable for printing as the "name" of an option.
50 * For example, if the option letter is 'x', just return "-x".
52 public char * propt(int c
)
54 static char buf
[MAX_PRCHAR_LEN
+2];
56 sprintf(buf
, "-%s", prchar(c
));
61 * Scan an argument (either from the command line or from the
62 * LESS environment variable) and process it.
64 public void scan_option(char *s
)
80 * If we have a pending option which requires an argument,
82 * This happens if the previous option was, for example, "-P"
83 * without a following string. In that case, the current
84 * option is simply the argument for the previous option.
88 switch (pendopt
->otype
& OTYPE
)
91 (*pendopt
->ofunc
)(INIT
, s
);
94 printopt
= opt_desc(pendopt
);
95 *(pendopt
->ovar
) = getnum(&s
, printopt
, (int*)NULL
);
108 * Check some special cases first.
114 case END_OPTION_STRING
:
118 * "--" indicates an option name instead of a letter.
126 * "-+" means set these options back to their defaults.
127 * (They may have been set otherwise by previous
130 set_default
= (*s
== '+');
136 * An option prefixed by a "+" is ungotten, so
137 * that it is interpreted as less commands
138 * processed at the start of the first input file.
139 * "++" means process the commands at the start of
143 s
= optstring(s
, &str
, propt('+'), NULL
);
148 if (every_first_cmd
!= NULL
)
149 free(every_first_cmd
);
150 every_first_cmd
= save(str
+1);
154 ungetcc_back(CHAR_END_COMMAND
);
158 case '0': case '1': case '2': case '3': case '4':
159 case '5': case '6': case '7': case '8': case '9':
161 * Special "more" compatibility form "-<number>"
162 * instead of -z<number> to set the scrolling
175 * Not a special case.
176 * Look up the option letter in the option table.
181 printopt
= propt(optc
);
182 lc
= ASCII_IS_LOWER(optc
);
187 lc
= ASCII_IS_LOWER(optname
[0]);
188 o
= findopt_name(&optname
, NULL
, &err
);
191 if (*s
== '\0' || *s
== ' ')
194 * The option name matches exactly.
197 } else if (*s
== '=')
200 * The option name is followed by "=value".
203 (o
->otype
& OTYPE
) != STRING
&&
204 (o
->otype
& OTYPE
) != NUMBER
)
206 parg
.p_string
= printopt
;
207 error("The %s option should not be followed by =",
215 * The specified name is longer than the
223 parg
.p_string
= printopt
;
224 if (err
== OPT_AMBIG
)
225 error("%s is an ambiguous abbreviation (\"less --help\" for help)",
228 error("There is no %s option (\"less --help\" for help)",
234 switch (o
->otype
& OTYPE
)
238 *(o
->ovar
) = o
->odefault
;
240 *(o
->ovar
) = ! o
->odefault
;
244 *(o
->ovar
) = o
->odefault
;
246 *(o
->ovar
) = flip_triple(o
->odefault
, lc
);
252 * Set pendopt and return.
253 * We will get the string next time
254 * scan_option is called.
260 * Don't do anything here.
261 * All processing of STRING options is done by
262 * the handling function.
266 s
= optstring(s
, &str
, printopt
, o
->odesc
[1]);
276 *(o
->ovar
) = getnum(&s
, printopt
, (int*)NULL
);
280 * If the option has a handling function, call it.
282 if (o
->ofunc
!= NULL
)
283 (*o
->ofunc
)(INIT
, str
);
290 * Toggle command line flags from within the program.
291 * Used by the "-" and "_" commands.
293 * OPT_NO_TOGGLE just report the current setting, without changing it.
294 * OPT_TOGGLE invert the current setting
295 * OPT_UNSET set to the default value
296 * OPT_SET set to the inverse of the default value
298 public void toggle_option(struct loption
*o
, int lower
, char *s
, int how_toggle
)
305 no_prompt
= (how_toggle
& OPT_NO_PROMPT
);
306 how_toggle
&= ~OPT_NO_PROMPT
;
310 error("No such option", NULL_PARG
);
314 if (how_toggle
== OPT_TOGGLE
&& (o
->otype
& NO_TOGGLE
))
316 parg
.p_string
= opt_desc(o
);
317 error("Cannot change the %s option", &parg
);
321 if (how_toggle
== OPT_NO_TOGGLE
&& (o
->otype
& NO_QUERY
))
323 parg
.p_string
= opt_desc(o
);
324 error("Cannot query the %s option", &parg
);
329 * Check for something which appears to be a do_toggle
330 * (because the "-" command was used), but really is not.
331 * This could be a string option with no string, or
332 * a number option with no number.
334 switch (o
->otype
& OTYPE
)
338 if (how_toggle
== OPT_TOGGLE
&& *s
== '\0')
339 how_toggle
= OPT_NO_TOGGLE
;
344 if (how_toggle
!= OPT_NO_TOGGLE
&& (o
->otype
& HL_REPAINT
))
349 * Now actually toggle (change) the variable.
351 if (how_toggle
!= OPT_NO_TOGGLE
)
353 switch (o
->otype
& OTYPE
)
362 *(o
->ovar
) = ! *(o
->ovar
);
365 *(o
->ovar
) = o
->odefault
;
368 *(o
->ovar
) = ! o
->odefault
;
375 * If user gave the lower case letter, then switch
376 * to 1 unless already 1, in which case make it 0.
377 * If user gave the upper case letter, then switch
378 * to 2 unless already 2, in which case make it 0.
383 *(o
->ovar
) = flip_triple(*(o
->ovar
), lower
);
386 *(o
->ovar
) = o
->odefault
;
389 *(o
->ovar
) = flip_triple(o
->odefault
, lower
);
395 * String: don't do anything here.
396 * The handling function will do everything.
402 error("Cannot use \"-+\" or \"--\" for a string option",
409 * Number: set the variable to the given number.
414 num
= getnum(&s
, NULL
, &err
);
419 *(o
->ovar
) = o
->odefault
;
422 error("Can't use \"-!\" for a numeric option",
431 * Call the handling function for any special action
432 * specific to this option.
434 if (o
->ofunc
!= NULL
)
435 (*o
->ofunc
)((how_toggle
==OPT_NO_TOGGLE
) ? QUERY
: TOGGLE
, s
);
438 if (how_toggle
!= OPT_NO_TOGGLE
&& (o
->otype
& HL_REPAINT
))
445 * Print a message describing the new setting.
447 switch (o
->otype
& OTYPE
)
452 * Print the odesc message.
454 error(o
->odesc
[*(o
->ovar
)], NULL_PARG
);
458 * The message is in odesc[1] and has a %d for
459 * the value of the variable.
461 parg
.p_int
= *(o
->ovar
);
462 error(o
->odesc
[1], &parg
);
466 * Message was already printed by the handling function.
472 if (how_toggle
!= OPT_NO_TOGGLE
&& (o
->otype
& REPAINT
))
473 screen_trashed
= TRUE
;
477 * "Toggle" a triple-valued option.
479 static int flip_triple(int val
, int lc
)
482 return ((val
== OPT_ON
) ? OPT_OFF
: OPT_ON
);
484 return ((val
== OPT_ONPLUS
) ? OPT_OFF
: OPT_ONPLUS
);
488 * Determine if an option takes a parameter.
490 public int opt_has_param(struct loption
*o
)
494 if (o
->otype
& (BOOL
|TRIPLE
|NOVAR
|NO_TOGGLE
))
500 * Return the prompt to be used for a given option letter.
501 * Only string and number valued options have prompts.
503 public char * opt_prompt(struct loption
*o
)
505 if (o
== NULL
|| (o
->otype
& (STRING
|NUMBER
)) == 0)
507 return (o
->odesc
[0]);
511 * If the specified option can be toggled, return NULL.
512 * Otherwise return an appropriate error message.
514 public char * opt_toggle_disallowed(int c
)
519 if (ch_getflags() & CH_CANSEEK
)
520 return "Input is not a pipe";
527 * Return whether or not there is a string option pending;
528 * that is, if the previous option was a string-valued option letter
529 * (like -P) without a following string.
530 * In that case, the current option is taken to be the string for
531 * the previous option.
533 public int isoptpending(void)
535 return (pendopt
!= NULL
);
539 * Print error message about missing string.
541 static void nostring(char *printopt
)
544 parg
.p_string
= printopt
;
545 error("Value is required after %s", &parg
);
549 * Print error message if a STRING type option is not followed by a string.
551 public void nopendopt(void)
553 nostring(opt_desc(pendopt
));
557 * Scan to end of string or to an END_OPTION_STRING character.
558 * In the latter case, replace the char with a null char.
559 * Return a pointer to the remainder of the string, if any.
561 static char * optstring(char *s
, char **p_str
, char *printopt
, char *validchars
)
571 /* Alloc could be more than needed, but not worth trimming. */
572 *p_str
= (char *) ecalloc(strlen(s
)+1, sizeof(char));
575 for (p
= s
; *p
!= '\0'; p
++)
577 if (opt_use_backslash
&& *p
== '\\' && p
[1] != '\0')
579 /* Take next char literally. */
583 if (*p
== END_OPTION_STRING
||
584 (validchars
!= NULL
&& strchr(validchars
, *p
) == NULL
))
585 /* End of option string. */
596 static int num_error(char *printopt
, int *errp
, int overflow
)
605 if (printopt
!= NULL
)
607 parg
.p_string
= printopt
;
609 ? "Number too large in '%s'"
610 : "Number is required after %s"),
617 * Translate a string into a number.
618 * Like atoi(), but takes a pointer to a char *, and updates
619 * the char * to point after the translated number.
621 public int getnum(char **sp
, char *printopt
, int *errp
)
634 if (*s
< '0' || *s
> '9')
635 return (num_error(printopt
, errp
, FALSE
));
637 n
= lstrtoi(s
, sp
, 10);
639 return (num_error(printopt
, errp
, TRUE
));
648 * Translate a string into a fraction, represented by the part of a
649 * number which would follow a decimal point.
650 * The value of the fraction is returned as parts per NUM_FRAC_DENOM.
651 * That is, if "n" is returned, the fraction intended is n/NUM_FRAC_DENOM.
653 public long getfraction(char **sp
, char *printopt
, int *errp
)
660 if (*s
< '0' || *s
> '9')
661 return (num_error(printopt
, errp
, FALSE
));
663 for ( ; *s
>= '0' && *s
<= '9'; s
++)
665 if (NUM_LOG_FRAC_DENOM
<= fraclen
)
667 frac
= (frac
* 10) + (*s
- '0');
670 while (fraclen
++ < NUM_LOG_FRAC_DENOM
)
680 * Get the value of the -e flag.
682 public int get_quit_at_eof(void)
686 /* When less_is_more is set, the -e flag semantics are different. */
687 return quit_at_eof
? OPT_ONPLUS
: OPT_ON
;