Hi,
here is the 6th iteration of my "status delayed" topic. Patch 1 to 3 are
minor t0021 test adjustments and haven't been changed since v3. Patch 4
is new and a minor style adjustment. Patch 5 is a minor "extract method"
refactoring in convert.c with an additional minor style adjustment in
this round. Patch 6 is the new feature.
Changes since v5 (full inter diff below!):
* commit message improvements (Torsten/Peff)
* reorder checkout struct to keep base_dir* member together (Peff)
* remove unnecessary initialization in CHECKOUT_INIT (Peff)
* adhere to Git style (";" instead "{}") for empty if blocks (Torsten)
* flag fields before flags (Peff)
* make operator precedence explicit with parenthesis (Peff)
* use skip_prefix() to simplify parsing and silently parse unknown keys (Peff)
* add ce_delay_state explanation (Peff)
* print files that the filter delayed but not delivered; clean up data
structure (Peff)
If you review this series then please read the "Delay" section in
"Documentation/gitattributes.txt" first for an overview of the delay mechanism.
The changes in "t/t0021/rot13-filter.pl" are easier to review if you ignore
whitespace changes.
I also noticed that the diff around the "static int apply_multi_file_filter"
looks really weird as my new function "int async_query_available_blobs" is
right behind it with similar lines.
Thanks,
Lars
RFC: http://public-inbox.org/git/[email protected]/
v1: http://public-inbox.org/git/[email protected]/
v2: http://public-inbox.org/git/[email protected]/
v3: http://public-inbox.org/git/[email protected]/
v4: http://public-inbox.org/git/[email protected]/
v5: http://public-inbox.org/git/[email protected]/
Base Ref: master
Web-Diff: https://github.com/larsxschneider/git/commit/ab3535aa60
Checkout: git fetch https://github.com/larsxschneider/git
filter-process/delay-v6 && git checkout ab3535aa60
Interdiff (v5..v6):
diff --git a/cache.h b/cache.h
index 523ead1d95..2ec12c4477 100644
--- a/cache.h
+++ b/cache.h
@@ -1543,14 +1543,14 @@ extern int ident_cmp(const struct ident_split *, const
struct ident_split *);
struct checkout {
struct index_state *istate;
const char *base_dir;
- struct delayed_checkout *delayed_checkout;
int base_dir_len;
+ struct delayed_checkout *delayed_checkout;
unsigned force:1,
quiet:1,
not_new:1,
refresh_cache:1;
};
-#define CHECKOUT_INIT { NULL, "", NULL }
+#define CHECKOUT_INIT { NULL, "" }
#define TEMPORARY_FILENAME_LENGTH 25
diff --git a/convert.c b/convert.c
index c4df174378..6452ab546a 100644
--- a/convert.c
+++ b/convert.c
@@ -572,9 +572,9 @@ static int start_multi_file_filter_fn(struct
subprocess_entry *subprocess)
static void handle_filter_error(const struct strbuf *filter_status,
struct cmd2process *entry,
const unsigned int wanted_capability) {
- if (!strcmp(filter_status->buf, "error")) {
- /* The filter signaled a problem with the file. */
- } else if (!strcmp(filter_status->buf, "abort") && wanted_capability) {
+ if (!strcmp(filter_status->buf, "error"))
+ ; /* The filter signaled a problem with the file. */
+ else if (!strcmp(filter_status->buf, "abort") && wanted_capability) {
/*
* The filter signaled a permanent problem. Don't try to filter
* files with the same command for the lifetime of the current
@@ -626,12 +626,12 @@ static int apply_multi_file_filter(const char *path,
const char *src, size_t len
}
process = &entry->subprocess.process;
- if (!(wanted_capability & entry->supported_capabilities))
+ if (!(entry->supported_capabilities & wanted_capability))
return 0;
- if (CAP_CLEAN & wanted_capability)
+ if (wanted_capability & CAP_CLEAN)
filter_type = "clean";
- else if (CAP_SMUDGE & wanted_capability)
+ else if (wanted_capability & CAP_SMUDGE)
filter_type = "smudge";
else
die("unexpected filter type");
@@ -653,7 +653,7 @@ static int apply_multi_file_filter(const char *path, const
char *src, size_t len
if (err)
goto done;
- if (CAP_DELAY & entry->supported_capabilities &&
+ if ((entry->supported_capabilities & CAP_DELAY) &&
dco && dco->state == CE_CAN_DELAY) {
can_delay = 1;
err = packet_write_fmt_gently(process->in, "can-delay=1\n");
@@ -736,16 +736,12 @@ int async_query_available_blobs(const char *cmd, struct
string_list *delayed_pat
if (err)
goto done;
- for (;;) {
- const char* pre = "pathname=";
- const int pre_len = strlen(pre);
- line = packet_read_line(process->out, NULL);
- if (!line)
- break;
- err = strlen(line) <= pre_len || strncmp(line, pre, pre_len);
- if (err)
- goto done;
- string_list_insert(delayed_paths, xstrdup(line+pre_len));
+ while ((line = packet_read_line(process->out, NULL))) {
+ const char *path;
+ if (skip_prefix(line, "pathname=", &path))
+ string_list_insert(delayed_paths, xstrdup(path));
+ else
+ ; /* ignore unknown keys */
}
err = subprocess_read_status(process->out, &filter_status);
@@ -784,9 +780,9 @@ static int apply_filter(const char *path, const char *src,
size_t len,
if (!dst)
return 1;
- if ((CAP_CLEAN & wanted_capability) && !drv->process && drv->clean)
+ if ((wanted_capability & CAP_CLEAN) && !drv->process && drv->clean)
cmd = drv->clean;
- else if ((CAP_SMUDGE & wanted_capability) && !drv->process &&
drv->smudge)
+ else if ((wanted_capability & CAP_SMUDGE) && !drv->process &&
drv->smudge)
cmd = drv->smudge;
if (cmd && *cmd)
diff --git a/convert.h b/convert.h
index c4beaa5101..51f0fb7fbe 100644
--- a/convert.h
+++ b/convert.h
@@ -42,6 +42,10 @@ enum ce_delay_state {
};
struct delayed_checkout {
+ /* State of the currently processed cache entry. If the state is
+ CE_CAN_DELAY, then the filter can change this to CE_DELAYED. If
+ the state is CE_RETRY, then this signals the filter that the cache
+ entry was requested before. */
enum ce_delay_state state;
/* List of filter drivers that signaled delayed blobs. */
struct string_list filters;
diff --git a/entry.c b/entry.c
index c9cb557559..c6d5cc01dc 100644
--- a/entry.c
+++ b/entry.c
@@ -159,10 +159,10 @@ int finish_delayed_checkout(struct checkout *state)
struct string_list_item *filter, *path;
struct delayed_checkout *dco = state->delayed_checkout;
- if (!state->delayed_checkout) {
+ if (!state->delayed_checkout)
return errs;
- }
+ dco->state = CE_RETRY;
while (dco->filters.nr > 0) {
for_each_string_list_item(filter, &dco->filters) {
struct string_list available_paths;
@@ -195,7 +195,7 @@ int finish_delayed_checkout(struct checkout *state)
struct cache_entry* ce = index_file_exists(
state->istate, path->string,
strlen(path->string), 0);
- dco->state = CE_RETRY;
+ assert(dco->state == CE_RETRY);
errs |= (ce ? checkout_entry(ce, state, NULL) :
1);
}
}
@@ -205,6 +205,10 @@ int finish_delayed_checkout(struct checkout *state)
/* At this point we should not have any delayed paths anymore. */
errs |= dco->paths.nr;
+ for_each_string_list_item(path, &dco->paths) {
+ warning(_("%s was not filtered properly."), path->string);
+ }
+ string_list_clear(&dco->paths, 0);
free(dco);
state->delayed_checkout = NULL;
Lars Schneider (6):
t0021: keep filter log files on comparison
t0021: make debug log file name configurable
t0021: write "OUT" only on success
convert: put the flags field before the flag itself for consistent
style
convert: move multiple file filter error handling to separate function
convert: add "status=delayed" to filter process protocol
Documentation/gitattributes.txt | 65 ++++++++++++-
builtin/checkout.c | 3 +
cache.h | 4 +
convert.c | 168 ++++++++++++++++++++++++---------
convert.h | 25 +++++
entry.c | 114 ++++++++++++++++++++++-
t/t0021-conversion.sh | 136 ++++++++++++++++++++-------
t/t0021/rot13-filter.pl | 199 ++++++++++++++++++++++++++--------------
unpack-trees.c | 2 +
9 files changed, 567 insertions(+), 149 deletions(-)
base-commit: 0339965c70d68fd3831c9a5306443c869de3f6a8
--
2.13.1