Johannes Schindelin <[email protected]> writes:
> diff --git a/fsck.c b/fsck.c
> index da5717c..8c3caff 100644
> --- a/fsck.c
> +++ b/fsck.c
> @@ -103,13 +103,85 @@ static int fsck_msg_type(enum fsck_msg_id msg_id,
> {
> int msg_type;
>
> - msg_type = msg_id_info[msg_id].msg_type;
> - if (options->strict && msg_type == FSCK_WARN)
> - msg_type = FSCK_ERROR;
> + assert(msg_id >= 0 && msg_id < FSCK_MSG_MAX);
> +
> + if (options->msg_type)
> + msg_type = options->msg_type[msg_id];
> + else {
> + msg_type = msg_id_info[msg_id].msg_type;
> + if (options->strict && msg_type == FSCK_WARN)
> + msg_type = FSCK_ERROR;
> + }
>
> return msg_type;
> }
Nice.
> +static inline int substrcmp(const char *string, int len, const char *match)
> +{
> + int match_len = strlen(match);
> + if (match_len != len)
> + return -1;
> + return memcmp(string, match, len);
> +}
What this does looks suspiciously like starts_with(), but its name
"substrcmp()" does not give any hint that this is about the beginnig
part of "string"; if anything, it gives a wrong hint that it may be
any substring. prefixcmp() might be a better name but that was the
old name for !starts_with() so we cannot use it here. It is a
mouthful, but starts_with_counted() may be.
But the whole thing may be moot.
If we take the "why not upcase the end-user string upfront"
suggestion from the previous review, fsck_set_msg_types() would have
an upcased copy of the end-user string that it can muck with; it can
turn "badfoo=error,poorbar=..." into "BADFOO=error,POORBAR=..."
that is stored in its own writable memory (possibly a strbuf), and
at that point it can afford to NUL-terminate BADFOO=error after
finding where one specification ends with strcspn() before calling
fsck_set_msg_type(), which in turn calls parse_msg_type().
So all parse_msg_type() needs to do is just !strcmp().
> +
> +static int parse_msg_type(const char *str, int len)
> +{
> + if (len < 0)
> + len = strlen(str);
> +
> + if (!substrcmp(str, len, "error"))
> + return FSCK_ERROR;
> + else if (!substrcmp(str, len, "warn"))
> + return FSCK_WARN;
> + else
> + die("Unknown fsck message type: '%.*s'",
> + len, str);
> +}
> +
> +void fsck_set_msg_type(struct fsck_options *options,
> + const char *msg_id, int msg_id_len,
> + const char *msg_type, int msg_type_len)
> +{
> + int id = parse_msg_id(msg_id, msg_id_len), type;
> +
> + if (id < 0)
> + die("Unhandled message id: %.*s", msg_id_len, msg_id);
> + type = parse_msg_type(msg_type, msg_type_len);
> +
> + if (!options->msg_type) {
> + int i;
> + int *msg_type = xmalloc(sizeof(int) * FSCK_MSG_MAX);
> + for (i = 0; i < FSCK_MSG_MAX; i++)
> + msg_type[i] = fsck_msg_type(i, options);
> + options->msg_type = msg_type;
> + }
> +
> + options->msg_type[id] = type;
> +}
> +
> +void fsck_set_msg_types(struct fsck_options *options, const char *values)
> +{
> + while (*values) {
> + int len = strcspn(values, " ,|"), equal;
> +
> + if (!len) {
> + values++;
> + continue;
> + }
> +
> + for (equal = 0; equal < len; equal++)
> + if (values[equal] == '=' || values[equal] == ':')
> + break;
> +
> + if (equal == len)
> + die("Missing '=': '%.*s'", len, values);
> +
> + fsck_set_msg_type(options, values, equal,
> + values + equal + 1, len - equal - 1);
> + values += len;
> + }
> +}
> +
> __attribute__((format (printf, 4, 5)))
> static int report(struct fsck_options *options, struct object *object,
> enum fsck_msg_id id, const char *fmt, ...)
> @@ -599,6 +671,10 @@ int fsck_object(struct object *obj, void *data, unsigned
> long size,
>
> int fsck_error_function(struct object *obj, int msg_type, const char
> *message)
> {
> + if (msg_type == FSCK_WARN) {
> + warning("object %s: %s", sha1_to_hex(obj->sha1), message);
> + return 0;
> + }
> error("object %s: %s", sha1_to_hex(obj->sha1), message);
> return 1;
> }
> diff --git a/fsck.h b/fsck.h
> index f6f268a..edb4540 100644
> --- a/fsck.h
> +++ b/fsck.h
> @@ -6,6 +6,11 @@
>
> struct fsck_options;
>
> +void fsck_set_msg_type(struct fsck_options *options,
> + const char *msg_id, int msg_id_len,
> + const char *msg_type, int msg_type_len);
> +void fsck_set_msg_types(struct fsck_options *options, const char *values);
> +
> /*
> * callback function for fsck_walk
> * type is the expected type of the object or OBJ_ANY
> @@ -25,10 +30,11 @@ struct fsck_options {
> fsck_walk_func walk;
> fsck_error error_func;
> unsigned strict:1;
> + int *msg_type;
> };
>
> -#define FSCK_OPTIONS_DEFAULT { NULL, fsck_error_function, 0 }
> -#define FSCK_OPTIONS_STRICT { NULL, fsck_error_function, 1 }
> +#define FSCK_OPTIONS_DEFAULT { NULL, fsck_error_function, 0, NULL }
> +#define FSCK_OPTIONS_STRICT { NULL, fsck_error_function, 1, NULL }
>
> /* descend in all linked child objects
> * the return value is:
--
To unsubscribe from this list: send the line "unsubscribe git" in