mirror of
https://github.com/rkd77/elinks.git
synced 2025-01-03 14:57:44 -05:00
Merge branch 'elinks-0.12' into elinks-0.13
Conflicts: po/pl.po src/config/conf.c src/terminal/kbd.c
This commit is contained in:
commit
5499926cc0
16
NEWS
16
NEWS
@ -31,7 +31,7 @@ To be released as ELinks 0.12.0.
|
|||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
This list now contains all the important changes from ELinks 0.11.0 to
|
This list now contains all the important changes from ELinks 0.11.0 to
|
||||||
ELinks 0.12.GIT (21052fafb0be0b87b7f376fb4edfa70304e365cc) and related
|
ELinks 0.12.GIT (054852ae234342765ab69da0eb95a18f96bd6028) and related
|
||||||
bug numbers. Each section is sorted by severity and grouped by topic.
|
bug numbers. Each section is sorted by severity and grouped by topic.
|
||||||
|
|
||||||
The list no doubt includes several changes that are not really
|
The list no doubt includes several changes that are not really
|
||||||
@ -82,6 +82,7 @@ Miscellaneous:
|
|||||||
* critical bug 869: long mailcap entry buffer overflow (non-security)
|
* critical bug 869: long mailcap entry buffer overflow (non-security)
|
||||||
when downloading
|
when downloading
|
||||||
* tabs opened by -remote now go behind existing dialogs
|
* tabs opened by -remote now go behind existing dialogs
|
||||||
|
* major bug 503: various fixes in parsing and updating of elinks.conf
|
||||||
* Debian bug 257762: turn terminal transparency off by default
|
* Debian bug 257762: turn terminal transparency off by default
|
||||||
* bug 724: better parsing of escape sequences and control
|
* bug 724: better parsing of escape sequences and control
|
||||||
sequences from the terminal
|
sequences from the terminal
|
||||||
@ -131,10 +132,14 @@ Miscellaneous:
|
|||||||
on enter
|
on enter
|
||||||
* enhancement: add support for parsing space separated CSS class
|
* enhancement: add support for parsing space separated CSS class
|
||||||
attribute values
|
attribute values
|
||||||
|
* enhancement: make meta refresh content attribute parsing more tolerant
|
||||||
|
* enhancement: recognize meta http-equiv="cache-control" even if no
|
||||||
|
refresh
|
||||||
* enhancement: mouse wheel support over GPM (contrib/gpm-wheel.patch),
|
* enhancement: mouse wheel support over GPM (contrib/gpm-wheel.patch),
|
||||||
and on BSD via moused -z 4
|
and on BSD via moused -z 4
|
||||||
* enhancement: 24-bit truecolor mode
|
* enhancement: 24-bit truecolor mode
|
||||||
* enhancement 622: -dump-color-mode
|
* enhancement 622: -dump-color-mode
|
||||||
|
* enhancement 994: treat only termios.c_cc[VERASE] as "Backspace"
|
||||||
* enhancement: support Ctrl+Alt+letter key combinations
|
* enhancement: support Ctrl+Alt+letter key combinations
|
||||||
* enhancement 381: reduce memory consumption of codepages and some
|
* enhancement 381: reduce memory consumption of codepages and some
|
||||||
other arrays
|
other arrays
|
||||||
@ -166,6 +171,12 @@ Changes in the experimental ECMAScript support:
|
|||||||
* enhancement: added document.location.href, input.selectedIndex,
|
* enhancement: added document.location.href, input.selectedIndex,
|
||||||
window.setTimeout, window.status
|
window.setTimeout, window.status
|
||||||
|
|
||||||
|
Changes in the experimental NNTP client:
|
||||||
|
|
||||||
|
* HTML escape header field values
|
||||||
|
* Add support for handling RFC2047 encoded words
|
||||||
|
* Improve listing of articles for groups
|
||||||
|
|
||||||
Changes in the experimental SGML/DOM implementation:
|
Changes in the experimental SGML/DOM implementation:
|
||||||
|
|
||||||
* enhancement: minimalistic RSS renderer
|
* enhancement: minimalistic RSS renderer
|
||||||
@ -231,6 +242,8 @@ have already been considered.
|
|||||||
memory
|
memory
|
||||||
- (bugfix 920) assertion priority >= prev_priority failed: queue is
|
- (bugfix 920) assertion priority >= prev_priority failed: queue is
|
||||||
not sorted
|
not sorted
|
||||||
|
- (bugfix 968) assertion width > 0 failed in copy_chars called from
|
||||||
|
justify_line
|
||||||
* Already backported to a previous release but not listed there:
|
* Already backported to a previous release but not listed there:
|
||||||
- (enhancement) Activate link only when onClick returns true.
|
- (enhancement) Activate link only when onClick returns true.
|
||||||
Fixed bug 786 in ELinks 0.11.2.
|
Fixed bug 786 in ELinks 0.11.2.
|
||||||
@ -296,6 +309,7 @@ have already been considered.
|
|||||||
this as part of the UTF-8 support.
|
this as part of the UTF-8 support.
|
||||||
* enhancement in user SMJS: elinks.alert no longer displays as an
|
* enhancement in user SMJS: elinks.alert no longer displays as an
|
||||||
"error"
|
"error"
|
||||||
|
* Really retry forever when connection.retries = 0
|
||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ELinks 0.11.3.GIT now:
|
ELinks 0.11.3.GIT now:
|
||||||
|
@ -57,16 +57,63 @@
|
|||||||
* value to an option, but sometimes you may want to first create the option
|
* value to an option, but sometimes you may want to first create the option
|
||||||
* ;). Then this will come handy. */
|
* ;). Then this will come handy. */
|
||||||
|
|
||||||
|
struct conf_parsing_state {
|
||||||
|
/** This part may be copied to a local variable as a bookmark
|
||||||
|
* and restored later. So it must not contain any pointers
|
||||||
|
* that would have to be freed in that situation. */
|
||||||
|
struct conf_parsing_pos {
|
||||||
|
/** Points to the next character to be parsed from the
|
||||||
|
* configuration file. */
|
||||||
|
unsigned char *look;
|
||||||
|
|
||||||
/* Skip comments and whitespace,
|
/** The line number corresponding to #look. This is
|
||||||
* setting *@line to the number of lines skipped. */
|
* shown in error messages. */
|
||||||
static unsigned char *
|
int line;
|
||||||
skip_white(unsigned char *start, int *line)
|
} pos;
|
||||||
|
|
||||||
|
/** When ELinks is rewriting the configuration file, @c mirrored
|
||||||
|
* indicates the end of the part that has already been copied
|
||||||
|
* to the mirror string. Otherwise, @c mirrored is not used.
|
||||||
|
*
|
||||||
|
* @invariant @c mirrored @<= @c pos.look */
|
||||||
|
unsigned char *mirrored;
|
||||||
|
|
||||||
|
/** File name for error messages. If NULL then do not display
|
||||||
|
* error messages. */
|
||||||
|
const unsigned char *filename;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Tell the user about an error in the configuration file.
|
||||||
|
* @return @a err, for convenience. */
|
||||||
|
static enum parse_error
|
||||||
|
show_parse_error(const struct conf_parsing_state *state, enum parse_error err)
|
||||||
{
|
{
|
||||||
|
static const unsigned char error_msg[][40] = {
|
||||||
|
"no error", /* ERROR_NONE */
|
||||||
|
"unknown command", /* ERROR_COMMAND */
|
||||||
|
"parse error", /* ERROR_PARSE */
|
||||||
|
"unknown option", /* ERROR_OPTION */
|
||||||
|
"bad value", /* ERROR_VALUE */
|
||||||
|
"no memory left", /* ERROR_NOMEM */
|
||||||
|
};
|
||||||
|
|
||||||
|
if (state->filename) {
|
||||||
|
fprintf(stderr, "%s:%d: %s\n",
|
||||||
|
state->filename, state->pos.line, error_msg[err]);
|
||||||
|
}
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Skip comments and whitespace. */
|
||||||
|
static void
|
||||||
|
skip_white(struct conf_parsing_pos *pos)
|
||||||
|
{
|
||||||
|
unsigned char *start = pos->look;
|
||||||
|
|
||||||
while (*start) {
|
while (*start) {
|
||||||
while (isspace(*start)) {
|
while (isspace(*start)) {
|
||||||
if (*start == '\n') {
|
if (*start == '\n') {
|
||||||
(*line)++;
|
pos->line++;
|
||||||
}
|
}
|
||||||
start++;
|
start++;
|
||||||
}
|
}
|
||||||
@ -74,128 +121,261 @@ skip_white(unsigned char *start, int *line)
|
|||||||
if (*start == '#') {
|
if (*start == '#') {
|
||||||
start += strcspn(start, "\n");
|
start += strcspn(start, "\n");
|
||||||
} else {
|
} else {
|
||||||
return start;
|
pos->look = start;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return start;
|
pos->look = start;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Skip a quoted string.
|
||||||
|
* This function allows "mismatching quotes' because str_rd() does so. */
|
||||||
|
static void
|
||||||
|
skip_quoted(struct conf_parsing_pos *pos)
|
||||||
|
{
|
||||||
|
assert(isquote(*pos->look));
|
||||||
|
if_assert_failed return;
|
||||||
|
pos->look++;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
if (!*pos->look)
|
||||||
|
return;
|
||||||
|
if (isquote(*pos->look)) {
|
||||||
|
pos->look++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (*pos->look == '\\' && pos->look[1])
|
||||||
|
pos->look++;
|
||||||
|
if (*pos->look == '\n')
|
||||||
|
pos->line++;
|
||||||
|
pos->look++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Skip the value of an option.
|
||||||
|
*
|
||||||
|
* This job is normally done by the reader function that corresponds
|
||||||
|
* to the type of the option. However, if ELinks does not recognize
|
||||||
|
* the name of the option, it cannot look up the type and has to use
|
||||||
|
* this function instead. */
|
||||||
|
static void
|
||||||
|
skip_option_value(struct conf_parsing_pos *pos)
|
||||||
|
{
|
||||||
|
if (isquote(*pos->look)) {
|
||||||
|
/* Looks like OPT_STRING, OPT_CODEPAGE, OPT_LANGUAGE,
|
||||||
|
* or OPT_COLOR. */
|
||||||
|
skip_quoted(pos);
|
||||||
|
} else {
|
||||||
|
/* Looks like OPT_BOOL, OPT_INT, or OPT_LONG. */
|
||||||
|
while (isasciialnum(*pos->look) || *pos->look == '.'
|
||||||
|
|| *pos->look == '+' || *pos->look == '-')
|
||||||
|
pos->look++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Skip to the next newline or comment that is not part of a quoted
|
||||||
|
* string. When ELinks hits a parse error in the configuration file,
|
||||||
|
* it calls this in order to find the place where is should resume
|
||||||
|
* parsing. This is intended to prevent ELinks from treating words
|
||||||
|
* in strings as commands. */
|
||||||
|
static void
|
||||||
|
skip_to_unquoted_newline_or_comment(struct conf_parsing_pos *pos)
|
||||||
|
{
|
||||||
|
while (*pos->look && *pos->look != '#' && *pos->look != '\n') {
|
||||||
|
if (isquote(*pos->look))
|
||||||
|
skip_quoted(pos);
|
||||||
|
else
|
||||||
|
pos->look++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Parse a command. Returns error code. */
|
/* Parse a command. Returns error code. */
|
||||||
/* If dynamic string credentials are supplied, we will mirror the command at
|
/* If dynamic string credentials are supplied, we will mirror the command at
|
||||||
* the end of the string; however, we won't load the option value to the tree,
|
* the end of the string; however, we won't load the option value to the tree,
|
||||||
* and we will even write option value from the tree to the output string. We
|
* and we will even write option value from the tree to the output string.
|
||||||
* will only possibly set OPT_WATERMARK flag to the option (if enabled). */
|
* We will only possibly set or clear OPT_MUST_SAVE flag in the option. */
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_set_common(struct option *opt_tree, unsigned char **file, int *line,
|
parse_set_common(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf, int want_domain)
|
struct string *mirror, int is_system_conf, int want_domain)
|
||||||
{
|
{
|
||||||
unsigned char *orig_pos = *file;
|
const unsigned char *domain_orig = NULL;
|
||||||
unsigned char *domain_name = NULL;
|
size_t domain_len = 0;
|
||||||
unsigned char *optname;
|
unsigned char *domain_copy = NULL;
|
||||||
unsigned char bin;
|
const unsigned char *optname_orig;
|
||||||
|
size_t optname_len;
|
||||||
|
unsigned char *optname_copy;
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
skip_white(&state->pos);
|
||||||
if (!**file) return ERROR_PARSE;
|
if (!*state->pos.look) return show_parse_error(state, ERROR_PARSE);
|
||||||
|
|
||||||
if (want_domain) {
|
if (want_domain) {
|
||||||
domain_name = *file;
|
domain_orig = state->pos.look;
|
||||||
while (isident(**file) || **file == '*' || **file == '.' || **file == '+')
|
while (isident(*state->pos.look) || *state->pos.look == '*'
|
||||||
(*file)++;
|
|| *state->pos.look == '.' || *state->pos.look == '+')
|
||||||
|
state->pos.look++;
|
||||||
|
domain_len = state->pos.look - domain_orig;
|
||||||
|
|
||||||
bin = **file;
|
skip_white(&state->pos);
|
||||||
**file = '\0';
|
|
||||||
domain_name = stracpy(domain_name);
|
|
||||||
**file = bin;
|
|
||||||
if (!domain_name) return ERROR_NOMEM;
|
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Option name */
|
/* Option name */
|
||||||
optname = *file;
|
optname_orig = state->pos.look;
|
||||||
while (isident(**file) || **file == '*' || **file == '.' || **file == '+')
|
while (isident(*state->pos.look) || *state->pos.look == '*'
|
||||||
(*file)++;
|
|| *state->pos.look == '.' || *state->pos.look == '+')
|
||||||
|
state->pos.look++;
|
||||||
|
optname_len = state->pos.look - optname_orig;
|
||||||
|
|
||||||
bin = **file;
|
skip_white(&state->pos);
|
||||||
**file = '\0';
|
|
||||||
optname = stracpy(optname);
|
|
||||||
**file = bin;
|
|
||||||
if (!optname) return ERROR_NOMEM;
|
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
|
||||||
|
|
||||||
/* Equal sign */
|
/* Equal sign */
|
||||||
if (**file != '=') { mem_free(optname); return ERROR_PARSE; }
|
if (*state->pos.look != '=')
|
||||||
(*file)++; /* '=' */
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
*file = skip_white(*file, line);
|
state->pos.look++; /* '=' */
|
||||||
if (!**file) { mem_free(optname); return ERROR_VALUE; }
|
skip_white(&state->pos);
|
||||||
|
if (!*state->pos.look)
|
||||||
|
return show_parse_error(state, ERROR_VALUE);
|
||||||
|
|
||||||
/* Mirror what we already have */
|
optname_copy = memacpy(optname_orig, optname_len);
|
||||||
if (mirror) add_bytes_to_string(mirror, orig_pos, *file - orig_pos);
|
if (!optname_copy) return show_parse_error(state, ERROR_NOMEM);
|
||||||
|
if (want_domain) {
|
||||||
|
domain_copy = memacpy(domain_orig, domain_len);
|
||||||
|
if (!domain_copy) {
|
||||||
|
mem_free(optname_copy);
|
||||||
|
return show_parse_error(state, ERROR_NOMEM);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Option value */
|
/* Option value */
|
||||||
{
|
{
|
||||||
struct option *opt;
|
struct option *opt;
|
||||||
unsigned char *val;
|
unsigned char *val;
|
||||||
|
const struct conf_parsing_pos pos_before_value = state->pos;
|
||||||
|
|
||||||
if (want_domain && *domain_name) {
|
if (want_domain && *domain_copy) {
|
||||||
struct option *domain_tree;
|
struct option *domain_tree;
|
||||||
|
|
||||||
domain_tree = get_domain_tree(domain_name);
|
domain_tree = get_domain_tree(domain_copy);
|
||||||
if (!domain_tree) {
|
if (!domain_tree) {
|
||||||
mem_free(domain_name);
|
mem_free(domain_copy);
|
||||||
mem_free(optname);
|
mem_free(optname_copy);
|
||||||
return ERROR_NOMEM;
|
skip_option_value(&state->pos);
|
||||||
|
return show_parse_error(state, ERROR_NOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mirror) {
|
if (mirror) {
|
||||||
opt = get_opt_rec_real(domain_tree, optname);
|
opt = get_opt_rec_real(domain_tree,
|
||||||
|
optname_copy);
|
||||||
} else {
|
} else {
|
||||||
opt = get_opt_rec(opt_tree, optname);
|
opt = get_opt_rec(opt_tree, optname_copy);
|
||||||
if (opt) {
|
if (opt) {
|
||||||
opt = get_option_shadow(opt, opt_tree,
|
opt = get_option_shadow(opt, opt_tree,
|
||||||
domain_tree);
|
domain_tree);
|
||||||
if (!opt) {
|
if (!opt) {
|
||||||
mem_free(domain_name);
|
mem_free(domain_copy);
|
||||||
mem_free(optname);
|
mem_free(optname_copy);
|
||||||
return ERROR_NOMEM;
|
skip_option_value(&state->pos);
|
||||||
|
return show_parse_error(state,
|
||||||
|
ERROR_NOMEM);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
opt = mirror ? get_opt_rec_real(opt_tree, optname) : get_opt_rec(opt_tree, optname);
|
opt = mirror
|
||||||
|
? get_opt_rec_real(opt_tree, optname_copy)
|
||||||
|
: get_opt_rec(opt_tree, optname_copy);
|
||||||
}
|
}
|
||||||
if (want_domain)
|
if (want_domain)
|
||||||
mem_free(domain_name);
|
mem_free(domain_copy);
|
||||||
mem_free(optname);
|
domain_copy = NULL;
|
||||||
|
mem_free(optname_copy);
|
||||||
|
optname_copy = NULL;
|
||||||
|
|
||||||
if (!opt || (opt->flags & OPT_HIDDEN))
|
if (!opt || (opt->flags & OPT_HIDDEN)) {
|
||||||
|
show_parse_error(state, ERROR_OPTION);
|
||||||
|
skip_option_value(&state->pos);
|
||||||
return ERROR_OPTION;
|
return ERROR_OPTION;
|
||||||
|
/* TODO: Distinguish between two scenarios:
|
||||||
if (!option_types[opt->type].read)
|
* - A newer version of ELinks has saved an
|
||||||
return ERROR_VALUE;
|
* option that this version does not recognize.
|
||||||
|
* The option must be preserved. (This works.)
|
||||||
val = option_types[opt->type].read(opt, file, line);
|
* - The user has added an option, saved
|
||||||
if (!val) return ERROR_VALUE;
|
* elinks.conf, restarted ELinks, deleted the
|
||||||
|
* option, and is now saving elinks.conf again.
|
||||||
if (mirror) {
|
* The option should be rewritten to "unset".
|
||||||
if (opt->flags & OPT_DELETED)
|
* (This does not work yet.)
|
||||||
opt->flags &= ~OPT_WATERMARK;
|
* In both cases, ELinks has no struct option
|
||||||
else
|
* for that name. Possible fixes:
|
||||||
opt->flags |= OPT_WATERMARK;
|
* - If the tree has OPT_AUTOCREATE, then
|
||||||
if (option_types[opt->type].write) {
|
* assume the user had created that option,
|
||||||
option_types[opt->type].write(opt, mirror);
|
* and rewrite it to "unset". Otherwise,
|
||||||
|
* keep it.
|
||||||
|
* - When the user deletes an option, just mark
|
||||||
|
* it with OPT_DELETED, and keep it in memory
|
||||||
|
* as long as OPT_TOUCHED is set. */
|
||||||
}
|
}
|
||||||
} else if (!option_types[opt->type].set
|
|
||||||
|
if (!option_types[opt->type].read) {
|
||||||
|
show_parse_error(state, ERROR_VALUE);
|
||||||
|
skip_option_value(&state->pos);
|
||||||
|
return ERROR_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
val = option_types[opt->type].read(opt, &state->pos.look,
|
||||||
|
&state->pos.line);
|
||||||
|
if (!val) {
|
||||||
|
/* The reader function failed. Jump back to
|
||||||
|
* the beginning of the value and skip it with
|
||||||
|
* the generic code. For the error message,
|
||||||
|
* use the line number at the beginning of the
|
||||||
|
* value, because the ending position is not
|
||||||
|
* interesting if there is an unclosed quote. */
|
||||||
|
state->pos = pos_before_value;
|
||||||
|
show_parse_error(state, ERROR_VALUE);
|
||||||
|
skip_option_value(&state->pos);
|
||||||
|
return ERROR_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!mirror) {
|
||||||
|
/* loading a configuration file */
|
||||||
|
if (!option_types[opt->type].set
|
||||||
|| !option_types[opt->type].set(opt, val)) {
|
|| !option_types[opt->type].set(opt, val)) {
|
||||||
mem_free(val);
|
mem_free(val);
|
||||||
return ERROR_VALUE;
|
return show_parse_error(state, ERROR_VALUE);
|
||||||
|
}
|
||||||
|
} else if (is_system_conf) {
|
||||||
|
/* scanning a file that will not be rewritten */
|
||||||
|
struct option *flagsite = indirect_option(opt);
|
||||||
|
|
||||||
|
if (!(flagsite->flags & OPT_DELETED)
|
||||||
|
&& option_types[opt->type].equals
|
||||||
|
&& option_types[opt->type].equals(opt, val))
|
||||||
|
flagsite->flags &= ~OPT_MUST_SAVE;
|
||||||
|
else
|
||||||
|
flagsite->flags |= OPT_MUST_SAVE;
|
||||||
|
} else {
|
||||||
|
/* rewriting a configuration file */
|
||||||
|
struct option *flagsite = indirect_option(opt);
|
||||||
|
|
||||||
|
if (flagsite->flags & OPT_DELETED) {
|
||||||
|
/* Replace the "set" command with an
|
||||||
|
* "unset" command. */
|
||||||
|
add_to_string(mirror, "unset ");
|
||||||
|
add_bytes_to_string(mirror, optname_orig,
|
||||||
|
optname_len);
|
||||||
|
state->mirrored = state->pos.look;
|
||||||
|
} else if (option_types[opt->type].write) {
|
||||||
|
add_bytes_to_string(mirror, state->mirrored,
|
||||||
|
pos_before_value.look
|
||||||
|
- state->mirrored);
|
||||||
|
option_types[opt->type].write(opt, mirror);
|
||||||
|
state->mirrored = state->pos.look;
|
||||||
|
}
|
||||||
|
/* Remember that the option need not be
|
||||||
|
* written to the end of the file. */
|
||||||
|
flagsite->flags &= ~OPT_MUST_SAVE;
|
||||||
}
|
}
|
||||||
/* This is not needed since this will be WATERMARK'd when
|
|
||||||
* saving it. We won't need to save it as touched. */
|
|
||||||
/* if (!str) opt->flags |= OPT_TOUCHED; */
|
|
||||||
mem_free(val);
|
mem_free(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -203,64 +383,91 @@ parse_set_common(struct option *opt_tree, unsigned char **file, int *line,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_set_domain(struct option *opt_tree, unsigned char **file, int *line,
|
parse_set_domain(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
return parse_set_common(opt_tree, file, line, mirror, is_system_conf, 1);
|
return parse_set_common(opt_tree, state, mirror, is_system_conf, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_set(struct option *opt_tree, unsigned char **file, int *line,
|
parse_set(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
return parse_set_common(opt_tree, file, line, mirror, is_system_conf, 0);
|
return parse_set_common(opt_tree, state, mirror, is_system_conf, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_unset(struct option *opt_tree, unsigned char **file, int *line,
|
parse_unset(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
unsigned char *orig_pos = *file;
|
const unsigned char *optname_orig;
|
||||||
unsigned char *optname;
|
size_t optname_len;
|
||||||
unsigned char bin;
|
unsigned char *optname_copy;
|
||||||
|
|
||||||
/* XXX: This does not handle the autorewriting well and is mostly a
|
skip_white(&state->pos);
|
||||||
* quick hack than anything now. --pasky */
|
if (!*state->pos.look) return show_parse_error(state, ERROR_PARSE);
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
|
||||||
if (!**file) return ERROR_PARSE;
|
|
||||||
|
|
||||||
/* Option name */
|
/* Option name */
|
||||||
optname = *file;
|
optname_orig = state->pos.look;
|
||||||
while (isident(**file) || **file == '*' || **file == '.' || **file == '+')
|
while (isident(*state->pos.look) || *state->pos.look == '*'
|
||||||
(*file)++;
|
|| *state->pos.look == '.' || *state->pos.look == '+')
|
||||||
|
state->pos.look++;
|
||||||
|
optname_len = state->pos.look - optname_orig;
|
||||||
|
|
||||||
bin = **file;
|
optname_copy = memacpy(optname_orig, optname_len);
|
||||||
**file = '\0';
|
if (!optname_copy) return show_parse_error(state, ERROR_NOMEM);
|
||||||
optname = stracpy(optname);
|
|
||||||
**file = bin;
|
|
||||||
if (!optname) return ERROR_NOMEM;
|
|
||||||
|
|
||||||
/* Mirror what we have */
|
|
||||||
if (mirror) add_bytes_to_string(mirror, orig_pos, *file - orig_pos);
|
|
||||||
|
|
||||||
{
|
{
|
||||||
struct option *opt;
|
struct option *opt;
|
||||||
|
|
||||||
opt = get_opt_rec_real(opt_tree, optname);
|
opt = get_opt_rec_real(opt_tree, optname_copy);
|
||||||
mem_free(optname);
|
mem_free(optname_copy);
|
||||||
|
optname_copy = NULL;
|
||||||
|
|
||||||
if (!opt || (opt->flags & OPT_HIDDEN))
|
if (!opt || (opt->flags & OPT_HIDDEN)) {
|
||||||
return ERROR_OPTION;
|
/* The user wanted to delete the option, and
|
||||||
|
* it has already been deleted; this is not an
|
||||||
|
* error. This might happen if a version of
|
||||||
|
* ELinks has a built-in URL rewriting rule,
|
||||||
|
* the user disables it, and a later version
|
||||||
|
* no longer has it. */
|
||||||
|
return ERROR_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
if (!mirror) {
|
if (!mirror) {
|
||||||
|
/* loading a configuration file */
|
||||||
if (opt->flags & OPT_ALLOC) delete_option(opt);
|
if (opt->flags & OPT_ALLOC) delete_option(opt);
|
||||||
} else {
|
else mark_option_as_deleted(opt);
|
||||||
if (opt->flags & OPT_DELETED)
|
} else if (is_system_conf) {
|
||||||
opt->flags |= OPT_WATERMARK;
|
/* scanning a file that will not be rewritten */
|
||||||
|
struct option *flagsite = indirect_option(opt);
|
||||||
|
|
||||||
|
if (flagsite->flags & OPT_DELETED)
|
||||||
|
flagsite->flags &= ~OPT_MUST_SAVE;
|
||||||
else
|
else
|
||||||
opt->flags &= ~OPT_WATERMARK;
|
flagsite->flags |= OPT_MUST_SAVE;
|
||||||
|
} else {
|
||||||
|
/* rewriting a configuration file */
|
||||||
|
struct option *flagsite = indirect_option(opt);
|
||||||
|
|
||||||
|
if (flagsite->flags & OPT_DELETED) {
|
||||||
|
/* The "unset" command is already in the file,
|
||||||
|
* and unlike with "set", there is no value
|
||||||
|
* to be updated. */
|
||||||
|
} else if (option_types[opt->type].write) {
|
||||||
|
/* Replace the "unset" command with a
|
||||||
|
* "set" command. */
|
||||||
|
add_to_string(mirror, "set ");
|
||||||
|
add_bytes_to_string(mirror, optname_orig,
|
||||||
|
optname_len);
|
||||||
|
add_to_string(mirror, " = ");
|
||||||
|
option_types[opt->type].write(opt, mirror);
|
||||||
|
state->mirrored = state->pos.look;
|
||||||
|
}
|
||||||
|
/* Remember that the option need not be
|
||||||
|
* written to the end of the file. */
|
||||||
|
flagsite->flags &= ~OPT_MUST_SAVE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -268,73 +475,90 @@ parse_unset(struct option *opt_tree, unsigned char **file, int *line,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_bind(struct option *opt_tree, unsigned char **file, int *line,
|
parse_bind(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
unsigned char *orig_pos = *file, *next_pos;
|
|
||||||
unsigned char *keymap, *keystroke, *action;
|
unsigned char *keymap, *keystroke, *action;
|
||||||
enum parse_error err = ERROR_NONE;
|
enum parse_error err = ERROR_NONE;
|
||||||
|
struct conf_parsing_pos before_error;
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
skip_white(&state->pos);
|
||||||
if (!*file) return ERROR_PARSE;
|
if (!*state->pos.look) return show_parse_error(state, ERROR_PARSE);
|
||||||
|
|
||||||
/* Keymap */
|
/* Keymap */
|
||||||
keymap = option_types[OPT_STRING].read(NULL, file, line);
|
before_error = state->pos;
|
||||||
*file = skip_white(*file, line);
|
keymap = option_types[OPT_STRING].read(NULL, &state->pos.look,
|
||||||
if (!keymap || !**file)
|
&state->pos.line);
|
||||||
return ERROR_OPTION;
|
skip_white(&state->pos);
|
||||||
|
if (!keymap || !*state->pos.look) {
|
||||||
|
state->pos = before_error;
|
||||||
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
|
}
|
||||||
|
|
||||||
/* Keystroke */
|
/* Keystroke */
|
||||||
keystroke = option_types[OPT_STRING].read(NULL, file, line);
|
before_error = state->pos;
|
||||||
*file = skip_white(*file, line);
|
keystroke = option_types[OPT_STRING].read(NULL, &state->pos.look,
|
||||||
if (!keystroke || !**file) {
|
&state->pos.line);
|
||||||
mem_free(keymap);
|
skip_white(&state->pos);
|
||||||
return ERROR_OPTION;
|
if (!keystroke || !*state->pos.look) {
|
||||||
|
mem_free(keymap); mem_free_if(keystroke);
|
||||||
|
state->pos = before_error;
|
||||||
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Equal sign */
|
/* Equal sign */
|
||||||
*file = skip_white(*file, line);
|
skip_white(&state->pos);
|
||||||
if (**file != '=') {
|
if (*state->pos.look != '=') {
|
||||||
mem_free(keymap); mem_free(keystroke);
|
mem_free(keymap); mem_free(keystroke);
|
||||||
return ERROR_PARSE;
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
}
|
}
|
||||||
(*file)++; /* '=' */
|
state->pos.look++; /* '=' */
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
skip_white(&state->pos);
|
||||||
if (!**file) {
|
if (!*state->pos.look) {
|
||||||
mem_free(keymap); mem_free(keystroke);
|
mem_free(keymap); mem_free(keystroke);
|
||||||
return ERROR_PARSE;
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Action */
|
/* Action */
|
||||||
next_pos = *file;
|
before_error = state->pos;
|
||||||
action = option_types[OPT_STRING].read(NULL, file, line);
|
action = option_types[OPT_STRING].read(NULL, &state->pos.look,
|
||||||
|
&state->pos.line);
|
||||||
if (!action) {
|
if (!action) {
|
||||||
mem_free(keymap);
|
mem_free(keymap); mem_free(keystroke);
|
||||||
return ERROR_VALUE;
|
state->pos = before_error;
|
||||||
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mirror) {
|
if (!mirror) {
|
||||||
/* Mirror what we already have */
|
/* loading a configuration file */
|
||||||
unsigned char *act_str = bind_act(keymap, keystroke);
|
|
||||||
|
|
||||||
if (act_str) {
|
|
||||||
add_bytes_to_string(mirror, orig_pos,
|
|
||||||
next_pos - orig_pos);
|
|
||||||
add_to_string(mirror, act_str);
|
|
||||||
mem_free(act_str);
|
|
||||||
} else {
|
|
||||||
err = ERROR_VALUE;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/* We don't bother to bind() if -default-keys. */
|
/* We don't bother to bind() if -default-keys. */
|
||||||
if (!get_cmd_opt_bool("default-keys")
|
if (!get_cmd_opt_bool("default-keys")
|
||||||
&& bind_do(keymap, keystroke, action, is_system_conf)) {
|
&& bind_do(keymap, keystroke, action, is_system_conf)) {
|
||||||
/* bind_do() tried but failed. */
|
/* bind_do() tried but failed. */
|
||||||
err = ERROR_VALUE;
|
err = show_parse_error(state, ERROR_VALUE);
|
||||||
} else {
|
} else {
|
||||||
err = ERROR_NONE;
|
err = ERROR_NONE;
|
||||||
}
|
}
|
||||||
|
} else if (is_system_conf) {
|
||||||
|
/* scanning a file that will not be rewritten */
|
||||||
|
/* TODO */
|
||||||
|
} else {
|
||||||
|
/* rewriting a configuration file */
|
||||||
|
/* Mirror what we already have. If the keystroke has
|
||||||
|
* been unbound, then act_str is simply "none" and
|
||||||
|
* this does not require special handling. */
|
||||||
|
unsigned char *act_str = bind_act(keymap, keystroke);
|
||||||
|
|
||||||
|
if (act_str) {
|
||||||
|
add_bytes_to_string(mirror, state->mirrored,
|
||||||
|
before_error.look - state->mirrored);
|
||||||
|
add_to_string(mirror, act_str);
|
||||||
|
mem_free(act_str);
|
||||||
|
state->mirrored = state->pos.look;
|
||||||
|
} else {
|
||||||
|
err = show_parse_error(state, ERROR_VALUE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
mem_free(keymap); mem_free(keystroke); mem_free(action);
|
mem_free(keymap); mem_free(keystroke); mem_free(action);
|
||||||
return err;
|
return err;
|
||||||
@ -344,27 +568,31 @@ static int load_config_file(unsigned char *, unsigned char *, struct option *,
|
|||||||
struct string *, int);
|
struct string *, int);
|
||||||
|
|
||||||
static enum parse_error
|
static enum parse_error
|
||||||
parse_include(struct option *opt_tree, unsigned char **file, int *line,
|
parse_include(struct option *opt_tree, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
unsigned char *orig_pos = *file;
|
|
||||||
unsigned char *fname;
|
unsigned char *fname;
|
||||||
struct string dumbstring;
|
struct string dumbstring;
|
||||||
|
struct conf_parsing_pos before_error;
|
||||||
|
|
||||||
if (!init_string(&dumbstring)) return ERROR_NOMEM;
|
if (!init_string(&dumbstring))
|
||||||
|
return show_parse_error(state, ERROR_NOMEM);
|
||||||
|
|
||||||
*file = skip_white(*file, line);
|
skip_white(&state->pos);
|
||||||
if (!*file) return ERROR_PARSE;
|
if (!*state->pos.look) {
|
||||||
|
|
||||||
/* File name */
|
|
||||||
fname = option_types[OPT_STRING].read(NULL, file, line);
|
|
||||||
if (!fname) {
|
|
||||||
done_string(&dumbstring);
|
done_string(&dumbstring);
|
||||||
return ERROR_VALUE;
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Mirror what we already have */
|
/* File name */
|
||||||
if (mirror) add_bytes_to_string(mirror, orig_pos, *file - orig_pos);
|
before_error = state->pos;
|
||||||
|
fname = option_types[OPT_STRING].read(NULL, &state->pos.look,
|
||||||
|
&state->pos.line);
|
||||||
|
if (!fname) {
|
||||||
|
done_string(&dumbstring);
|
||||||
|
state->pos = before_error;
|
||||||
|
return show_parse_error(state, ERROR_PARSE);
|
||||||
|
}
|
||||||
|
|
||||||
/* We want load_config_file() to watermark stuff, but not to load
|
/* We want load_config_file() to watermark stuff, but not to load
|
||||||
* anything, polluting our beloved options tree - thus, we will feed it
|
* anything, polluting our beloved options tree - thus, we will feed it
|
||||||
@ -375,10 +603,11 @@ parse_include(struct option *opt_tree, unsigned char **file, int *line,
|
|||||||
* CONFDIR/<otherfile> ;). --pasky */
|
* CONFDIR/<otherfile> ;). --pasky */
|
||||||
if (load_config_file(fname[0] == '/' ? (unsigned char *) ""
|
if (load_config_file(fname[0] == '/' ? (unsigned char *) ""
|
||||||
: elinks_home,
|
: elinks_home,
|
||||||
fname, opt_tree, &dumbstring, is_system_conf)) {
|
fname, opt_tree,
|
||||||
|
mirror ? &dumbstring : NULL, 1)) {
|
||||||
done_string(&dumbstring);
|
done_string(&dumbstring);
|
||||||
mem_free(fname);
|
mem_free(fname);
|
||||||
return ERROR_VALUE;
|
return show_parse_error(state, ERROR_VALUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
done_string(&dumbstring);
|
done_string(&dumbstring);
|
||||||
@ -388,13 +617,13 @@ parse_include(struct option *opt_tree, unsigned char **file, int *line,
|
|||||||
|
|
||||||
|
|
||||||
struct parse_handler {
|
struct parse_handler {
|
||||||
unsigned char *command;
|
const unsigned char *command;
|
||||||
enum parse_error (*handler)(struct option *opt_tree,
|
enum parse_error (*handler)(struct option *opt_tree,
|
||||||
unsigned char **file, int *line,
|
struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf);
|
struct string *mirror, int is_system_conf);
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct parse_handler parse_handlers[] = {
|
static const struct parse_handler parse_handlers[] = {
|
||||||
{ "set_domain", parse_set_domain },
|
{ "set_domain", parse_set_domain },
|
||||||
{ "set", parse_set },
|
{ "set", parse_set },
|
||||||
{ "unset", parse_unset },
|
{ "unset", parse_unset },
|
||||||
@ -404,102 +633,113 @@ static struct parse_handler parse_handlers[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
enum parse_error
|
static enum parse_error
|
||||||
parse_config_command(struct option *options, unsigned char **file, int *line,
|
parse_config_command(struct option *options, struct conf_parsing_state *state,
|
||||||
struct string *mirror, int is_system_conf)
|
struct string *mirror, int is_system_conf)
|
||||||
{
|
{
|
||||||
struct parse_handler *handler;
|
const struct parse_handler *handler;
|
||||||
|
|
||||||
|
/* If we're mirroring, then everything up to this point must
|
||||||
|
* have already been mirrored. */
|
||||||
|
assert(mirror == NULL || state->mirrored == state->pos.look);
|
||||||
|
if_assert_failed return show_parse_error(state, ERROR_PARSE);
|
||||||
|
|
||||||
for (handler = parse_handlers; handler->command;
|
for (handler = parse_handlers; handler->command;
|
||||||
handler++) {
|
handler++) {
|
||||||
int cmdlen = strlen(handler->command);
|
int cmdlen = strlen(handler->command);
|
||||||
|
|
||||||
if (!strncmp(*file, handler->command, cmdlen)
|
if (!strncmp(state->pos.look, handler->command, cmdlen)
|
||||||
&& isspace((*file)[cmdlen])) {
|
&& isspace(state->pos.look[cmdlen])) {
|
||||||
enum parse_error err;
|
enum parse_error err;
|
||||||
struct string mirror2 = NULL_STRING;
|
|
||||||
struct string *m2 = NULL;
|
|
||||||
|
|
||||||
/* Mirror what we already have */
|
state->pos.look += cmdlen;
|
||||||
if (mirror && init_string(&mirror2)) {
|
err = handler->handler(options, state, mirror,
|
||||||
m2 = &mirror2;
|
|
||||||
add_bytes_to_string(m2, *file, cmdlen);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
*file += cmdlen;
|
|
||||||
err = handler->handler(options, file, line, m2,
|
|
||||||
is_system_conf);
|
is_system_conf);
|
||||||
if (!err && mirror && m2) {
|
if (mirror) {
|
||||||
add_string_to_string(mirror, m2);
|
/* Mirror any characters that the handler
|
||||||
|
* consumed but did not already mirror. */
|
||||||
|
add_bytes_to_string(mirror, state->mirrored,
|
||||||
|
state->pos.look - state->mirrored);
|
||||||
|
state->mirrored = state->pos.look;
|
||||||
}
|
}
|
||||||
if (m2) done_string(m2);
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ERROR_COMMAND;
|
return show_parse_error(state, ERROR_COMMAND);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_EXMODE
|
||||||
|
enum parse_error
|
||||||
|
parse_config_exmode_command(unsigned char *cmd)
|
||||||
|
{
|
||||||
|
struct conf_parsing_state state = {{ 0 }};
|
||||||
|
|
||||||
|
state.pos.look = cmd;
|
||||||
|
state.pos.line = 0;
|
||||||
|
state.mirrored = NULL; /* not read because mirror is NULL too */
|
||||||
|
state.filename = NULL; /* prevent error messages */
|
||||||
|
|
||||||
|
return parse_config_command(config_options, &state, NULL, 0);
|
||||||
|
}
|
||||||
|
#endif /* CONFIG_EXMODE */
|
||||||
|
|
||||||
void
|
void
|
||||||
parse_config_file(struct option *options, unsigned char *name,
|
parse_config_file(struct option *options, unsigned char *name,
|
||||||
unsigned char *file, struct string *mirror,
|
unsigned char *file, struct string *mirror,
|
||||||
int is_system_conf)
|
int is_system_conf)
|
||||||
{
|
{
|
||||||
int line = 1;
|
struct conf_parsing_state state = {{ 0 }};
|
||||||
int error_occurred = 0;
|
int error_occurred = 0;
|
||||||
enum parse_error err = 0;
|
|
||||||
enum verbose_level verbose = get_cmd_opt_int("verbose");
|
|
||||||
unsigned char error_msg[][40] = {
|
|
||||||
"no error",
|
|
||||||
"parse error",
|
|
||||||
"unknown command",
|
|
||||||
"unknown option",
|
|
||||||
"bad value",
|
|
||||||
"no memory left",
|
|
||||||
};
|
|
||||||
|
|
||||||
while (file && *file) {
|
state.pos.look = file;
|
||||||
unsigned char *orig_pos = file;
|
state.pos.line = 1;
|
||||||
|
state.mirrored = file;
|
||||||
|
if (!mirror && get_cmd_opt_int("verbose") >= VERBOSE_WARNINGS)
|
||||||
|
state.filename = name;
|
||||||
|
|
||||||
|
while (state.pos.look && *state.pos.look) {
|
||||||
|
enum parse_error err = ERROR_NONE;
|
||||||
|
|
||||||
/* Skip all possible comments and whitespace. */
|
/* Skip all possible comments and whitespace. */
|
||||||
file = skip_white(file, &line);
|
skip_white(&state.pos);
|
||||||
|
|
||||||
/* Mirror what we already have */
|
/* Mirror what we already have */
|
||||||
if (mirror)
|
if (mirror) {
|
||||||
add_bytes_to_string(mirror, orig_pos, file - orig_pos);
|
add_bytes_to_string(mirror, state.mirrored,
|
||||||
|
state.pos.look - state.mirrored);
|
||||||
|
state.mirrored = state.pos.look;
|
||||||
|
}
|
||||||
|
|
||||||
/* Second chance to escape from the hell. */
|
/* Second chance to escape from the hell. */
|
||||||
if (!*file) break;
|
if (!*state.pos.look) break;
|
||||||
|
|
||||||
err = parse_config_command(options, &file, &line, mirror,
|
err = parse_config_command(options, &state, mirror,
|
||||||
is_system_conf);
|
is_system_conf);
|
||||||
|
switch (err) {
|
||||||
|
case ERROR_NONE:
|
||||||
|
break;
|
||||||
|
|
||||||
if (err == ERROR_COMMAND) {
|
case ERROR_COMMAND:
|
||||||
orig_pos = file;
|
case ERROR_PARSE:
|
||||||
/* Jump over this crap we can't understand. */
|
/* Jump over this crap we can't understand. */
|
||||||
while (!isspace(*file) && *file != '#' && *file)
|
skip_to_unquoted_newline_or_comment(&state.pos);
|
||||||
file++;
|
|
||||||
|
|
||||||
/* Mirror what we already have */
|
/* Mirror what we already have */
|
||||||
if (mirror) add_bytes_to_string(mirror, orig_pos,
|
if (mirror) {
|
||||||
file - orig_pos);
|
add_bytes_to_string(mirror, state.mirrored,
|
||||||
|
state.pos.look - state.mirrored);
|
||||||
|
state.mirrored = state.pos.look;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!mirror && err) {
|
/* fall through */
|
||||||
/* TODO: Make this a macro and report error directly
|
default:
|
||||||
* as it's stumbled upon; line info may not be accurate
|
|
||||||
* anymore now (?). --pasky */
|
|
||||||
if (verbose >= VERBOSE_WARNINGS) {
|
|
||||||
fprintf(stderr, "%s:%d: %s\n",
|
|
||||||
name, line, error_msg[err]);
|
|
||||||
error_occurred = 1;
|
error_occurred = 1;
|
||||||
}
|
break;
|
||||||
err = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!error_occurred) return;
|
if (!error_occurred || !state.filename) return;
|
||||||
|
|
||||||
/* If an error occurred make sure that the user is notified and is able
|
/* If an error occurred make sure that the user is notified and is able
|
||||||
* to see it. First sound the bell. Then, if the text viewer is going to
|
* to see it. First sound the bell. Then, if the text viewer is going to
|
||||||
@ -605,7 +845,6 @@ load_config(void)
|
|||||||
static int indentation = 2;
|
static int indentation = 2;
|
||||||
/* 0 -> none, 1 -> only option full name+type, 2 -> only desc, 3 -> both */
|
/* 0 -> none, 1 -> only option full name+type, 2 -> only desc, 3 -> both */
|
||||||
static int comments = 3;
|
static int comments = 3;
|
||||||
static int touching = 0;
|
|
||||||
|
|
||||||
static inline unsigned char *
|
static inline unsigned char *
|
||||||
conf_i18n(unsigned char *s, int i18n)
|
conf_i18n(unsigned char *s, int i18n)
|
||||||
@ -634,12 +873,6 @@ smart_config_output_fn(struct string *string, struct option *option,
|
|||||||
if (option->type == OPT_ALIAS)
|
if (option->type == OPT_ALIAS)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* XXX: OPT_LANGUAGE shouldn't have any bussiness here, but we're just
|
|
||||||
* weird in that area. */
|
|
||||||
if (touching && !(option->flags & OPT_TOUCHED)
|
|
||||||
&& option->type != OPT_LANGUAGE)
|
|
||||||
return;
|
|
||||||
|
|
||||||
switch (action) {
|
switch (action) {
|
||||||
case 0:
|
case 0:
|
||||||
if (!(comments & 1)) break;
|
if (!(comments & 1)) break;
|
||||||
@ -765,25 +998,26 @@ create_config_string(unsigned char *prefix, unsigned char *name)
|
|||||||
|
|
||||||
if (!init_string(&config)) return NULL;
|
if (!init_string(&config)) return NULL;
|
||||||
|
|
||||||
if (savestyle == 3) {
|
{
|
||||||
touching = 1;
|
int set_all = (savestyle == 1 || savestyle == 2);
|
||||||
savestyle = 1;
|
struct domain_tree *domain;
|
||||||
} else {
|
|
||||||
touching = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (savestyle == 2) watermark_deleted_options(options->value.tree);
|
prepare_mustsave_flags(options->value.tree, set_all);
|
||||||
|
foreach (domain, domain_trees) {
|
||||||
|
prepare_mustsave_flags(domain->tree->value.tree,
|
||||||
|
set_all);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Scaring. */
|
/* Scaring. */
|
||||||
if (savestyle == 2
|
if (savestyle == 2
|
||||||
|| (savestyle < 2
|
|| load_config_file(prefix, name, options, &config, 0)
|
||||||
&& (load_config_file(prefix, name, options, &config, 0)
|
|| !config.length) {
|
||||||
|| !config.length))) {
|
|
||||||
/* At first line, and in English, write ELinks version, may be
|
/* At first line, and in English, write ELinks version, may be
|
||||||
* of some help in future. Please keep that format for it.
|
* of some help in future. Please keep that format for it.
|
||||||
* --Zas */
|
* --Zas */
|
||||||
add_to_string(&config, "## ELinks " VERSION " configuration file\n\n");
|
add_to_string(&config, "## ELinks " VERSION " configuration file\n\n");
|
||||||
assert(savestyle >= 0 && savestyle <= 2);
|
assert(savestyle >= 0 && savestyle <= 3);
|
||||||
switch (savestyle) {
|
switch (savestyle) {
|
||||||
case 0:
|
case 0:
|
||||||
add_to_string(&config, conf_i18n(N_(
|
add_to_string(&config, conf_i18n(N_(
|
||||||
@ -793,7 +1027,7 @@ create_config_string(unsigned char *prefix, unsigned char *name)
|
|||||||
"## and all your formatting, own comments etc will be kept as-is.\n"),
|
"## and all your formatting, own comments etc will be kept as-is.\n"),
|
||||||
i18n));
|
i18n));
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1: case 3:
|
||||||
add_to_string(&config, conf_i18n(N_(
|
add_to_string(&config, conf_i18n(N_(
|
||||||
"## This is ELinks configuration file. You can edit it manually,\n"
|
"## This is ELinks configuration file. You can edit it manually,\n"
|
||||||
"## if you wish so; this file is edited by ELinks when you save\n"
|
"## if you wish so; this file is edited by ELinks when you save\n"
|
||||||
@ -844,7 +1078,6 @@ create_config_string(unsigned char *prefix, unsigned char *name)
|
|||||||
domain->tree->value.tree,
|
domain->tree->value.tree,
|
||||||
NULL, 0,
|
NULL, 0,
|
||||||
smart_config_output_fn);
|
smart_config_output_fn);
|
||||||
unmark_options_tree(domain->tree->value.tree);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
smart_config_output_fn_domain = NULL;
|
smart_config_output_fn_domain = NULL;
|
||||||
@ -866,8 +1099,6 @@ create_config_string(unsigned char *prefix, unsigned char *name)
|
|||||||
done_string(&tmpstring);
|
done_string(&tmpstring);
|
||||||
|
|
||||||
get_me_out:
|
get_me_out:
|
||||||
unmark_options_tree(options->value.tree);
|
|
||||||
|
|
||||||
return config.source;
|
return config.source;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -895,6 +1126,13 @@ write_config_file(unsigned char *prefix, unsigned char *name,
|
|||||||
if (ssi) {
|
if (ssi) {
|
||||||
secure_fputs(ssi, cfg_str);
|
secure_fputs(ssi, cfg_str);
|
||||||
ret = secure_close(ssi);
|
ret = secure_close(ssi);
|
||||||
|
if (!ret) {
|
||||||
|
struct domain_tree *domain;
|
||||||
|
|
||||||
|
untouch_options(config_options->value.tree);
|
||||||
|
foreach (domain, domain_trees)
|
||||||
|
untouch_options(domain->tree->value.tree);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
write_config_dialog(term, config_file, secsave_errno, ret);
|
write_config_dialog(term, config_file, secsave_errno, ret);
|
||||||
|
@ -14,10 +14,9 @@ enum parse_error {
|
|||||||
};
|
};
|
||||||
|
|
||||||
void load_config(void);
|
void load_config(void);
|
||||||
enum parse_error parse_config_command(struct option *options,
|
#ifdef CONFIG_EXMODE
|
||||||
unsigned char **file, int *line,
|
enum parse_error parse_config_exmode_command(unsigned char *cmd);
|
||||||
struct string *mirror,
|
#endif
|
||||||
int is_system_conf);
|
|
||||||
void parse_config_file(struct option *options, unsigned char *name,
|
void parse_config_file(struct option *options, unsigned char *name,
|
||||||
unsigned char *file, struct string *mirror,
|
unsigned char *file, struct string *mirror,
|
||||||
int is_system_conf);
|
int is_system_conf);
|
||||||
|
@ -158,7 +158,14 @@ debug_check_option_syntax(struct option *option)
|
|||||||
/* Ugly kludge */
|
/* Ugly kludge */
|
||||||
static int no_autocreate = 0;
|
static int no_autocreate = 0;
|
||||||
|
|
||||||
/* Get record of option of given name, or NULL if there's no such option. */
|
/** Get record of option of given name, or NULL if there's no such option.
|
||||||
|
*
|
||||||
|
* If the specified option is an ::OPT_ALIAS, this function returns the
|
||||||
|
* alias, rather than the option to which the alias refers. It must
|
||||||
|
* work this way because the alias may have the ::OPT_ALIAS_NEGATE flag.
|
||||||
|
* Instead, if the caller tries to read or set the value of the alias,
|
||||||
|
* the functions associated with ::OPT_ALIAS will forward the operation
|
||||||
|
* to the underlying option. However, see indirect_option(). */
|
||||||
struct option *
|
struct option *
|
||||||
get_opt_rec(struct option *tree, const unsigned char *name_)
|
get_opt_rec(struct option *tree, const unsigned char *name_)
|
||||||
{
|
{
|
||||||
@ -242,6 +249,27 @@ get_opt_rec_real(struct option *tree, const unsigned char *name)
|
|||||||
return opt;
|
return opt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** If @a opt is an alias, return the option to which it refers.
|
||||||
|
*
|
||||||
|
* @warning Because the alias may have the ::OPT_ALIAS_NEGATE flag,
|
||||||
|
* the caller must not access the value of the returned option as if
|
||||||
|
* it were also the value of the alias. However, it is safe to access
|
||||||
|
* flags such as ::OPT_MUST_SAVE and ::OPT_DELETED. */
|
||||||
|
struct option *
|
||||||
|
indirect_option(struct option *alias)
|
||||||
|
{
|
||||||
|
struct option *real;
|
||||||
|
|
||||||
|
if (alias->type != OPT_ALIAS) return alias; /* not an error */
|
||||||
|
|
||||||
|
real = get_opt_rec(config_options, alias->value.string);
|
||||||
|
assertm(real != NULL, "%s aliased to unknown option %s!",
|
||||||
|
alias->name, alias->value.string);
|
||||||
|
if_assert_failed return alias;
|
||||||
|
|
||||||
|
return real;
|
||||||
|
}
|
||||||
|
|
||||||
/* Fetch pointer to value of certain option. It is guaranteed to never return
|
/* Fetch pointer to value of certain option. It is guaranteed to never return
|
||||||
* NULL. Note that you are supposed to use wrapper get_opt(). */
|
* NULL. Note that you are supposed to use wrapper get_opt(). */
|
||||||
union option_value *
|
union option_value *
|
||||||
@ -793,27 +821,35 @@ register_change_hooks(const struct change_hook_info *change_hooks)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
unmark_options_tree(LIST_OF(struct option) *tree)
|
prepare_mustsave_flags(LIST_OF(struct option) *tree, int set_all)
|
||||||
{
|
{
|
||||||
struct option *option;
|
struct option *option;
|
||||||
|
|
||||||
foreach (option, *tree) {
|
foreach (option, *tree) {
|
||||||
option->flags &= ~OPT_WATERMARK;
|
/* XXX: OPT_LANGUAGE shouldn't have any bussiness
|
||||||
|
* here, but we're just weird in that area. */
|
||||||
|
if (set_all
|
||||||
|
|| (option->flags & (OPT_TOUCHED | OPT_DELETED))
|
||||||
|
|| option->type == OPT_LANGUAGE)
|
||||||
|
option->flags |= OPT_MUST_SAVE;
|
||||||
|
else
|
||||||
|
option->flags &= ~OPT_MUST_SAVE;
|
||||||
|
|
||||||
if (option->type == OPT_TREE)
|
if (option->type == OPT_TREE)
|
||||||
unmark_options_tree(option->value.tree);
|
prepare_mustsave_flags(option->value.tree, set_all);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
watermark_deleted_options(LIST_OF(struct option) *tree)
|
untouch_options(LIST_OF(struct option) *tree)
|
||||||
{
|
{
|
||||||
struct option *option;
|
struct option *option;
|
||||||
|
|
||||||
foreach (option, *tree) {
|
foreach (option, *tree) {
|
||||||
if (option->flags & OPT_DELETED)
|
option->flags &= ~OPT_TOUCHED;
|
||||||
option->flags |= OPT_WATERMARK;
|
|
||||||
else if (option->type == OPT_TREE)
|
if (option->type == OPT_TREE)
|
||||||
watermark_deleted_options(option->value.tree);
|
untouch_options(option->value.tree);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -826,7 +862,7 @@ check_nonempty_tree(LIST_OF(struct option) *options)
|
|||||||
if (opt->type == OPT_TREE) {
|
if (opt->type == OPT_TREE) {
|
||||||
if (check_nonempty_tree(opt->value.tree))
|
if (check_nonempty_tree(opt->value.tree))
|
||||||
return 1;
|
return 1;
|
||||||
} else if (!(opt->flags & OPT_WATERMARK)) {
|
} else if (opt->flags & OPT_MUST_SAVE) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -847,14 +883,14 @@ smart_config_string(struct string *str, int print_comment, int i18n,
|
|||||||
int do_print_comment = 1;
|
int do_print_comment = 1;
|
||||||
|
|
||||||
if (option->flags & OPT_HIDDEN ||
|
if (option->flags & OPT_HIDDEN ||
|
||||||
option->flags & OPT_WATERMARK ||
|
|
||||||
option->type == OPT_ALIAS ||
|
option->type == OPT_ALIAS ||
|
||||||
!strcmp(option->name, "_template_"))
|
!strcmp(option->name, "_template_"))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* Is there anything to be printed anyway? */
|
/* Is there anything to be printed anyway? */
|
||||||
if (option->type == OPT_TREE
|
if (option->type == OPT_TREE
|
||||||
&& !check_nonempty_tree(option->value.tree))
|
? !check_nonempty_tree(option->value.tree)
|
||||||
|
: !(option->flags & OPT_MUST_SAVE))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* We won't pop out the description when we're in autocreate
|
/* We won't pop out the description when we're in autocreate
|
||||||
@ -920,10 +956,6 @@ smart_config_string(struct string *str, int print_comment, int i18n,
|
|||||||
|
|
||||||
fn(str, option, path, depth, /*pc*/1, 3, i18n);
|
fn(str, option, path, depth, /*pc*/1, 3, i18n);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO: We should maybe clear the touched flag only when really
|
|
||||||
* saving the stuff...? --pasky */
|
|
||||||
option->flags &= ~OPT_TOUCHED;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,15 +24,33 @@ enum option_flags {
|
|||||||
* this category when adding an option. The 'template' for the added
|
* this category when adding an option. The 'template' for the added
|
||||||
* hiearchy piece (category) is stored as "_template_" category. */
|
* hiearchy piece (category) is stored as "_template_" category. */
|
||||||
OPT_AUTOCREATE = 2,
|
OPT_AUTOCREATE = 2,
|
||||||
/* This is used just for marking various options for some very dark,
|
/* The option has been modified in some way and must be saved
|
||||||
* nasty and dirty purposes. This watermarking should be kept inside
|
* to elinks.conf. ELinks uses this flag only while it is
|
||||||
* some very closed and clearly bounded piece of ELinks module, not
|
* saving the options. When the config.saving_style option
|
||||||
* spreaded along whole ELinks code, and you should clear it everytime
|
* has value 3, saving works like this:
|
||||||
* when sneaking outside of the module (except some trivial common
|
* - First, ELinks sets OPT_MUST_SAVE in the options that have
|
||||||
* utility functions). Basically, you don't want to use this flag
|
* OPT_TOUCHED or OPT_DELETED, and clears it in the rest.
|
||||||
* normally ;). It doesn't affect how the option is handled by common
|
* - ELinks then parses the old configuration file and any
|
||||||
* option handling functions in any way. */
|
* files named in "include" commands.
|
||||||
OPT_WATERMARK = 4,
|
* - If the old configuration file contains a "set" or "unset"
|
||||||
|
* command for this option, ELinks rewrites the command and
|
||||||
|
* clears OPT_MUST_SAVE.
|
||||||
|
* - If an included file contains a "set" or "unset" command
|
||||||
|
* for this option, ELinks compares the value of the option
|
||||||
|
* to the value given in the command. ELinks clears
|
||||||
|
* OPT_MUST_SAVE if the values match, or sets it if they
|
||||||
|
* differ.
|
||||||
|
* - After ELinks has rewritten the configuration file and
|
||||||
|
* parsed the included files, it appends the options that
|
||||||
|
* still have the OPT_MUST_SAVE flag.
|
||||||
|
* Other saving styles are variants of this:
|
||||||
|
* - 0: ELinks does not append any options to the
|
||||||
|
* configuration file. So OPT_MUST_SAVE has no effect.
|
||||||
|
* - 1: ELinks initially sets OPT_MUST_SAVE in all options,
|
||||||
|
* regardless of OPT_TOUCHED and OPT_DELETED.
|
||||||
|
* - 2: ELinks initially sets OPT_MUST_SAVE in all options,
|
||||||
|
* and does not read any configuration files. */
|
||||||
|
OPT_MUST_SAVE = 4,
|
||||||
/* This is used to mark options modified after the last save. That's
|
/* This is used to mark options modified after the last save. That's
|
||||||
* being useful if you want to save only the options whose value
|
* being useful if you want to save only the options whose value
|
||||||
* changed. */
|
* changed. */
|
||||||
@ -161,8 +179,8 @@ extern void register_change_hooks(const struct change_hook_info *change_hooks);
|
|||||||
|
|
||||||
|
|
||||||
extern LIST_OF(struct option) *init_options_tree(void);
|
extern LIST_OF(struct option) *init_options_tree(void);
|
||||||
extern void unmark_options_tree(LIST_OF(struct option) *);
|
extern void prepare_mustsave_flags(LIST_OF(struct option) *, int set_all);
|
||||||
void watermark_deleted_options(LIST_OF(struct option) *);
|
extern void untouch_options(LIST_OF(struct option) *);
|
||||||
|
|
||||||
extern void smart_config_string(struct string *, int, int,
|
extern void smart_config_string(struct string *, int, int,
|
||||||
LIST_OF(struct option) *, unsigned char *, int,
|
LIST_OF(struct option) *, unsigned char *, int,
|
||||||
@ -224,6 +242,7 @@ extern void checkout_option_values(struct option_resolver *resolvers,
|
|||||||
|
|
||||||
extern struct option *get_opt_rec(struct option *, const unsigned char *);
|
extern struct option *get_opt_rec(struct option *, const unsigned char *);
|
||||||
extern struct option *get_opt_rec_real(struct option *, const unsigned char *);
|
extern struct option *get_opt_rec_real(struct option *, const unsigned char *);
|
||||||
|
struct option *indirect_option(struct option *);
|
||||||
#ifdef CONFIG_DEBUG
|
#ifdef CONFIG_DEBUG
|
||||||
extern union option_value *get_opt_(unsigned char *, int, enum option_type, struct option *, unsigned char *, struct session *);
|
extern union option_value *get_opt_(unsigned char *, int, enum option_type, struct option *, unsigned char *, struct session *);
|
||||||
#define get_opt(tree, name, ses, type) get_opt_(__FILE__, __LINE__, type, tree, name, ses)
|
#define get_opt(tree, name, ses, type) get_opt_(__FILE__, __LINE__, type, tree, name, ses)
|
||||||
|
@ -90,19 +90,6 @@ exec_cmd(struct option *o, unsigned char ***argv, int *argc)
|
|||||||
* possibly changing ptr to structure containing target name and pointer to
|
* possibly changing ptr to structure containing target name and pointer to
|
||||||
* options list? --pasky */
|
* options list? --pasky */
|
||||||
|
|
||||||
#define wrap_or_(name_, call_, ret_) \
|
|
||||||
{ \
|
|
||||||
struct option *real = get_opt_rec(config_options, opt->value.string); \
|
|
||||||
\
|
|
||||||
assertm(real != NULL, "%s aliased to unknown option %s!", opt->name, opt->value.string); \
|
|
||||||
if_assert_failed { return ret_; } \
|
|
||||||
\
|
|
||||||
if (option_types[real->type].name_) \
|
|
||||||
return option_types[real->type].call_; \
|
|
||||||
\
|
|
||||||
return ret_; \
|
|
||||||
}
|
|
||||||
|
|
||||||
static unsigned char *
|
static unsigned char *
|
||||||
redir_cmd(struct option *opt, unsigned char ***argv, int *argc)
|
redir_cmd(struct option *opt, unsigned char ***argv, int *argc)
|
||||||
{
|
{
|
||||||
@ -124,7 +111,22 @@ redir_cmd(struct option *opt, unsigned char ***argv, int *argc)
|
|||||||
|
|
||||||
static unsigned char *
|
static unsigned char *
|
||||||
redir_rd(struct option *opt, unsigned char **file, int *line)
|
redir_rd(struct option *opt, unsigned char **file, int *line)
|
||||||
wrap_or_(read, read(real, file, line), NULL);
|
{
|
||||||
|
struct option *real = get_opt_rec(config_options, opt->value.string);
|
||||||
|
unsigned char *ret = NULL;
|
||||||
|
|
||||||
|
assertm(real != NULL, "%s aliased to unknown option %s!", opt->name, opt->value.string);
|
||||||
|
if_assert_failed { return ret; }
|
||||||
|
|
||||||
|
if (option_types[real->type].read) {
|
||||||
|
ret = option_types[real->type].read(real, file, line);
|
||||||
|
if (ret && (opt->flags & OPT_ALIAS_NEGATE) && real->type == OPT_BOOL) {
|
||||||
|
*(long *) ret = !*(long *) ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
redir_wr(struct option *opt, struct string *string)
|
redir_wr(struct option *opt, struct string *string)
|
||||||
@ -148,10 +150,35 @@ redir_set(struct option *opt, unsigned char *str)
|
|||||||
if_assert_failed { return ret; }
|
if_assert_failed { return ret; }
|
||||||
|
|
||||||
if (option_types[real->type].set) {
|
if (option_types[real->type].set) {
|
||||||
ret = option_types[real->type].set(real, str);
|
long negated;
|
||||||
|
|
||||||
if ((opt->flags & OPT_ALIAS_NEGATE) && real->type == OPT_BOOL) {
|
if ((opt->flags & OPT_ALIAS_NEGATE) && real->type == OPT_BOOL) {
|
||||||
real->value.number = !real->value.number;
|
negated = !*(long *) str;
|
||||||
|
str = (unsigned char *) &negated;
|
||||||
}
|
}
|
||||||
|
ret = option_types[real->type].set(real, str);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
redir_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
struct option *real = get_opt_rec(config_options, opt->value.string);
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
assertm(real != NULL, "%s aliased to unknown option %s!", opt->name, opt->value.string);
|
||||||
|
if_assert_failed { return ret; }
|
||||||
|
|
||||||
|
if (option_types[real->type].equals) {
|
||||||
|
long negated;
|
||||||
|
|
||||||
|
if ((opt->flags & OPT_ALIAS_NEGATE) && real->type == OPT_BOOL) {
|
||||||
|
negated = !*(const long *) str;
|
||||||
|
str = (unsigned char *) &negated;
|
||||||
|
}
|
||||||
|
ret = option_types[real->type].equals(real, str);
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
@ -202,6 +229,12 @@ num_set(struct option *opt, unsigned char *str)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
num_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
return str && opt->value.number == *(const long *) str;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
num_wr(struct option *option, struct string *string)
|
num_wr(struct option *option, struct string *string)
|
||||||
{
|
{
|
||||||
@ -216,6 +249,12 @@ long_set(struct option *opt, unsigned char *str)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
long_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
return str && opt->value.big_number == *(const long *) str;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
long_wr(struct option *option, struct string *string)
|
long_wr(struct option *option, struct string *string)
|
||||||
{
|
{
|
||||||
@ -248,9 +287,9 @@ str_rd(struct option *opt, unsigned char **file, int *line)
|
|||||||
* thus we will never test for it in while () condition
|
* thus we will never test for it in while () condition
|
||||||
* and we will treat it just as '"', ignoring the
|
* and we will treat it just as '"', ignoring the
|
||||||
* backslash itself. */
|
* backslash itself. */
|
||||||
if (isquote(str[1])) str++;
|
else if (isquote(str[1])) str++;
|
||||||
/* \\ means \. */
|
/* \\ means \. */
|
||||||
if (str[1] == '\\') str++;
|
else if (str[1] == '\\') str++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*str == '\n') (*line)++;
|
if (*str == '\n') (*line)++;
|
||||||
@ -285,6 +324,12 @@ str_set(struct option *opt, unsigned char *str)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
str_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
return str && strcmp(opt->value.string, str) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
str_wr(struct option *o, struct string *s)
|
str_wr(struct option *o, struct string *s)
|
||||||
{
|
{
|
||||||
@ -315,6 +360,12 @@ cp_set(struct option *opt, unsigned char *str)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cp_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
return str && get_cp_index(str) == opt->value.number;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
cp_wr(struct option *o, struct string *s)
|
cp_wr(struct option *o, struct string *s)
|
||||||
{
|
{
|
||||||
@ -334,6 +385,16 @@ lang_set(struct option *opt, unsigned char *str)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
lang_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
#ifdef CONFIG_NLS
|
||||||
|
return str && name_to_language(str) == opt->value.number;
|
||||||
|
#else
|
||||||
|
return 1; /* All languages are the same. */
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
lang_wr(struct option *o, struct string *s)
|
lang_wr(struct option *o, struct string *s)
|
||||||
{
|
{
|
||||||
@ -355,6 +416,15 @@ color_set(struct option *opt, unsigned char *str)
|
|||||||
return !decode_color(str, strlen(str), &opt->value.color);
|
return !decode_color(str, strlen(str), &opt->value.color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
color_eq(struct option *opt, const unsigned char *str)
|
||||||
|
{
|
||||||
|
color_T color;
|
||||||
|
|
||||||
|
return str && !decode_color(str, strlen(str), &color)
|
||||||
|
&& color == opt->value.color;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
color_wr(struct option *opt, struct string *str)
|
color_wr(struct option *opt, struct string *str)
|
||||||
{
|
{
|
||||||
@ -402,29 +472,29 @@ const struct option_type_info option_types[] = {
|
|||||||
/* The OPT_ comments below are here to be found by grep. */
|
/* The OPT_ comments below are here to be found by grep. */
|
||||||
|
|
||||||
/* OPT_BOOL */
|
/* OPT_BOOL */
|
||||||
{ N_("Boolean"), bool_cmd, num_rd, num_wr, NULL, num_set, N_("[0|1]") },
|
{ N_("Boolean"), bool_cmd, num_rd, num_wr, NULL, num_set, num_eq, N_("[0|1]") },
|
||||||
/* OPT_INT */
|
/* OPT_INT */
|
||||||
{ N_("Integer"), gen_cmd, num_rd, num_wr, NULL, num_set, N_("<num>") },
|
{ N_("Integer"), gen_cmd, num_rd, num_wr, NULL, num_set, num_eq, N_("<num>") },
|
||||||
/* OPT_LONG */
|
/* OPT_LONG */
|
||||||
{ N_("Longint"), gen_cmd, num_rd, long_wr, NULL, long_set, N_("<num>") },
|
{ N_("Longint"), gen_cmd, num_rd, long_wr, NULL, long_set, long_eq, N_("<num>") },
|
||||||
/* OPT_STRING */
|
/* OPT_STRING */
|
||||||
{ N_("String"), gen_cmd, str_rd, str_wr, str_dup, str_set, N_("<str>") },
|
{ N_("String"), gen_cmd, str_rd, str_wr, str_dup, str_set, str_eq, N_("<str>") },
|
||||||
|
|
||||||
/* OPT_CODEPAGE */
|
/* OPT_CODEPAGE */
|
||||||
{ N_("Codepage"), gen_cmd, str_rd, cp_wr, NULL, cp_set, N_("<codepage>") },
|
{ N_("Codepage"), gen_cmd, str_rd, cp_wr, NULL, cp_set, cp_eq, N_("<codepage>") },
|
||||||
/* OPT_LANGUAGE */
|
/* OPT_LANGUAGE */
|
||||||
{ N_("Language"), gen_cmd, str_rd, lang_wr, NULL, lang_set, N_("<language>") },
|
{ N_("Language"), gen_cmd, str_rd, lang_wr, NULL, lang_set, lang_eq, N_("<language>") },
|
||||||
/* OPT_COLOR */
|
/* OPT_COLOR */
|
||||||
{ N_("Color"), gen_cmd, str_rd, color_wr, NULL, color_set, N_("<color|#rrggbb>") },
|
{ N_("Color"), gen_cmd, str_rd, color_wr, NULL, color_set, color_eq, N_("<color|#rrggbb>") },
|
||||||
|
|
||||||
/* OPT_COMMAND */
|
/* OPT_COMMAND */
|
||||||
{ N_("Special"), exec_cmd, NULL, NULL, NULL, NULL, "" },
|
{ N_("Special"), exec_cmd, NULL, NULL, NULL, NULL, NULL, "" },
|
||||||
|
|
||||||
/* OPT_ALIAS */
|
/* OPT_ALIAS */
|
||||||
{ N_("Alias"), redir_cmd, redir_rd, redir_wr, NULL, redir_set, "" },
|
{ N_("Alias"), redir_cmd, redir_rd, redir_wr, NULL, redir_set, redir_eq, "" },
|
||||||
|
|
||||||
/* OPT_TREE */
|
/* OPT_TREE */
|
||||||
{ N_("Folder"), NULL, NULL, NULL, tree_dup, NULL, "" },
|
{ N_("Folder"), NULL, NULL, NULL, tree_dup, NULL, NULL, "" },
|
||||||
};
|
};
|
||||||
|
|
||||||
unsigned char *
|
unsigned char *
|
||||||
|
@ -11,6 +11,7 @@ struct option_type_info {
|
|||||||
void (*write)(struct option *, struct string *);
|
void (*write)(struct option *, struct string *);
|
||||||
void (*dup)(struct option *, struct option *, int);
|
void (*dup)(struct option *, struct option *, int);
|
||||||
int (*set)(struct option *, unsigned char *);
|
int (*set)(struct option *, unsigned char *);
|
||||||
|
int (*equals)(struct option *, const unsigned char *);
|
||||||
unsigned char *help_str;
|
unsigned char *help_str;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -63,7 +63,6 @@ static int
|
|||||||
exmode_confcmd_handler(struct session *ses, unsigned char *command,
|
exmode_confcmd_handler(struct session *ses, unsigned char *command,
|
||||||
unsigned char *args)
|
unsigned char *args)
|
||||||
{
|
{
|
||||||
int dummyline = 0;
|
|
||||||
enum parse_error err;
|
enum parse_error err;
|
||||||
|
|
||||||
assert(ses && command && args);
|
assert(ses && command && args);
|
||||||
@ -74,8 +73,7 @@ exmode_confcmd_handler(struct session *ses, unsigned char *command,
|
|||||||
/* Undo the arguments separation. */
|
/* Undo the arguments separation. */
|
||||||
if (*args) *(--args) = ' ';
|
if (*args) *(--args) = ' ';
|
||||||
|
|
||||||
err = parse_config_command(config_options, &command, &dummyline, NULL,
|
err = parse_config_exmode_command(command);
|
||||||
0);
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1371,7 +1371,7 @@ flush:
|
|||||||
|
|
||||||
#ifndef USE_FASTFIND
|
#ifndef USE_FASTFIND
|
||||||
int
|
int
|
||||||
get_cp_index(unsigned char *name)
|
get_cp_index(const unsigned char *name)
|
||||||
{
|
{
|
||||||
int i, a;
|
int i, a;
|
||||||
int syscp = 0;
|
int syscp = 0;
|
||||||
@ -1452,7 +1452,7 @@ static struct fastfind_index ff_charsets_index
|
|||||||
/* It searchs for a charset named @name or one of its aliases and
|
/* It searchs for a charset named @name or one of its aliases and
|
||||||
* returns index for it or -1 if not found. */
|
* returns index for it or -1 if not found. */
|
||||||
int
|
int
|
||||||
get_cp_index(unsigned char *name)
|
get_cp_index(const unsigned char *name)
|
||||||
{
|
{
|
||||||
const struct codepage_desc *codepage;
|
const struct codepage_desc *codepage;
|
||||||
int syscp = 0;
|
int syscp = 0;
|
||||||
|
@ -117,7 +117,7 @@ unsigned char *convert_string(struct conv_table *convert_table,
|
|||||||
void (*callback)(void *data, unsigned char *buf, int buflen),
|
void (*callback)(void *data, unsigned char *buf, int buflen),
|
||||||
void *callback_data);
|
void *callback_data);
|
||||||
|
|
||||||
int get_cp_index(unsigned char *);
|
int get_cp_index(const unsigned char *);
|
||||||
unsigned char *get_cp_name(int);
|
unsigned char *get_cp_name(int);
|
||||||
unsigned char *get_cp_config_name(int);
|
unsigned char *get_cp_config_name(int);
|
||||||
unsigned char *get_cp_mime_name(int);
|
unsigned char *get_cp_mime_name(int);
|
||||||
|
@ -147,7 +147,7 @@ language_to_iso639(int language)
|
|||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
name_to_language(unsigned char *name)
|
name_to_language(const unsigned char *name)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -170,7 +170,7 @@ struct language {
|
|||||||
extern struct language languages[];
|
extern struct language languages[];
|
||||||
|
|
||||||
/* These two calls return 1 (english) if the code/name wasn't found. */
|
/* These two calls return 1 (english) if the code/name wasn't found. */
|
||||||
extern int name_to_language(unsigned char *name);
|
extern int name_to_language(const unsigned char *name);
|
||||||
extern int iso639_to_language(unsigned char *iso639);
|
extern int iso639_to_language(unsigned char *iso639);
|
||||||
|
|
||||||
extern unsigned char *language_to_name(int language);
|
extern unsigned char *language_to_name(int language);
|
||||||
|
@ -100,6 +100,7 @@ struct itrm {
|
|||||||
void *mouse_h; /**< Mouse handle */
|
void *mouse_h; /**< Mouse handle */
|
||||||
unsigned char *orig_title; /**< For restoring window title */
|
unsigned char *orig_title; /**< For restoring window title */
|
||||||
|
|
||||||
|
int verase; /**< Byte to map to KBD_BS, or -1 */
|
||||||
unsigned int blocked:1; /**< Whether it was blocked */
|
unsigned int blocked:1; /**< Whether it was blocked */
|
||||||
unsigned int altscreen:1; /**< Whether to use alternate screen */
|
unsigned int altscreen:1; /**< Whether to use alternate screen */
|
||||||
unsigned int touched_title:1; /**< Whether the term title was changed */
|
unsigned int touched_title:1; /**< Whether the term title was changed */
|
||||||
|
@ -231,14 +231,25 @@ get_terminal_name(unsigned char name[MAX_TERM_LEN])
|
|||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
setraw(int fd, struct termios *p)
|
setraw(struct itrm *itrm, int save_orig)
|
||||||
{
|
{
|
||||||
struct termios t;
|
struct termios t;
|
||||||
|
long vdisable;
|
||||||
|
|
||||||
memset(&t, 0, sizeof(t));
|
memset(&t, 0, sizeof(t));
|
||||||
if (tcgetattr(fd, &t)) return -1;
|
if (tcgetattr(itrm->in.ctl, &t)) return -1;
|
||||||
|
|
||||||
if (p) copy_struct(p, &t);
|
if (save_orig) copy_struct(&itrm->t, &t);
|
||||||
|
|
||||||
|
#ifdef _POSIX_VDISABLE
|
||||||
|
vdisable = _POSIX_VDISABLE;
|
||||||
|
#else
|
||||||
|
vdisable = fpathconf(itrm->in.ctl, _PC_VDISABLE);
|
||||||
|
#endif
|
||||||
|
if (vdisable != -1 && t.c_cc[VERASE] == vdisable)
|
||||||
|
itrm->verase = -1;
|
||||||
|
else
|
||||||
|
itrm->verase = (unsigned char) t.c_cc[VERASE];
|
||||||
|
|
||||||
elinks_cfmakeraw(&t);
|
elinks_cfmakeraw(&t);
|
||||||
t.c_lflag |= ISIG;
|
t.c_lflag |= ISIG;
|
||||||
@ -246,7 +257,7 @@ setraw(int fd, struct termios *p)
|
|||||||
t.c_lflag |= TOSTOP;
|
t.c_lflag |= TOSTOP;
|
||||||
#endif
|
#endif
|
||||||
t.c_oflag |= OPOST;
|
t.c_oflag |= OPOST;
|
||||||
if (tcsetattr(fd, TCSANOW, &t)) return -1;
|
if (tcsetattr(itrm->in.ctl, TCSANOW, &t)) return -1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -326,7 +337,7 @@ handle_trm(int std_in, int std_out, int sock_in, int sock_out, int ctl_in,
|
|||||||
itrm->altscreen = 1;
|
itrm->altscreen = 1;
|
||||||
|
|
||||||
if (!remote) {
|
if (!remote) {
|
||||||
if (ctl_in >= 0) setraw(ctl_in, &itrm->t);
|
if (ctl_in >= 0) setraw(itrm, 1);
|
||||||
send_init_sequence(std_out, itrm->altscreen);
|
send_init_sequence(std_out, itrm->altscreen);
|
||||||
handle_terminal_resize(ctl_in, resize_terminal);
|
handle_terminal_resize(ctl_in, resize_terminal);
|
||||||
#ifdef CONFIG_MOUSE
|
#ifdef CONFIG_MOUSE
|
||||||
@ -373,7 +384,7 @@ unblock_itrm(void)
|
|||||||
{
|
{
|
||||||
if (!ditrm) return -1;
|
if (!ditrm) return -1;
|
||||||
|
|
||||||
if (ditrm->in.ctl >= 0 && setraw(ditrm->in.ctl, NULL)) return -1;
|
if (ditrm->in.ctl >= 0 && setraw(ditrm, 0)) return -1;
|
||||||
ditrm->blocked = 0;
|
ditrm->blocked = 0;
|
||||||
send_init_sequence(ditrm->out.std, ditrm->altscreen);
|
send_init_sequence(ditrm->out.std, ditrm->altscreen);
|
||||||
|
|
||||||
@ -949,26 +960,18 @@ decode_terminal_application_key(struct itrm *itrm, struct interlink_event *ev)
|
|||||||
/** Initialize @a *ev to match the byte @a key received from the terminal.
|
/** Initialize @a *ev to match the byte @a key received from the terminal.
|
||||||
* @a key must not be a value from enum term_event_special_key. */
|
* @a key must not be a value from enum term_event_special_key. */
|
||||||
static void
|
static void
|
||||||
set_kbd_event(struct interlink_event *ev,
|
set_kbd_event(const struct itrm *itrm, struct interlink_event *ev,
|
||||||
int key, term_event_modifier_T modifier)
|
int key, term_event_modifier_T modifier)
|
||||||
{
|
{
|
||||||
switch (key) {
|
if (key == itrm->verase)
|
||||||
|
key = KBD_BS;
|
||||||
|
else switch (key) {
|
||||||
case ASCII_TAB:
|
case ASCII_TAB:
|
||||||
key = KBD_TAB;
|
key = KBD_TAB;
|
||||||
break;
|
break;
|
||||||
#if defined(HAVE_SYS_CONSIO_H) || defined(HAVE_MACHINE_CONSOLE_H) /* BSD */
|
case ASCII_DEL: /* often overridden by itrm->verase above */
|
||||||
case ASCII_BS:
|
|
||||||
key = KBD_BS;
|
|
||||||
break;
|
|
||||||
case ASCII_DEL:
|
|
||||||
key = KBD_DEL;
|
key = KBD_DEL;
|
||||||
break;
|
break;
|
||||||
#else
|
|
||||||
case ASCII_BS:
|
|
||||||
case ASCII_DEL:
|
|
||||||
key = KBD_BS;
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
case ASCII_LF:
|
case ASCII_LF:
|
||||||
case ASCII_CR:
|
case ASCII_CR:
|
||||||
key = KBD_ENTER;
|
key = KBD_ENTER;
|
||||||
@ -978,6 +981,7 @@ set_kbd_event(struct interlink_event *ev,
|
|||||||
key = KBD_ESC;
|
key = KBD_ESC;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case ASCII_BS: /* often overridden by itrm->verase above */
|
||||||
default:
|
default:
|
||||||
if (key < ' ') {
|
if (key < ' ') {
|
||||||
key += 'A' - 1;
|
key += 'A' - 1;
|
||||||
@ -1010,10 +1014,10 @@ kbd_timeout(struct itrm *itrm)
|
|||||||
|
|
||||||
if (itrm->in.queue.len >= 2 && itrm->in.queue.data[0] == ASCII_ESC) {
|
if (itrm->in.queue.len >= 2 && itrm->in.queue.data[0] == ASCII_ESC) {
|
||||||
/* This is used for ESC [ and ESC O. */
|
/* This is used for ESC [ and ESC O. */
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[1], KBD_MOD_ALT);
|
set_kbd_event(itrm, &ev, itrm->in.queue.data[1], KBD_MOD_ALT);
|
||||||
el = 2;
|
el = 2;
|
||||||
} else {
|
} else {
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[0], KBD_MOD_NONE);
|
set_kbd_event(itrm, &ev, itrm->in.queue.data[0], KBD_MOD_NONE);
|
||||||
el = 1;
|
el = 1;
|
||||||
}
|
}
|
||||||
itrm->bracketed_pasting = 0;
|
itrm->bracketed_pasting = 0;
|
||||||
@ -1098,20 +1102,22 @@ process_queue(struct itrm *itrm)
|
|||||||
* by an escape sequence. Treat it as
|
* by an escape sequence. Treat it as
|
||||||
* a standalone Esc. */
|
* a standalone Esc. */
|
||||||
el = 1;
|
el = 1;
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[0],
|
set_kbd_event(itrm, &ev,
|
||||||
|
itrm->in.queue.data[0],
|
||||||
KBD_MOD_NONE);
|
KBD_MOD_NONE);
|
||||||
} else {
|
} else {
|
||||||
/* The second ESC of ESC ESC is not the
|
/* The second ESC of ESC ESC is not the
|
||||||
* beginning of any known escape sequence.
|
* beginning of any known escape sequence.
|
||||||
* This must be Alt-Esc, then. */
|
* This must be Alt-Esc, then. */
|
||||||
el = 2;
|
el = 2;
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[1],
|
set_kbd_event(itrm, &ev,
|
||||||
|
itrm->in.queue.data[1],
|
||||||
KBD_MOD_ALT);
|
KBD_MOD_ALT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (el == 0) { /* Begins with ESC, but none of the above */
|
if (el == 0) { /* Begins with ESC, but none of the above */
|
||||||
el = 2;
|
el = 2;
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[1],
|
set_kbd_event(itrm, &ev, itrm->in.queue.data[1],
|
||||||
KBD_MOD_ALT);
|
KBD_MOD_ALT);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1132,7 +1138,7 @@ process_queue(struct itrm *itrm)
|
|||||||
|
|
||||||
if (el == 0) {
|
if (el == 0) {
|
||||||
el = 1;
|
el = 1;
|
||||||
set_kbd_event(&ev, itrm->in.queue.data[0],
|
set_kbd_event(itrm, &ev, itrm->in.queue.data[0],
|
||||||
itrm->bracketed_pasting ? KBD_MOD_PASTE : KBD_MOD_NONE);
|
itrm->bracketed_pasting ? KBD_MOD_PASTE : KBD_MOD_NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user