6 #include "parallel-checkout.h"
7 #include "parse-options.h"
10 static void packet_to_pc_item(const char *buffer
, int len
,
11 struct parallel_checkout_item
*pc_item
)
13 const struct pc_item_fixed_portion
*fixed_portion
;
17 if (len
< sizeof(struct pc_item_fixed_portion
))
18 BUG("checkout worker received too short item (got %dB, exp %dB)",
19 len
, (int)sizeof(struct pc_item_fixed_portion
));
21 fixed_portion
= (struct pc_item_fixed_portion
*)buffer
;
23 if (len
- sizeof(struct pc_item_fixed_portion
) !=
24 fixed_portion
->name_len
+ fixed_portion
->working_tree_encoding_len
)
25 BUG("checkout worker received corrupted item");
27 variant
= buffer
+ sizeof(struct pc_item_fixed_portion
);
30 * Note: the main process uses zero length to communicate that the
31 * encoding is NULL. There is no use case that requires sending an
32 * actual empty string, since convert_attrs() never sets
33 * ca.working_tree_enconding to "".
35 if (fixed_portion
->working_tree_encoding_len
) {
36 encoding
= xmemdupz(variant
,
37 fixed_portion
->working_tree_encoding_len
);
38 variant
+= fixed_portion
->working_tree_encoding_len
;
43 memset(pc_item
, 0, sizeof(*pc_item
));
44 pc_item
->ce
= make_empty_transient_cache_entry(fixed_portion
->name_len
, NULL
);
45 pc_item
->ce
->ce_namelen
= fixed_portion
->name_len
;
46 pc_item
->ce
->ce_mode
= fixed_portion
->ce_mode
;
47 memcpy(pc_item
->ce
->name
, variant
, pc_item
->ce
->ce_namelen
);
48 oidcpy(&pc_item
->ce
->oid
, &fixed_portion
->oid
);
50 pc_item
->id
= fixed_portion
->id
;
51 pc_item
->ca
.crlf_action
= fixed_portion
->crlf_action
;
52 pc_item
->ca
.ident
= fixed_portion
->ident
;
53 pc_item
->ca
.working_tree_encoding
= encoding
;
56 static void report_result(struct parallel_checkout_item
*pc_item
)
58 struct pc_item_result res
= { 0 };
62 res
.status
= pc_item
->status
;
64 if (pc_item
->status
== PC_ITEM_WRITTEN
) {
68 size
= PC_ITEM_RESULT_BASE_SIZE
;
71 packet_write(1, (const char *)&res
, size
);
74 /* Free the worker-side malloced data, but not pc_item itself. */
75 static void release_pc_item_data(struct parallel_checkout_item
*pc_item
)
77 free((char *)pc_item
->ca
.working_tree_encoding
);
78 discard_cache_entry(pc_item
->ce
);
81 static void worker_loop(struct checkout
*state
)
83 struct parallel_checkout_item
*items
= NULL
;
84 size_t i
, nr
= 0, alloc
= 0;
87 int len
= packet_read(0, packet_buffer
, sizeof(packet_buffer
),
91 BUG("packet_read() returned negative value");
95 ALLOC_GROW(items
, nr
+ 1, alloc
);
96 packet_to_pc_item(packet_buffer
, len
, &items
[nr
++]);
99 for (i
= 0; i
< nr
; i
++) {
100 struct parallel_checkout_item
*pc_item
= &items
[i
];
101 write_pc_item(pc_item
, state
);
102 report_result(pc_item
);
103 release_pc_item_data(pc_item
);
111 static const char * const checkout_worker_usage
[] = {
112 N_("git checkout--worker [<options>]"),
116 int cmd_checkout__worker(int argc
, const char **argv
, const char *prefix
)
118 struct checkout state
= CHECKOUT_INIT
;
119 struct option checkout_worker_options
[] = {
120 OPT_STRING(0, "prefix", &state
.base_dir
, N_("string"),
121 N_("when creating files, prepend <string>")),
125 if (argc
== 2 && !strcmp(argv
[1], "-h"))
126 usage_with_options(checkout_worker_usage
,
127 checkout_worker_options
);
129 git_config(git_default_config
, NULL
);
130 argc
= parse_options(argc
, argv
, prefix
, checkout_worker_options
,
131 checkout_worker_usage
, 0);
133 usage_with_options(checkout_worker_usage
, checkout_worker_options
);
136 state
.base_dir_len
= strlen(state
.base_dir
);
139 * Setting this on a worker won't actually update the index. We just
140 * need to tell the checkout machinery to lstat() the written entries,
141 * so that we can send this data back to the main process.
143 state
.refresh_cache
= 1;