Blame


1 5c860e29 2018-03-12 stsp /*
2 f42b1b34 2018-03-12 stsp * Copyright (c) 2017 Martin Pieuchot <mpi@openbsd.org>
3 f42b1b34 2018-03-12 stsp * Copyright (c) 2018 Stefan Sperling <stsp@openbsd.org>
4 5c860e29 2018-03-12 stsp *
5 5c860e29 2018-03-12 stsp * Permission to use, copy, modify, and distribute this software for any
6 5c860e29 2018-03-12 stsp * purpose with or without fee is hereby granted, provided that the above
7 5c860e29 2018-03-12 stsp * copyright notice and this permission notice appear in all copies.
8 5c860e29 2018-03-12 stsp *
9 5c860e29 2018-03-12 stsp * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 5c860e29 2018-03-12 stsp * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 5c860e29 2018-03-12 stsp * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 5c860e29 2018-03-12 stsp * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 5c860e29 2018-03-12 stsp * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 5c860e29 2018-03-12 stsp * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 5c860e29 2018-03-12 stsp * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16 5c860e29 2018-03-12 stsp */
17 5c860e29 2018-03-12 stsp
18 f42b1b34 2018-03-12 stsp #include <sys/queue.h>
19 64a96a6d 2018-04-01 stsp #include <sys/limits.h>
20 c0768b0f 2018-06-10 stsp #include <sys/types.h>
21 5de5890b 2018-10-18 stsp #include <sys/stat.h>
22 f42b1b34 2018-03-12 stsp
23 5c860e29 2018-03-12 stsp #include <err.h>
24 5c860e29 2018-03-12 stsp #include <errno.h>
25 5c860e29 2018-03-12 stsp #include <locale.h>
26 99437157 2018-11-11 stsp #include <signal.h>
27 5c860e29 2018-03-12 stsp #include <stdio.h>
28 5c860e29 2018-03-12 stsp #include <stdlib.h>
29 5c860e29 2018-03-12 stsp #include <string.h>
30 5c860e29 2018-03-12 stsp #include <unistd.h>
31 c09a553d 2018-03-12 stsp #include <libgen.h>
32 c0768b0f 2018-06-10 stsp #include <time.h>
33 5c860e29 2018-03-12 stsp
34 f42b1b34 2018-03-12 stsp #include "got_error.h"
35 f42b1b34 2018-03-12 stsp #include "got_object.h"
36 5261c201 2018-04-01 stsp #include "got_reference.h"
37 f42b1b34 2018-03-12 stsp #include "got_repository.h"
38 c09a553d 2018-03-12 stsp #include "got_worktree.h"
39 79109fed 2018-03-27 stsp #include "got_diff.h"
40 372ccdbb 2018-06-10 stsp #include "got_commit_graph.h"
41 404c43c4 2018-06-21 stsp #include "got_blame.h"
42 5c860e29 2018-03-12 stsp
43 5c860e29 2018-03-12 stsp #ifndef nitems
44 5c860e29 2018-03-12 stsp #define nitems(_a) (sizeof((_a)) / sizeof((_a)[0]))
45 5c860e29 2018-03-12 stsp #endif
46 99437157 2018-11-11 stsp
47 99437157 2018-11-11 stsp static volatile sig_atomic_t sigint_received;
48 99437157 2018-11-11 stsp static volatile sig_atomic_t sigpipe_received;
49 99437157 2018-11-11 stsp
50 99437157 2018-11-11 stsp static void
51 99437157 2018-11-11 stsp catch_sigint(int signo)
52 99437157 2018-11-11 stsp {
53 99437157 2018-11-11 stsp sigint_received = 1;
54 99437157 2018-11-11 stsp }
55 99437157 2018-11-11 stsp
56 99437157 2018-11-11 stsp static void
57 99437157 2018-11-11 stsp catch_sigpipe(int signo)
58 99437157 2018-11-11 stsp {
59 99437157 2018-11-11 stsp sigpipe_received = 1;
60 99437157 2018-11-11 stsp }
61 5c860e29 2018-03-12 stsp
62 99437157 2018-11-11 stsp
63 5c860e29 2018-03-12 stsp struct cmd {
64 5c860e29 2018-03-12 stsp const char *cmd_name;
65 d7d4f210 2018-03-12 stsp const struct got_error *(*cmd_main)(int, char *[]);
66 1b6b95a8 2018-03-12 stsp void (*cmd_usage)(void);
67 46a0db7d 2018-03-12 stsp const char *cmd_descr;
68 5c860e29 2018-03-12 stsp };
69 5c860e29 2018-03-12 stsp
70 4ed7e80c 2018-05-20 stsp __dead static void usage(void);
71 4ed7e80c 2018-05-20 stsp __dead static void usage_checkout(void);
72 4ed7e80c 2018-05-20 stsp __dead static void usage_log(void);
73 4ed7e80c 2018-05-20 stsp __dead static void usage_diff(void);
74 404c43c4 2018-06-21 stsp __dead static void usage_blame(void);
75 5de5890b 2018-10-18 stsp __dead static void usage_tree(void);
76 5c860e29 2018-03-12 stsp
77 4ed7e80c 2018-05-20 stsp static const struct got_error* cmd_checkout(int, char *[]);
78 4ed7e80c 2018-05-20 stsp static const struct got_error* cmd_log(int, char *[]);
79 4ed7e80c 2018-05-20 stsp static const struct got_error* cmd_diff(int, char *[]);
80 404c43c4 2018-06-21 stsp static const struct got_error* cmd_blame(int, char *[]);
81 5de5890b 2018-10-18 stsp static const struct got_error* cmd_tree(int, char *[]);
82 4ed7e80c 2018-05-20 stsp #ifdef notyet
83 4ed7e80c 2018-05-20 stsp static const struct got_error* cmd_status(int, char *[]);
84 4ed7e80c 2018-05-20 stsp #endif
85 5c860e29 2018-03-12 stsp
86 4ed7e80c 2018-05-20 stsp static struct cmd got_commands[] = {
87 c09a553d 2018-03-12 stsp { "checkout", cmd_checkout, usage_checkout,
88 0bb8a95e 2018-03-12 stsp "check out a new work tree from a repository" },
89 1b6b95a8 2018-03-12 stsp { "log", cmd_log, usage_log,
90 1b6b95a8 2018-03-12 stsp "show repository history" },
91 b00d56cd 2018-04-01 stsp { "diff", cmd_diff, usage_diff,
92 b00d56cd 2018-04-01 stsp "compare files and directories" },
93 404c43c4 2018-06-21 stsp { "blame", cmd_blame, usage_blame,
94 404c43c4 2018-06-21 stsp " show when lines in a file were changed" },
95 5de5890b 2018-10-18 stsp { "tree", cmd_tree, usage_tree,
96 5de5890b 2018-10-18 stsp " list files and directories in repository" },
97 f42b1b34 2018-03-12 stsp #ifdef notyet
98 1b6b95a8 2018-03-12 stsp { "status", cmd_status, usage_status,
99 1b6b95a8 2018-03-12 stsp "show modification status of files" },
100 f42b1b34 2018-03-12 stsp #endif
101 5c860e29 2018-03-12 stsp };
102 5c860e29 2018-03-12 stsp
103 5c860e29 2018-03-12 stsp int
104 5c860e29 2018-03-12 stsp main(int argc, char *argv[])
105 5c860e29 2018-03-12 stsp {
106 5c860e29 2018-03-12 stsp struct cmd *cmd;
107 5c860e29 2018-03-12 stsp unsigned int i;
108 5c860e29 2018-03-12 stsp int ch;
109 1b6b95a8 2018-03-12 stsp int hflag = 0;
110 5c860e29 2018-03-12 stsp
111 5c860e29 2018-03-12 stsp setlocale(LC_ALL, "");
112 5c860e29 2018-03-12 stsp
113 1b6b95a8 2018-03-12 stsp while ((ch = getopt(argc, argv, "h")) != -1) {
114 5c860e29 2018-03-12 stsp switch (ch) {
115 1b6b95a8 2018-03-12 stsp case 'h':
116 1b6b95a8 2018-03-12 stsp hflag = 1;
117 1b6b95a8 2018-03-12 stsp break;
118 5c860e29 2018-03-12 stsp default:
119 5c860e29 2018-03-12 stsp usage();
120 5c860e29 2018-03-12 stsp /* NOTREACHED */
121 5c860e29 2018-03-12 stsp }
122 5c860e29 2018-03-12 stsp }
123 5c860e29 2018-03-12 stsp
124 5c860e29 2018-03-12 stsp argc -= optind;
125 5c860e29 2018-03-12 stsp argv += optind;
126 1e70621d 2018-03-27 stsp optind = 0;
127 5c860e29 2018-03-12 stsp
128 5c860e29 2018-03-12 stsp if (argc <= 0)
129 5c860e29 2018-03-12 stsp usage();
130 5c860e29 2018-03-12 stsp
131 99437157 2018-11-11 stsp signal(SIGINT, catch_sigint);
132 99437157 2018-11-11 stsp signal(SIGPIPE, catch_sigpipe);
133 99437157 2018-11-11 stsp
134 5c860e29 2018-03-12 stsp for (i = 0; i < nitems(got_commands); i++) {
135 d7d4f210 2018-03-12 stsp const struct got_error *error;
136 d7d4f210 2018-03-12 stsp
137 5c860e29 2018-03-12 stsp cmd = &got_commands[i];
138 5c860e29 2018-03-12 stsp
139 5c860e29 2018-03-12 stsp if (strncmp(cmd->cmd_name, argv[0], strlen(argv[0])))
140 5c860e29 2018-03-12 stsp continue;
141 5c860e29 2018-03-12 stsp
142 1b6b95a8 2018-03-12 stsp if (hflag)
143 1b6b95a8 2018-03-12 stsp got_commands[i].cmd_usage();
144 1b6b95a8 2018-03-12 stsp
145 d7d4f210 2018-03-12 stsp error = got_commands[i].cmd_main(argc, argv);
146 80d5f134 2018-11-11 stsp if (error && !(sigint_received || sigpipe_received)) {
147 d7d4f210 2018-03-12 stsp fprintf(stderr, "%s: %s\n", getprogname(), error->msg);
148 d7d4f210 2018-03-12 stsp return 1;
149 d7d4f210 2018-03-12 stsp }
150 d7d4f210 2018-03-12 stsp
151 d7d4f210 2018-03-12 stsp return 0;
152 5c860e29 2018-03-12 stsp }
153 5c860e29 2018-03-12 stsp
154 20ecf764 2018-03-12 stsp fprintf(stderr, "%s: unknown command '%s'\n", getprogname(), argv[0]);
155 5c860e29 2018-03-12 stsp return 1;
156 5c860e29 2018-03-12 stsp }
157 5c860e29 2018-03-12 stsp
158 4ed7e80c 2018-05-20 stsp __dead static void
159 5c860e29 2018-03-12 stsp usage(void)
160 5c860e29 2018-03-12 stsp {
161 46a0db7d 2018-03-12 stsp int i;
162 46a0db7d 2018-03-12 stsp
163 987e94ba 2018-03-12 stsp fprintf(stderr, "usage: %s [-h] command [arg ...]\n\n"
164 987e94ba 2018-03-12 stsp "Available commands:\n", getprogname());
165 46a0db7d 2018-03-12 stsp for (i = 0; i < nitems(got_commands); i++) {
166 46a0db7d 2018-03-12 stsp struct cmd *cmd = &got_commands[i];
167 46a0db7d 2018-03-12 stsp fprintf(stderr, " %s: %s\n", cmd->cmd_name, cmd->cmd_descr);
168 46a0db7d 2018-03-12 stsp }
169 5c860e29 2018-03-12 stsp exit(1);
170 5c860e29 2018-03-12 stsp }
171 5c860e29 2018-03-12 stsp
172 4ed7e80c 2018-05-20 stsp __dead static void
173 c09a553d 2018-03-12 stsp usage_checkout(void)
174 c09a553d 2018-03-12 stsp {
175 0bb8a95e 2018-03-12 stsp fprintf(stderr, "usage: %s checkout [-p prefix] repository-path "
176 0bb8a95e 2018-03-12 stsp "[worktree-path]\n", getprogname());
177 c09a553d 2018-03-12 stsp exit(1);
178 92a684f4 2018-03-12 stsp }
179 92a684f4 2018-03-12 stsp
180 92a684f4 2018-03-12 stsp static void
181 92a684f4 2018-03-12 stsp checkout_progress(void *arg, const char *path)
182 92a684f4 2018-03-12 stsp {
183 92a684f4 2018-03-12 stsp char *worktree_path = arg;
184 92a684f4 2018-03-12 stsp
185 92a684f4 2018-03-12 stsp while (path[0] == '/')
186 92a684f4 2018-03-12 stsp path++;
187 92a684f4 2018-03-12 stsp
188 92a684f4 2018-03-12 stsp printf("A %s/%s\n", worktree_path, path);
189 99437157 2018-11-11 stsp }
190 99437157 2018-11-11 stsp
191 99437157 2018-11-11 stsp static const struct got_error *
192 99437157 2018-11-11 stsp checkout_cancel(void *arg)
193 99437157 2018-11-11 stsp {
194 99437157 2018-11-11 stsp if (sigint_received || sigpipe_received)
195 99437157 2018-11-11 stsp return got_error(GOT_ERR_CANCELLED);
196 99437157 2018-11-11 stsp return NULL;
197 c09a553d 2018-03-12 stsp }
198 c09a553d 2018-03-12 stsp
199 4ed7e80c 2018-05-20 stsp static const struct got_error *
200 c09a553d 2018-03-12 stsp cmd_checkout(int argc, char *argv[])
201 c09a553d 2018-03-12 stsp {
202 c09a553d 2018-03-12 stsp const struct got_error *error = NULL;
203 c09a553d 2018-03-12 stsp struct got_repository *repo = NULL;
204 c09a553d 2018-03-12 stsp struct got_reference *head_ref = NULL;
205 c09a553d 2018-03-12 stsp struct got_worktree *worktree = NULL;
206 c09a553d 2018-03-12 stsp char *repo_path = NULL;
207 c09a553d 2018-03-12 stsp char *worktree_path = NULL;
208 0bb8a95e 2018-03-12 stsp const char *path_prefix = "";
209 0bb8a95e 2018-03-12 stsp int ch;
210 c09a553d 2018-03-12 stsp
211 0bb8a95e 2018-03-12 stsp while ((ch = getopt(argc, argv, "p:")) != -1) {
212 0bb8a95e 2018-03-12 stsp switch (ch) {
213 0bb8a95e 2018-03-12 stsp case 'p':
214 0bb8a95e 2018-03-12 stsp path_prefix = optarg;
215 0bb8a95e 2018-03-12 stsp break;
216 0bb8a95e 2018-03-12 stsp default:
217 0bb8a95e 2018-03-12 stsp usage();
218 0bb8a95e 2018-03-12 stsp /* NOTREACHED */
219 0bb8a95e 2018-03-12 stsp }
220 0bb8a95e 2018-03-12 stsp }
221 0bb8a95e 2018-03-12 stsp
222 0bb8a95e 2018-03-12 stsp argc -= optind;
223 0bb8a95e 2018-03-12 stsp argv += optind;
224 0bb8a95e 2018-03-12 stsp
225 6715a751 2018-03-16 stsp #ifndef PROFILE
226 ad242220 2018-09-08 stsp if (pledge("stdio rpath wpath cpath flock proc exec sendfd", NULL)
227 ad242220 2018-09-08 stsp == -1)
228 c09a553d 2018-03-12 stsp err(1, "pledge");
229 6715a751 2018-03-16 stsp #endif
230 0bb8a95e 2018-03-12 stsp if (argc == 1) {
231 c09a553d 2018-03-12 stsp char *cwd, *base, *dotgit;
232 76089277 2018-04-01 stsp repo_path = realpath(argv[0], NULL);
233 76089277 2018-04-01 stsp if (repo_path == NULL)
234 76089277 2018-04-01 stsp return got_error_from_errno();
235 c09a553d 2018-03-12 stsp cwd = getcwd(NULL, 0);
236 76089277 2018-04-01 stsp if (cwd == NULL) {
237 76089277 2018-04-01 stsp error = got_error_from_errno();
238 76089277 2018-04-01 stsp goto done;
239 76089277 2018-04-01 stsp }
240 5d7c1dab 2018-04-01 stsp if (path_prefix[0])
241 5d7c1dab 2018-04-01 stsp base = basename(path_prefix);
242 5d7c1dab 2018-04-01 stsp else
243 5d7c1dab 2018-04-01 stsp base = basename(repo_path);
244 76089277 2018-04-01 stsp if (base == NULL) {
245 76089277 2018-04-01 stsp error = got_error_from_errno();
246 76089277 2018-04-01 stsp goto done;
247 76089277 2018-04-01 stsp }
248 c09a553d 2018-03-12 stsp dotgit = strstr(base, ".git");
249 c09a553d 2018-03-12 stsp if (dotgit)
250 c09a553d 2018-03-12 stsp *dotgit = '\0';
251 c09a553d 2018-03-12 stsp if (asprintf(&worktree_path, "%s/%s", cwd, base) == -1) {
252 0a585a0d 2018-03-17 stsp error = got_error_from_errno();
253 c09a553d 2018-03-12 stsp free(cwd);
254 76089277 2018-04-01 stsp goto done;
255 c09a553d 2018-03-12 stsp }
256 c09a553d 2018-03-12 stsp free(cwd);
257 0bb8a95e 2018-03-12 stsp } else if (argc == 2) {
258 76089277 2018-04-01 stsp repo_path = realpath(argv[0], NULL);
259 76089277 2018-04-01 stsp if (repo_path == NULL) {
260 76089277 2018-04-01 stsp error = got_error_from_errno();
261 76089277 2018-04-01 stsp goto done;
262 76089277 2018-04-01 stsp }
263 f7b38925 2018-04-01 stsp worktree_path = realpath(argv[1], NULL);
264 76089277 2018-04-01 stsp if (worktree_path == NULL) {
265 76089277 2018-04-01 stsp error = got_error_from_errno();
266 76089277 2018-04-01 stsp goto done;
267 76089277 2018-04-01 stsp }
268 c09a553d 2018-03-12 stsp } else
269 c09a553d 2018-03-12 stsp usage_checkout();
270 c09a553d 2018-03-12 stsp
271 c09a553d 2018-03-12 stsp error = got_repo_open(&repo, repo_path);
272 c09a553d 2018-03-12 stsp if (error != NULL)
273 c09a553d 2018-03-12 stsp goto done;
274 c09a553d 2018-03-12 stsp error = got_ref_open(&head_ref, repo, GOT_REF_HEAD);
275 c09a553d 2018-03-12 stsp if (error != NULL)
276 c09a553d 2018-03-12 stsp goto done;
277 c09a553d 2018-03-12 stsp
278 0bb8a95e 2018-03-12 stsp error = got_worktree_init(worktree_path, head_ref, path_prefix, repo);
279 c09a553d 2018-03-12 stsp if (error != NULL)
280 c09a553d 2018-03-12 stsp goto done;
281 c09a553d 2018-03-12 stsp
282 c09a553d 2018-03-12 stsp error = got_worktree_open(&worktree, worktree_path);
283 c09a553d 2018-03-12 stsp if (error != NULL)
284 c09a553d 2018-03-12 stsp goto done;
285 c09a553d 2018-03-12 stsp
286 92a684f4 2018-03-12 stsp error = got_worktree_checkout_files(worktree, head_ref, repo,
287 99437157 2018-11-11 stsp checkout_progress, worktree_path, checkout_cancel, NULL);
288 c09a553d 2018-03-12 stsp if (error != NULL)
289 c09a553d 2018-03-12 stsp goto done;
290 c09a553d 2018-03-12 stsp
291 b65ae19a 2018-04-24 stsp printf("Now shut up and hack\n");
292 c09a553d 2018-03-12 stsp
293 c09a553d 2018-03-12 stsp done:
294 76089277 2018-04-01 stsp free(repo_path);
295 c09a553d 2018-03-12 stsp free(worktree_path);
296 c09a553d 2018-03-12 stsp return error;
297 c09a553d 2018-03-12 stsp }
298 c09a553d 2018-03-12 stsp
299 f42b1b34 2018-03-12 stsp static const struct got_error *
300 79109fed 2018-03-27 stsp print_patch(struct got_commit_object *commit, struct got_object_id *id,
301 c0cc5c62 2018-10-18 stsp int diff_context, struct got_repository *repo)
302 5c860e29 2018-03-12 stsp {
303 f42b1b34 2018-03-12 stsp const struct got_error *err = NULL;
304 79109fed 2018-03-27 stsp struct got_tree_object *tree1 = NULL, *tree2;
305 79f35eb3 2018-06-11 stsp struct got_object_qid *qid;
306 0f2b3dca 2018-12-22 stsp char *id_str1 = NULL, *id_str2;
307 adacb96f 2018-12-24 stsp time_t time1 = 0, time2;
308 79109fed 2018-03-27 stsp
309 adacb96f 2018-12-24 stsp time2 = got_object_commit_get_committer_time(commit);
310 adacb96f 2018-12-24 stsp
311 45d799e2 2018-12-23 stsp err = got_object_open_as_tree(&tree2, repo,
312 45d799e2 2018-12-23 stsp got_object_commit_get_tree_id(commit));
313 79109fed 2018-03-27 stsp if (err)
314 79109fed 2018-03-27 stsp return err;
315 79109fed 2018-03-27 stsp
316 45d799e2 2018-12-23 stsp qid = SIMPLEQ_FIRST(got_object_commit_get_parent_ids(commit));
317 79f35eb3 2018-06-11 stsp if (qid != NULL) {
318 79109fed 2018-03-27 stsp struct got_commit_object *pcommit;
319 79109fed 2018-03-27 stsp
320 117e771c 2018-07-23 stsp err = got_object_open_as_commit(&pcommit, repo, qid->id);
321 79109fed 2018-03-27 stsp if (err)
322 79109fed 2018-03-27 stsp return err;
323 79109fed 2018-03-27 stsp
324 adacb96f 2018-12-24 stsp time1 = got_object_commit_get_committer_time(pcommit);
325 45d799e2 2018-12-23 stsp err = got_object_open_as_tree(&tree1, repo,
326 45d799e2 2018-12-23 stsp got_object_commit_get_tree_id(pcommit));
327 79109fed 2018-03-27 stsp got_object_commit_close(pcommit);
328 0f2b3dca 2018-12-22 stsp if (err)
329 0f2b3dca 2018-12-22 stsp return err;
330 0f2b3dca 2018-12-22 stsp
331 0f2b3dca 2018-12-22 stsp err = got_object_id_str(&id_str1, qid->id);
332 79109fed 2018-03-27 stsp if (err)
333 79109fed 2018-03-27 stsp return err;
334 79109fed 2018-03-27 stsp }
335 79109fed 2018-03-27 stsp
336 0f2b3dca 2018-12-22 stsp err = got_object_id_str(&id_str2, id);
337 0f2b3dca 2018-12-22 stsp if (err)
338 0f2b3dca 2018-12-22 stsp goto done;
339 0f2b3dca 2018-12-22 stsp
340 56765ebb 2018-12-23 stsp printf("diff %s %s\n", id_str1 ? id_str1 : "/dev/null", id_str2);
341 adacb96f 2018-12-24 stsp err = got_diff_tree(tree1, tree2, "", "", time1, time2, diff_context,
342 adacb96f 2018-12-24 stsp repo, stdout);
343 0f2b3dca 2018-12-22 stsp done:
344 79109fed 2018-03-27 stsp if (tree1)
345 79109fed 2018-03-27 stsp got_object_tree_close(tree1);
346 79109fed 2018-03-27 stsp got_object_tree_close(tree2);
347 0f2b3dca 2018-12-22 stsp free(id_str1);
348 0f2b3dca 2018-12-22 stsp free(id_str2);
349 79109fed 2018-03-27 stsp return err;
350 79109fed 2018-03-27 stsp }
351 79109fed 2018-03-27 stsp
352 4bb494d5 2018-06-16 stsp static char *
353 6c281f94 2018-06-11 stsp get_datestr(time_t *time, char *datebuf)
354 6c281f94 2018-06-11 stsp {
355 6c281f94 2018-06-11 stsp char *p, *s = ctime_r(time, datebuf);
356 6c281f94 2018-06-11 stsp p = strchr(s, '\n');
357 6c281f94 2018-06-11 stsp if (p)
358 6c281f94 2018-06-11 stsp *p = '\0';
359 6c281f94 2018-06-11 stsp return s;
360 6c281f94 2018-06-11 stsp }
361 6c281f94 2018-06-11 stsp
362 79109fed 2018-03-27 stsp static const struct got_error *
363 79109fed 2018-03-27 stsp print_commit(struct got_commit_object *commit, struct got_object_id *id,
364 c0cc5c62 2018-10-18 stsp struct got_repository *repo, int show_patch, int diff_context)
365 79109fed 2018-03-27 stsp {
366 79109fed 2018-03-27 stsp const struct got_error *err = NULL;
367 621015ac 2018-07-23 stsp char *id_str, *datestr, *logmsg0, *logmsg, *line;
368 6c281f94 2018-06-11 stsp char datebuf[26];
369 45d799e2 2018-12-23 stsp time_t committer_time;
370 45d799e2 2018-12-23 stsp const char *author, *committer;
371 5c860e29 2018-03-12 stsp
372 832c249c 2018-06-10 stsp err = got_object_id_str(&id_str, id);
373 8bf5b3c9 2018-03-17 stsp if (err)
374 8bf5b3c9 2018-03-17 stsp return err;
375 788c352e 2018-06-16 stsp
376 33d869be 2018-12-22 stsp printf("-----------------------------------------------\n");
377 832c249c 2018-06-10 stsp printf("commit %s\n", id_str);
378 832c249c 2018-06-10 stsp free(id_str);
379 45d799e2 2018-12-23 stsp printf("from: %s\n", got_object_commit_get_author(commit));
380 45d799e2 2018-12-23 stsp committer_time = got_object_commit_get_committer_time(commit);
381 45d799e2 2018-12-23 stsp datestr = get_datestr(&committer_time, datebuf);
382 dab5fe87 2018-09-14 stsp printf("date: %s UTC\n", datestr);
383 45d799e2 2018-12-23 stsp author = got_object_commit_get_author(commit);
384 45d799e2 2018-12-23 stsp committer = got_object_commit_get_committer(commit);
385 45d799e2 2018-12-23 stsp if (strcmp(author, committer) != 0)
386 45d799e2 2018-12-23 stsp printf("via: %s\n", committer);
387 45d799e2 2018-12-23 stsp if (got_object_commit_get_nparents(commit) > 1) {
388 45d799e2 2018-12-23 stsp const struct got_object_id_queue *parent_ids;
389 79f35eb3 2018-06-11 stsp struct got_object_qid *qid;
390 3fe1abad 2018-06-10 stsp int n = 1;
391 45d799e2 2018-12-23 stsp parent_ids = got_object_commit_get_parent_ids(commit);
392 45d799e2 2018-12-23 stsp SIMPLEQ_FOREACH(qid, parent_ids, entry) {
393 79f35eb3 2018-06-11 stsp err = got_object_id_str(&id_str, qid->id);
394 a0603db2 2018-06-10 stsp if (err)
395 a0603db2 2018-06-10 stsp return err;
396 3fe1abad 2018-06-10 stsp printf("parent %d: %s\n", n++, id_str);
397 a0603db2 2018-06-10 stsp free(id_str);
398 a0603db2 2018-06-10 stsp }
399 a0603db2 2018-06-10 stsp }
400 832c249c 2018-06-10 stsp
401 45d799e2 2018-12-23 stsp logmsg0 = strdup(got_object_commit_get_logmsg(commit));
402 621015ac 2018-07-23 stsp if (logmsg0 == NULL)
403 832c249c 2018-06-10 stsp return got_error_from_errno();
404 8bf5b3c9 2018-03-17 stsp
405 621015ac 2018-07-23 stsp logmsg = logmsg0;
406 832c249c 2018-06-10 stsp do {
407 832c249c 2018-06-10 stsp line = strsep(&logmsg, "\n");
408 832c249c 2018-06-10 stsp if (line)
409 832c249c 2018-06-10 stsp printf(" %s\n", line);
410 832c249c 2018-06-10 stsp } while (line);
411 621015ac 2018-07-23 stsp free(logmsg0);
412 832c249c 2018-06-10 stsp
413 971751ac 2018-03-27 stsp if (show_patch) {
414 c0cc5c62 2018-10-18 stsp err = print_patch(commit, id, diff_context, repo);
415 971751ac 2018-03-27 stsp if (err == 0)
416 971751ac 2018-03-27 stsp printf("\n");
417 971751ac 2018-03-27 stsp }
418 07862c20 2018-09-15 stsp
419 d82de447 2018-09-15 stsp fflush(stdout);
420 07862c20 2018-09-15 stsp return err;
421 f42b1b34 2018-03-12 stsp }
422 5c860e29 2018-03-12 stsp
423 f42b1b34 2018-03-12 stsp static const struct got_error *
424 15a94983 2018-12-23 stsp print_commits(struct got_object_id *root_id, struct got_repository *repo,
425 15a94983 2018-12-23 stsp char *path, int show_patch, int diff_context, int limit,
426 15a94983 2018-12-23 stsp int first_parent_traversal)
427 f42b1b34 2018-03-12 stsp {
428 f42b1b34 2018-03-12 stsp const struct got_error *err;
429 372ccdbb 2018-06-10 stsp struct got_commit_graph *graph;
430 372ccdbb 2018-06-10 stsp
431 31cedeaf 2018-09-15 stsp err = got_commit_graph_open(&graph, root_id, path,
432 31cedeaf 2018-09-15 stsp first_parent_traversal, repo);
433 f42b1b34 2018-03-12 stsp if (err)
434 f42b1b34 2018-03-12 stsp return err;
435 31cedeaf 2018-09-15 stsp err = got_commit_graph_iter_start(graph, root_id, repo);
436 372ccdbb 2018-06-10 stsp if (err)
437 fcc85cad 2018-07-22 stsp goto done;
438 31cedeaf 2018-09-15 stsp while (1) {
439 372ccdbb 2018-06-10 stsp struct got_commit_object *commit;
440 372ccdbb 2018-06-10 stsp struct got_object_id *id;
441 8bf5b3c9 2018-03-17 stsp
442 84453469 2018-11-11 stsp if (sigint_received || sigpipe_received)
443 84453469 2018-11-11 stsp break;
444 84453469 2018-11-11 stsp
445 b43fbaa0 2018-06-11 stsp err = got_commit_graph_iter_next(&id, graph);
446 372ccdbb 2018-06-10 stsp if (err) {
447 9ba79e04 2018-06-11 stsp if (err->code == GOT_ERR_ITER_COMPLETED) {
448 9ba79e04 2018-06-11 stsp err = NULL;
449 9ba79e04 2018-06-11 stsp break;
450 9ba79e04 2018-06-11 stsp }
451 372ccdbb 2018-06-10 stsp if (err->code != GOT_ERR_ITER_NEED_MORE)
452 372ccdbb 2018-06-10 stsp break;
453 31cedeaf 2018-09-15 stsp err = got_commit_graph_fetch_commits(graph, 1, repo);
454 372ccdbb 2018-06-10 stsp if (err)
455 372ccdbb 2018-06-10 stsp break;
456 372ccdbb 2018-06-10 stsp else
457 372ccdbb 2018-06-10 stsp continue;
458 7e665116 2018-04-02 stsp }
459 b43fbaa0 2018-06-11 stsp if (id == NULL)
460 7e665116 2018-04-02 stsp break;
461 b43fbaa0 2018-06-11 stsp
462 f8e900f3 2018-06-11 stsp err = got_object_open_as_commit(&commit, repo, id);
463 b43fbaa0 2018-06-11 stsp if (err)
464 fcc85cad 2018-07-22 stsp break;
465 c0cc5c62 2018-10-18 stsp err = print_commit(commit, id, repo, show_patch, diff_context);
466 b43fbaa0 2018-06-11 stsp got_object_commit_close(commit);
467 372ccdbb 2018-06-10 stsp if (err || (limit && --limit == 0))
468 7e665116 2018-04-02 stsp break;
469 31cedeaf 2018-09-15 stsp }
470 fcc85cad 2018-07-22 stsp done:
471 372ccdbb 2018-06-10 stsp got_commit_graph_close(graph);
472 f42b1b34 2018-03-12 stsp return err;
473 f42b1b34 2018-03-12 stsp }
474 5c860e29 2018-03-12 stsp
475 4ed7e80c 2018-05-20 stsp __dead static void
476 6f3d1eb0 2018-03-12 stsp usage_log(void)
477 6f3d1eb0 2018-03-12 stsp {
478 c0cc5c62 2018-10-18 stsp fprintf(stderr, "usage: %s log [-c commit] [-C number] [-f] [ -l N ] [-p] "
479 04ca23f4 2018-07-16 stsp "[-r repository-path] [path]\n", getprogname());
480 6f3d1eb0 2018-03-12 stsp exit(1);
481 6f3d1eb0 2018-03-12 stsp }
482 6f3d1eb0 2018-03-12 stsp
483 4ed7e80c 2018-05-20 stsp static const struct got_error *
484 f42b1b34 2018-03-12 stsp cmd_log(int argc, char *argv[])
485 f42b1b34 2018-03-12 stsp {
486 f42b1b34 2018-03-12 stsp const struct got_error *error;
487 04ca23f4 2018-07-16 stsp struct got_repository *repo = NULL;
488 15a94983 2018-12-23 stsp struct got_commit_object *commit = NULL;
489 3235492e 2018-04-01 stsp struct got_object_id *id = NULL;
490 04ca23f4 2018-07-16 stsp char *repo_path = NULL, *path = NULL, *cwd = NULL, *in_repo_path = NULL;
491 d142fc45 2018-04-01 stsp char *start_commit = NULL;
492 c0cc5c62 2018-10-18 stsp int diff_context = 3, ch;
493 1fd6d7ea 2018-06-13 stsp int show_patch = 0, limit = 0, first_parent_traversal = 0;
494 64a96a6d 2018-04-01 stsp const char *errstr;
495 5c860e29 2018-03-12 stsp
496 6715a751 2018-03-16 stsp #ifndef PROFILE
497 ad242220 2018-09-08 stsp if (pledge("stdio rpath wpath cpath flock proc exec sendfd", NULL)
498 ad242220 2018-09-08 stsp == -1)
499 f42b1b34 2018-03-12 stsp err(1, "pledge");
500 6715a751 2018-03-16 stsp #endif
501 79109fed 2018-03-27 stsp
502 c0cc5c62 2018-10-18 stsp while ((ch = getopt(argc, argv, "pc:C:l:fr:")) != -1) {
503 79109fed 2018-03-27 stsp switch (ch) {
504 79109fed 2018-03-27 stsp case 'p':
505 79109fed 2018-03-27 stsp show_patch = 1;
506 d142fc45 2018-04-01 stsp break;
507 d142fc45 2018-04-01 stsp case 'c':
508 d142fc45 2018-04-01 stsp start_commit = optarg;
509 64a96a6d 2018-04-01 stsp break;
510 c0cc5c62 2018-10-18 stsp case 'C':
511 4a8520aa 2018-10-18 stsp diff_context = strtonum(optarg, 0, GOT_DIFF_MAX_CONTEXT,
512 4a8520aa 2018-10-18 stsp &errstr);
513 c0cc5c62 2018-10-18 stsp if (errstr != NULL)
514 c0cc5c62 2018-10-18 stsp err(1, "-C option %s", errstr);
515 c0cc5c62 2018-10-18 stsp break;
516 64a96a6d 2018-04-01 stsp case 'l':
517 64a96a6d 2018-04-01 stsp limit = strtonum(optarg, 1, INT_MAX, &errstr);
518 64a96a6d 2018-04-01 stsp if (errstr != NULL)
519 64a96a6d 2018-04-01 stsp err(1, "-l option %s", errstr);
520 79109fed 2018-03-27 stsp break;
521 0ed6ed4c 2018-06-13 stsp case 'f':
522 0ed6ed4c 2018-06-13 stsp first_parent_traversal = 1;
523 a0603db2 2018-06-10 stsp break;
524 04ca23f4 2018-07-16 stsp case 'r':
525 04ca23f4 2018-07-16 stsp repo_path = realpath(optarg, NULL);
526 04ca23f4 2018-07-16 stsp if (repo_path == NULL)
527 04ca23f4 2018-07-16 stsp err(1, "-r option");
528 04ca23f4 2018-07-16 stsp break;
529 79109fed 2018-03-27 stsp default:
530 79109fed 2018-03-27 stsp usage();
531 79109fed 2018-03-27 stsp /* NOTREACHED */
532 79109fed 2018-03-27 stsp }
533 79109fed 2018-03-27 stsp }
534 79109fed 2018-03-27 stsp
535 79109fed 2018-03-27 stsp argc -= optind;
536 79109fed 2018-03-27 stsp argv += optind;
537 79109fed 2018-03-27 stsp
538 04ca23f4 2018-07-16 stsp if (argc == 0)
539 04ca23f4 2018-07-16 stsp path = strdup("");
540 04ca23f4 2018-07-16 stsp else if (argc == 1)
541 04ca23f4 2018-07-16 stsp path = strdup(argv[0]);
542 04ca23f4 2018-07-16 stsp else
543 6f3d1eb0 2018-03-12 stsp usage_log();
544 04ca23f4 2018-07-16 stsp if (path == NULL)
545 04ca23f4 2018-07-16 stsp return got_error_from_errno();
546 f42b1b34 2018-03-12 stsp
547 04ca23f4 2018-07-16 stsp cwd = getcwd(NULL, 0);
548 04ca23f4 2018-07-16 stsp if (cwd == NULL) {
549 04ca23f4 2018-07-16 stsp error = got_error_from_errno();
550 04ca23f4 2018-07-16 stsp goto done;
551 04ca23f4 2018-07-16 stsp }
552 04ca23f4 2018-07-16 stsp if (repo_path == NULL) {
553 04ca23f4 2018-07-16 stsp repo_path = strdup(cwd);
554 04ca23f4 2018-07-16 stsp if (repo_path == NULL) {
555 04ca23f4 2018-07-16 stsp error = got_error_from_errno();
556 04ca23f4 2018-07-16 stsp goto done;
557 04ca23f4 2018-07-16 stsp }
558 04ca23f4 2018-07-16 stsp }
559 04ca23f4 2018-07-16 stsp
560 f42b1b34 2018-03-12 stsp error = got_repo_open(&repo, repo_path);
561 d7d4f210 2018-03-12 stsp if (error != NULL)
562 04ca23f4 2018-07-16 stsp goto done;
563 f42b1b34 2018-03-12 stsp
564 d142fc45 2018-04-01 stsp if (start_commit == NULL) {
565 3235492e 2018-04-01 stsp struct got_reference *head_ref;
566 3235492e 2018-04-01 stsp error = got_ref_open(&head_ref, repo, GOT_REF_HEAD);
567 3235492e 2018-04-01 stsp if (error != NULL)
568 3235492e 2018-04-01 stsp return error;
569 3235492e 2018-04-01 stsp error = got_ref_resolve(&id, repo, head_ref);
570 3235492e 2018-04-01 stsp got_ref_close(head_ref);
571 3235492e 2018-04-01 stsp if (error != NULL)
572 3235492e 2018-04-01 stsp return error;
573 15a94983 2018-12-23 stsp error = got_object_open_as_commit(&commit, repo, id);
574 3235492e 2018-04-01 stsp } else {
575 d1f2edc9 2018-06-13 stsp struct got_reference *ref;
576 d1f2edc9 2018-06-13 stsp error = got_ref_open(&ref, repo, start_commit);
577 3235492e 2018-04-01 stsp if (error == NULL) {
578 d1f2edc9 2018-06-13 stsp error = got_ref_resolve(&id, repo, ref);
579 d1f2edc9 2018-06-13 stsp got_ref_close(ref);
580 d1f2edc9 2018-06-13 stsp if (error != NULL)
581 d1f2edc9 2018-06-13 stsp return error;
582 15a94983 2018-12-23 stsp error = got_object_open_as_commit(&commit, repo, id);
583 d1f2edc9 2018-06-13 stsp if (error != NULL)
584 d1f2edc9 2018-06-13 stsp return error;
585 d1f2edc9 2018-06-13 stsp }
586 15a94983 2018-12-23 stsp if (commit == NULL) {
587 15a94983 2018-12-23 stsp error = got_object_resolve_id_str(&id, repo,
588 d1f2edc9 2018-06-13 stsp start_commit);
589 d1f2edc9 2018-06-13 stsp if (error != NULL)
590 d1f2edc9 2018-06-13 stsp return error;
591 3235492e 2018-04-01 stsp }
592 3235492e 2018-04-01 stsp }
593 d7d4f210 2018-03-12 stsp if (error != NULL)
594 04ca23f4 2018-07-16 stsp goto done;
595 04ca23f4 2018-07-16 stsp
596 23721109 2018-10-22 stsp error = got_repo_map_path(&in_repo_path, repo, path, 1);
597 04ca23f4 2018-07-16 stsp if (error != NULL)
598 04ca23f4 2018-07-16 stsp goto done;
599 04ca23f4 2018-07-16 stsp if (in_repo_path) {
600 04ca23f4 2018-07-16 stsp free(path);
601 04ca23f4 2018-07-16 stsp path = in_repo_path;
602 04ca23f4 2018-07-16 stsp }
603 04ca23f4 2018-07-16 stsp
604 15a94983 2018-12-23 stsp error = print_commits(id, repo, path, show_patch,
605 c0cc5c62 2018-10-18 stsp diff_context, limit, first_parent_traversal);
606 04ca23f4 2018-07-16 stsp done:
607 04ca23f4 2018-07-16 stsp free(path);
608 04ca23f4 2018-07-16 stsp free(repo_path);
609 04ca23f4 2018-07-16 stsp free(cwd);
610 f42b1b34 2018-03-12 stsp free(id);
611 ad242220 2018-09-08 stsp if (repo) {
612 ad242220 2018-09-08 stsp const struct got_error *repo_error;
613 ad242220 2018-09-08 stsp repo_error = got_repo_close(repo);
614 ad242220 2018-09-08 stsp if (error == NULL)
615 ad242220 2018-09-08 stsp error = repo_error;
616 ad242220 2018-09-08 stsp }
617 8bf5b3c9 2018-03-17 stsp return error;
618 5c860e29 2018-03-12 stsp }
619 5c860e29 2018-03-12 stsp
620 4ed7e80c 2018-05-20 stsp __dead static void
621 3f8b7d6a 2018-04-01 stsp usage_diff(void)
622 3f8b7d6a 2018-04-01 stsp {
623 c0cc5c62 2018-10-18 stsp fprintf(stderr, "usage: %s diff [-C number] [repository-path] "
624 c0cc5c62 2018-10-18 stsp "object1 object2\n", getprogname());
625 3f8b7d6a 2018-04-01 stsp exit(1);
626 b00d56cd 2018-04-01 stsp }
627 b00d56cd 2018-04-01 stsp
628 4ed7e80c 2018-05-20 stsp static const struct got_error *
629 b00d56cd 2018-04-01 stsp cmd_diff(int argc, char *argv[])
630 b00d56cd 2018-04-01 stsp {
631 b00d56cd 2018-04-01 stsp const struct got_error *error;
632 b00d56cd 2018-04-01 stsp struct got_repository *repo = NULL;
633 b00d56cd 2018-04-01 stsp char *repo_path = NULL;
634 15a94983 2018-12-23 stsp struct got_object_id *id1 = NULL, *id2 = NULL;
635 15a94983 2018-12-23 stsp char *id_str1 = NULL, *id_str2 = NULL;
636 15a94983 2018-12-23 stsp int type1, type2;
637 c0cc5c62 2018-10-18 stsp int diff_context = 3, ch;
638 c0cc5c62 2018-10-18 stsp const char *errstr;
639 b00d56cd 2018-04-01 stsp
640 b00d56cd 2018-04-01 stsp #ifndef PROFILE
641 ad242220 2018-09-08 stsp if (pledge("stdio rpath wpath cpath flock proc exec sendfd", NULL)
642 ad242220 2018-09-08 stsp == -1)
643 b00d56cd 2018-04-01 stsp err(1, "pledge");
644 b00d56cd 2018-04-01 stsp #endif
645 b00d56cd 2018-04-01 stsp
646 c0cc5c62 2018-10-18 stsp while ((ch = getopt(argc, argv, "C:")) != -1) {
647 b00d56cd 2018-04-01 stsp switch (ch) {
648 c0cc5c62 2018-10-18 stsp case 'C':
649 c0cc5c62 2018-10-18 stsp diff_context = strtonum(optarg, 1, INT_MAX, &errstr);
650 c0cc5c62 2018-10-18 stsp if (errstr != NULL)
651 c0cc5c62 2018-10-18 stsp err(1, "-C option %s", errstr);
652 c0cc5c62 2018-10-18 stsp break;
653 b00d56cd 2018-04-01 stsp default:
654 b00d56cd 2018-04-01 stsp usage();
655 b00d56cd 2018-04-01 stsp /* NOTREACHED */
656 b00d56cd 2018-04-01 stsp }
657 b00d56cd 2018-04-01 stsp }
658 b00d56cd 2018-04-01 stsp
659 b00d56cd 2018-04-01 stsp argc -= optind;
660 b00d56cd 2018-04-01 stsp argv += optind;
661 b00d56cd 2018-04-01 stsp
662 b00d56cd 2018-04-01 stsp if (argc == 0) {
663 b00d56cd 2018-04-01 stsp usage_diff(); /* TODO show local worktree changes */
664 3f8b7d6a 2018-04-01 stsp } else if (argc == 2) {
665 3f8b7d6a 2018-04-01 stsp repo_path = getcwd(NULL, 0);
666 3f8b7d6a 2018-04-01 stsp if (repo_path == NULL)
667 e1e3f570 2018-04-01 stsp return got_error_from_errno();
668 15a94983 2018-12-23 stsp id_str1 = argv[0];
669 15a94983 2018-12-23 stsp id_str2 = argv[1];
670 b00d56cd 2018-04-01 stsp } else if (argc == 3) {
671 76089277 2018-04-01 stsp repo_path = realpath(argv[0], NULL);
672 76089277 2018-04-01 stsp if (repo_path == NULL)
673 76089277 2018-04-01 stsp return got_error_from_errno();
674 15a94983 2018-12-23 stsp id_str1 = argv[1];
675 15a94983 2018-12-23 stsp id_str2 = argv[2];
676 b00d56cd 2018-04-01 stsp } else
677 b00d56cd 2018-04-01 stsp usage_diff();
678 b00d56cd 2018-04-01 stsp
679 b00d56cd 2018-04-01 stsp error = got_repo_open(&repo, repo_path);
680 76089277 2018-04-01 stsp free(repo_path);
681 b00d56cd 2018-04-01 stsp if (error != NULL)
682 b00d56cd 2018-04-01 stsp goto done;
683 b00d56cd 2018-04-01 stsp
684 15a94983 2018-12-23 stsp error = got_object_resolve_id_str(&id1, repo, id_str1);
685 6402fb3c 2018-09-15 stsp if (error)
686 b00d56cd 2018-04-01 stsp goto done;
687 b00d56cd 2018-04-01 stsp
688 15a94983 2018-12-23 stsp error = got_object_resolve_id_str(&id2, repo, id_str2);
689 6402fb3c 2018-09-15 stsp if (error)
690 b00d56cd 2018-04-01 stsp goto done;
691 b00d56cd 2018-04-01 stsp
692 15a94983 2018-12-23 stsp error = got_object_get_type(&type1, repo, id1);
693 15a94983 2018-12-23 stsp if (error)
694 15a94983 2018-12-23 stsp goto done;
695 15a94983 2018-12-23 stsp
696 15a94983 2018-12-23 stsp error = got_object_get_type(&type2, repo, id2);
697 15a94983 2018-12-23 stsp if (error)
698 15a94983 2018-12-23 stsp goto done;
699 15a94983 2018-12-23 stsp
700 15a94983 2018-12-23 stsp if (type1 != type2) {
701 b00d56cd 2018-04-01 stsp error = got_error(GOT_ERR_OBJ_TYPE);
702 b00d56cd 2018-04-01 stsp goto done;
703 b00d56cd 2018-04-01 stsp }
704 b00d56cd 2018-04-01 stsp
705 15a94983 2018-12-23 stsp switch (type1) {
706 b00d56cd 2018-04-01 stsp case GOT_OBJ_TYPE_BLOB:
707 15a94983 2018-12-23 stsp error = got_diff_objects_as_blobs(id1, id2, NULL, NULL,
708 adacb96f 2018-12-24 stsp 0, 0, diff_context, repo, stdout);
709 b00d56cd 2018-04-01 stsp break;
710 b00d56cd 2018-04-01 stsp case GOT_OBJ_TYPE_TREE:
711 15a94983 2018-12-23 stsp error = got_diff_objects_as_trees(id1, id2, "", "",
712 adacb96f 2018-12-24 stsp 0, 0, diff_context, repo, stdout);
713 b00d56cd 2018-04-01 stsp break;
714 b00d56cd 2018-04-01 stsp case GOT_OBJ_TYPE_COMMIT:
715 15a94983 2018-12-23 stsp printf("diff %s %s\n", id_str1 ? id_str1 : "/dev/null",
716 15a94983 2018-12-23 stsp id_str2);
717 15a94983 2018-12-23 stsp error = got_diff_objects_as_commits(id1, id2, diff_context,
718 c0cc5c62 2018-10-18 stsp repo, stdout);
719 b00d56cd 2018-04-01 stsp break;
720 b00d56cd 2018-04-01 stsp default:
721 b00d56cd 2018-04-01 stsp error = got_error(GOT_ERR_OBJ_TYPE);
722 b00d56cd 2018-04-01 stsp }
723 b00d56cd 2018-04-01 stsp
724 b00d56cd 2018-04-01 stsp done:
725 15a94983 2018-12-23 stsp free(id1);
726 15a94983 2018-12-23 stsp free(id2);
727 ad242220 2018-09-08 stsp if (repo) {
728 ad242220 2018-09-08 stsp const struct got_error *repo_error;
729 ad242220 2018-09-08 stsp repo_error = got_repo_close(repo);
730 ad242220 2018-09-08 stsp if (error == NULL)
731 ad242220 2018-09-08 stsp error = repo_error;
732 ad242220 2018-09-08 stsp }
733 b00d56cd 2018-04-01 stsp return error;
734 404c43c4 2018-06-21 stsp }
735 404c43c4 2018-06-21 stsp
736 404c43c4 2018-06-21 stsp __dead static void
737 404c43c4 2018-06-21 stsp usage_blame(void)
738 404c43c4 2018-06-21 stsp {
739 66bea077 2018-08-02 stsp fprintf(stderr, "usage: %s blame [-c commit] [-r repository-path] path\n",
740 404c43c4 2018-06-21 stsp getprogname());
741 404c43c4 2018-06-21 stsp exit(1);
742 b00d56cd 2018-04-01 stsp }
743 b00d56cd 2018-04-01 stsp
744 404c43c4 2018-06-21 stsp static const struct got_error *
745 404c43c4 2018-06-21 stsp cmd_blame(int argc, char *argv[])
746 404c43c4 2018-06-21 stsp {
747 404c43c4 2018-06-21 stsp const struct got_error *error;
748 404c43c4 2018-06-21 stsp struct got_repository *repo = NULL;
749 66bea077 2018-08-02 stsp char *path, *cwd = NULL, *repo_path = NULL, *in_repo_path = NULL;
750 404c43c4 2018-06-21 stsp struct got_object_id *commit_id = NULL;
751 404c43c4 2018-06-21 stsp char *commit_id_str = NULL;
752 404c43c4 2018-06-21 stsp int ch;
753 404c43c4 2018-06-21 stsp
754 404c43c4 2018-06-21 stsp #ifndef PROFILE
755 ad242220 2018-09-08 stsp if (pledge("stdio rpath wpath cpath flock proc exec sendfd", NULL)
756 ad242220 2018-09-08 stsp == -1)
757 404c43c4 2018-06-21 stsp err(1, "pledge");
758 404c43c4 2018-06-21 stsp #endif
759 404c43c4 2018-06-21 stsp
760 66bea077 2018-08-02 stsp while ((ch = getopt(argc, argv, "c:r:")) != -1) {
761 404c43c4 2018-06-21 stsp switch (ch) {
762 404c43c4 2018-06-21 stsp case 'c':
763 404c43c4 2018-06-21 stsp commit_id_str = optarg;
764 404c43c4 2018-06-21 stsp break;
765 66bea077 2018-08-02 stsp case 'r':
766 66bea077 2018-08-02 stsp repo_path = realpath(optarg, NULL);
767 66bea077 2018-08-02 stsp if (repo_path == NULL)
768 66bea077 2018-08-02 stsp err(1, "-r option");
769 66bea077 2018-08-02 stsp break;
770 404c43c4 2018-06-21 stsp default:
771 404c43c4 2018-06-21 stsp usage();
772 404c43c4 2018-06-21 stsp /* NOTREACHED */
773 404c43c4 2018-06-21 stsp }
774 404c43c4 2018-06-21 stsp }
775 404c43c4 2018-06-21 stsp
776 404c43c4 2018-06-21 stsp argc -= optind;
777 404c43c4 2018-06-21 stsp argv += optind;
778 404c43c4 2018-06-21 stsp
779 a39318fd 2018-08-02 stsp if (argc == 1)
780 404c43c4 2018-06-21 stsp path = argv[0];
781 a39318fd 2018-08-02 stsp else
782 404c43c4 2018-06-21 stsp usage_blame();
783 404c43c4 2018-06-21 stsp
784 66bea077 2018-08-02 stsp cwd = getcwd(NULL, 0);
785 66bea077 2018-08-02 stsp if (cwd == NULL) {
786 66bea077 2018-08-02 stsp error = got_error_from_errno();
787 66bea077 2018-08-02 stsp goto done;
788 66bea077 2018-08-02 stsp }
789 66bea077 2018-08-02 stsp if (repo_path == NULL) {
790 66bea077 2018-08-02 stsp repo_path = strdup(cwd);
791 66bea077 2018-08-02 stsp if (repo_path == NULL) {
792 66bea077 2018-08-02 stsp error = got_error_from_errno();
793 66bea077 2018-08-02 stsp goto done;
794 66bea077 2018-08-02 stsp }
795 66bea077 2018-08-02 stsp }
796 66bea077 2018-08-02 stsp
797 404c43c4 2018-06-21 stsp error = got_repo_open(&repo, repo_path);
798 404c43c4 2018-06-21 stsp if (error != NULL)
799 404c43c4 2018-06-21 stsp goto done;
800 404c43c4 2018-06-21 stsp
801 23721109 2018-10-22 stsp error = got_repo_map_path(&in_repo_path, repo, path, 1);
802 66bea077 2018-08-02 stsp if (error != NULL)
803 66bea077 2018-08-02 stsp goto done;
804 66bea077 2018-08-02 stsp
805 404c43c4 2018-06-21 stsp if (commit_id_str == NULL) {
806 404c43c4 2018-06-21 stsp struct got_reference *head_ref;
807 404c43c4 2018-06-21 stsp error = got_ref_open(&head_ref, repo, GOT_REF_HEAD);
808 404c43c4 2018-06-21 stsp if (error != NULL)
809 66bea077 2018-08-02 stsp goto done;
810 404c43c4 2018-06-21 stsp error = got_ref_resolve(&commit_id, repo, head_ref);
811 404c43c4 2018-06-21 stsp got_ref_close(head_ref);
812 404c43c4 2018-06-21 stsp if (error != NULL)
813 66bea077 2018-08-02 stsp goto done;
814 404c43c4 2018-06-21 stsp } else {
815 15a94983 2018-12-23 stsp error = got_object_resolve_id_str(&commit_id, repo,
816 15a94983 2018-12-23 stsp commit_id_str);
817 404c43c4 2018-06-21 stsp if (error != NULL)
818 66bea077 2018-08-02 stsp goto done;
819 404c43c4 2018-06-21 stsp }
820 404c43c4 2018-06-21 stsp
821 66bea077 2018-08-02 stsp error = got_blame(in_repo_path, commit_id, repo, stdout);
822 404c43c4 2018-06-21 stsp done:
823 66bea077 2018-08-02 stsp free(in_repo_path);
824 66bea077 2018-08-02 stsp free(repo_path);
825 66bea077 2018-08-02 stsp free(cwd);
826 404c43c4 2018-06-21 stsp free(commit_id);
827 ad242220 2018-09-08 stsp if (repo) {
828 ad242220 2018-09-08 stsp const struct got_error *repo_error;
829 ad242220 2018-09-08 stsp repo_error = got_repo_close(repo);
830 ad242220 2018-09-08 stsp if (error == NULL)
831 ad242220 2018-09-08 stsp error = repo_error;
832 ad242220 2018-09-08 stsp }
833 404c43c4 2018-06-21 stsp return error;
834 5de5890b 2018-10-18 stsp }
835 5de5890b 2018-10-18 stsp
836 5de5890b 2018-10-18 stsp __dead static void
837 5de5890b 2018-10-18 stsp usage_tree(void)
838 5de5890b 2018-10-18 stsp {
839 5de5890b 2018-10-18 stsp fprintf(stderr, "usage: %s tree [-c commit] [-r repository-path] [-i] path\n",
840 5de5890b 2018-10-18 stsp getprogname());
841 5de5890b 2018-10-18 stsp exit(1);
842 5de5890b 2018-10-18 stsp }
843 5de5890b 2018-10-18 stsp
844 5de5890b 2018-10-18 stsp
845 5de5890b 2018-10-18 stsp static const struct got_error *
846 5de5890b 2018-10-18 stsp print_tree(const char *path, struct got_object_id *commit_id,
847 5de5890b 2018-10-18 stsp int show_ids, struct got_repository *repo)
848 5de5890b 2018-10-18 stsp {
849 5de5890b 2018-10-18 stsp const struct got_error *err = NULL;
850 5de5890b 2018-10-18 stsp struct got_object_id *tree_id = NULL;
851 5de5890b 2018-10-18 stsp struct got_tree_object *tree = NULL;
852 5de5890b 2018-10-18 stsp const struct got_tree_entries *entries;
853 5de5890b 2018-10-18 stsp struct got_tree_entry *te;
854 5de5890b 2018-10-18 stsp
855 5de5890b 2018-10-18 stsp err = got_object_id_by_path(&tree_id, repo, commit_id, path);
856 5de5890b 2018-10-18 stsp if (err)
857 5de5890b 2018-10-18 stsp goto done;
858 5de5890b 2018-10-18 stsp
859 5de5890b 2018-10-18 stsp err = got_object_open_as_tree(&tree, repo, tree_id);
860 5de5890b 2018-10-18 stsp if (err)
861 5de5890b 2018-10-18 stsp goto done;
862 5de5890b 2018-10-18 stsp entries = got_object_tree_get_entries(tree);
863 5de5890b 2018-10-18 stsp te = SIMPLEQ_FIRST(&entries->head);
864 5de5890b 2018-10-18 stsp while (te) {
865 5de5890b 2018-10-18 stsp char *id = NULL;
866 84453469 2018-11-11 stsp
867 84453469 2018-11-11 stsp if (sigint_received || sigpipe_received)
868 84453469 2018-11-11 stsp break;
869 84453469 2018-11-11 stsp
870 5de5890b 2018-10-18 stsp if (show_ids) {
871 5de5890b 2018-10-18 stsp char *id_str;
872 5de5890b 2018-10-18 stsp err = got_object_id_str(&id_str, te->id);
873 5de5890b 2018-10-18 stsp if (err)
874 5de5890b 2018-10-18 stsp goto done;
875 5de5890b 2018-10-18 stsp if (asprintf(&id, "%s ", id_str) == -1) {
876 5de5890b 2018-10-18 stsp err = got_error_from_errno();
877 5de5890b 2018-10-18 stsp free(id_str);
878 5de5890b 2018-10-18 stsp goto done;
879 5de5890b 2018-10-18 stsp }
880 5de5890b 2018-10-18 stsp free(id_str);
881 5de5890b 2018-10-18 stsp }
882 5de5890b 2018-10-18 stsp printf("%s%s%s\n", id ? id : "",
883 5de5890b 2018-10-18 stsp te->name, S_ISDIR(te->mode) ? "/" : "");
884 5de5890b 2018-10-18 stsp te = SIMPLEQ_NEXT(te, entry);
885 5de5890b 2018-10-18 stsp free(id);
886 5de5890b 2018-10-18 stsp }
887 5de5890b 2018-10-18 stsp done:
888 5de5890b 2018-10-18 stsp if (tree)
889 5de5890b 2018-10-18 stsp got_object_tree_close(tree);
890 5de5890b 2018-10-18 stsp free(tree_id);
891 5de5890b 2018-10-18 stsp return err;
892 404c43c4 2018-06-21 stsp }
893 404c43c4 2018-06-21 stsp
894 5de5890b 2018-10-18 stsp static const struct got_error *
895 5de5890b 2018-10-18 stsp cmd_tree(int argc, char *argv[])
896 5de5890b 2018-10-18 stsp {
897 5de5890b 2018-10-18 stsp const struct got_error *error;
898 5de5890b 2018-10-18 stsp struct got_repository *repo = NULL;
899 5de5890b 2018-10-18 stsp char *path, *cwd = NULL, *repo_path = NULL, *in_repo_path = NULL;
900 5de5890b 2018-10-18 stsp struct got_object_id *commit_id = NULL;
901 5de5890b 2018-10-18 stsp char *commit_id_str = NULL;
902 5de5890b 2018-10-18 stsp int show_ids = 0;
903 5de5890b 2018-10-18 stsp int ch;
904 5de5890b 2018-10-18 stsp
905 5de5890b 2018-10-18 stsp #ifndef PROFILE
906 5de5890b 2018-10-18 stsp if (pledge("stdio rpath wpath cpath flock proc exec sendfd", NULL)
907 5de5890b 2018-10-18 stsp == -1)
908 5de5890b 2018-10-18 stsp err(1, "pledge");
909 5de5890b 2018-10-18 stsp #endif
910 5de5890b 2018-10-18 stsp
911 5de5890b 2018-10-18 stsp while ((ch = getopt(argc, argv, "c:r:i")) != -1) {
912 5de5890b 2018-10-18 stsp switch (ch) {
913 5de5890b 2018-10-18 stsp case 'c':
914 5de5890b 2018-10-18 stsp commit_id_str = optarg;
915 5de5890b 2018-10-18 stsp break;
916 5de5890b 2018-10-18 stsp case 'r':
917 5de5890b 2018-10-18 stsp repo_path = realpath(optarg, NULL);
918 5de5890b 2018-10-18 stsp if (repo_path == NULL)
919 5de5890b 2018-10-18 stsp err(1, "-r option");
920 5de5890b 2018-10-18 stsp break;
921 5de5890b 2018-10-18 stsp case 'i':
922 5de5890b 2018-10-18 stsp show_ids = 1;
923 5de5890b 2018-10-18 stsp break;
924 5de5890b 2018-10-18 stsp default:
925 5de5890b 2018-10-18 stsp usage();
926 5de5890b 2018-10-18 stsp /* NOTREACHED */
927 5de5890b 2018-10-18 stsp }
928 5de5890b 2018-10-18 stsp }
929 5de5890b 2018-10-18 stsp
930 5de5890b 2018-10-18 stsp argc -= optind;
931 5de5890b 2018-10-18 stsp argv += optind;
932 5de5890b 2018-10-18 stsp
933 5de5890b 2018-10-18 stsp if (argc == 1)
934 5de5890b 2018-10-18 stsp path = argv[0];
935 5de5890b 2018-10-18 stsp else if (argc > 1)
936 5de5890b 2018-10-18 stsp usage_tree();
937 5de5890b 2018-10-18 stsp else
938 5de5890b 2018-10-18 stsp path = "/";
939 5de5890b 2018-10-18 stsp
940 5de5890b 2018-10-18 stsp cwd = getcwd(NULL, 0);
941 5de5890b 2018-10-18 stsp if (cwd == NULL) {
942 5de5890b 2018-10-18 stsp error = got_error_from_errno();
943 5de5890b 2018-10-18 stsp goto done;
944 5de5890b 2018-10-18 stsp }
945 5de5890b 2018-10-18 stsp if (repo_path == NULL) {
946 5de5890b 2018-10-18 stsp repo_path = strdup(cwd);
947 5de5890b 2018-10-18 stsp if (repo_path == NULL) {
948 5de5890b 2018-10-18 stsp error = got_error_from_errno();
949 5de5890b 2018-10-18 stsp goto done;
950 5de5890b 2018-10-18 stsp }
951 5de5890b 2018-10-18 stsp }
952 5de5890b 2018-10-18 stsp
953 5de5890b 2018-10-18 stsp error = got_repo_open(&repo, repo_path);
954 5de5890b 2018-10-18 stsp if (error != NULL)
955 5de5890b 2018-10-18 stsp goto done;
956 5de5890b 2018-10-18 stsp
957 23721109 2018-10-22 stsp error = got_repo_map_path(&in_repo_path, repo, path, 1);
958 5de5890b 2018-10-18 stsp if (error != NULL)
959 5de5890b 2018-10-18 stsp goto done;
960 5de5890b 2018-10-18 stsp
961 5de5890b 2018-10-18 stsp if (commit_id_str == NULL) {
962 5de5890b 2018-10-18 stsp struct got_reference *head_ref;
963 5de5890b 2018-10-18 stsp error = got_ref_open(&head_ref, repo, GOT_REF_HEAD);
964 5de5890b 2018-10-18 stsp if (error != NULL)
965 5de5890b 2018-10-18 stsp goto done;
966 5de5890b 2018-10-18 stsp error = got_ref_resolve(&commit_id, repo, head_ref);
967 5de5890b 2018-10-18 stsp got_ref_close(head_ref);
968 5de5890b 2018-10-18 stsp if (error != NULL)
969 5de5890b 2018-10-18 stsp goto done;
970 5de5890b 2018-10-18 stsp } else {
971 15a94983 2018-12-23 stsp error = got_object_resolve_id_str(&commit_id, repo,
972 15a94983 2018-12-23 stsp commit_id_str);
973 5de5890b 2018-10-18 stsp if (error != NULL)
974 5de5890b 2018-10-18 stsp goto done;
975 5de5890b 2018-10-18 stsp }
976 5de5890b 2018-10-18 stsp
977 5de5890b 2018-10-18 stsp error = print_tree(in_repo_path, commit_id, show_ids, repo);
978 5de5890b 2018-10-18 stsp done:
979 5de5890b 2018-10-18 stsp free(in_repo_path);
980 5de5890b 2018-10-18 stsp free(repo_path);
981 5de5890b 2018-10-18 stsp free(cwd);
982 5de5890b 2018-10-18 stsp free(commit_id);
983 5de5890b 2018-10-18 stsp if (repo) {
984 5de5890b 2018-10-18 stsp const struct got_error *repo_error;
985 5de5890b 2018-10-18 stsp repo_error = got_repo_close(repo);
986 5de5890b 2018-10-18 stsp if (error == NULL)
987 5de5890b 2018-10-18 stsp error = repo_error;
988 5de5890b 2018-10-18 stsp }
989 5de5890b 2018-10-18 stsp return error;
990 5de5890b 2018-10-18 stsp }
991 5de5890b 2018-10-18 stsp
992 f42b1b34 2018-03-12 stsp #ifdef notyet
993 4ed7e80c 2018-05-20 stsp static const struct got_error *
994 5c860e29 2018-03-12 stsp cmd_status(int argc __unused, char *argv[] __unused)
995 5c860e29 2018-03-12 stsp {
996 5c860e29 2018-03-12 stsp git_repository *repo = NULL;
997 5c860e29 2018-03-12 stsp git_status_list *status;
998 5c860e29 2018-03-12 stsp git_status_options statusopts;
999 5c860e29 2018-03-12 stsp size_t i;
1000 5c860e29 2018-03-12 stsp
1001 5c860e29 2018-03-12 stsp git_libgit2_init();
1002 5c860e29 2018-03-12 stsp
1003 5c860e29 2018-03-12 stsp if (git_repository_open_ext(&repo, ".", 0, NULL))
1004 5c860e29 2018-03-12 stsp errx(1, "git_repository_open: %s", giterr_last()->message);
1005 5c860e29 2018-03-12 stsp
1006 5c860e29 2018-03-12 stsp if (git_repository_is_bare(repo))
1007 5c860e29 2018-03-12 stsp errx(1, "bar repository");
1008 5c860e29 2018-03-12 stsp
1009 5c860e29 2018-03-12 stsp if (git_status_init_options(&statusopts, GIT_STATUS_OPTIONS_VERSION))
1010 5c860e29 2018-03-12 stsp errx(1, "git_status_init_options: %s", giterr_last()->message);
1011 5c860e29 2018-03-12 stsp
1012 5c860e29 2018-03-12 stsp statusopts.show = GIT_STATUS_SHOW_INDEX_AND_WORKDIR;
1013 5c860e29 2018-03-12 stsp statusopts.flags = GIT_STATUS_OPT_INCLUDE_UNTRACKED |
1014 5c860e29 2018-03-12 stsp GIT_STATUS_OPT_RENAMES_HEAD_TO_INDEX |
1015 5c860e29 2018-03-12 stsp GIT_STATUS_OPT_SORT_CASE_SENSITIVELY;
1016 5c860e29 2018-03-12 stsp
1017 5c860e29 2018-03-12 stsp if (git_status_list_new(&status, repo, &statusopts))
1018 5c860e29 2018-03-12 stsp errx(1, "git_status_list_new: %s", giterr_last()->message);
1019 5c860e29 2018-03-12 stsp
1020 5c860e29 2018-03-12 stsp for (i = 0; i < git_status_list_entrycount(status); i++) {
1021 5c860e29 2018-03-12 stsp const git_status_entry *se;
1022 5c860e29 2018-03-12 stsp
1023 5c860e29 2018-03-12 stsp se = git_status_byindex(status, i);
1024 5c860e29 2018-03-12 stsp switch (se->status) {
1025 5c860e29 2018-03-12 stsp case GIT_STATUS_WT_NEW:
1026 5c860e29 2018-03-12 stsp printf("? %s\n", se->index_to_workdir->new_file.path);
1027 5c860e29 2018-03-12 stsp break;
1028 5c860e29 2018-03-12 stsp case GIT_STATUS_WT_MODIFIED:
1029 5c860e29 2018-03-12 stsp printf("M %s\n", se->index_to_workdir->new_file.path);
1030 5c860e29 2018-03-12 stsp break;
1031 5c860e29 2018-03-12 stsp case GIT_STATUS_WT_DELETED:
1032 5c860e29 2018-03-12 stsp printf("R %s\n", se->index_to_workdir->new_file.path);
1033 5c860e29 2018-03-12 stsp break;
1034 5c860e29 2018-03-12 stsp case GIT_STATUS_WT_RENAMED:
1035 5c860e29 2018-03-12 stsp printf("m %s -> %s\n",
1036 5c860e29 2018-03-12 stsp se->index_to_workdir->old_file.path,
1037 5c860e29 2018-03-12 stsp se->index_to_workdir->new_file.path);
1038 5c860e29 2018-03-12 stsp break;
1039 5c860e29 2018-03-12 stsp case GIT_STATUS_WT_TYPECHANGE:
1040 5c860e29 2018-03-12 stsp printf("t %s\n", se->index_to_workdir->new_file.path);
1041 5c860e29 2018-03-12 stsp break;
1042 5c860e29 2018-03-12 stsp case GIT_STATUS_INDEX_NEW:
1043 5c860e29 2018-03-12 stsp printf("A %s\n", se->head_to_index->new_file.path);
1044 5c860e29 2018-03-12 stsp break;
1045 5c860e29 2018-03-12 stsp case GIT_STATUS_INDEX_MODIFIED:
1046 5c860e29 2018-03-12 stsp printf("M %s\n", se->head_to_index->old_file.path);
1047 5c860e29 2018-03-12 stsp break;
1048 5c860e29 2018-03-12 stsp case GIT_STATUS_INDEX_DELETED:
1049 5c860e29 2018-03-12 stsp printf("R %s\n", se->head_to_index->old_file.path);
1050 5c860e29 2018-03-12 stsp break;
1051 5c860e29 2018-03-12 stsp case GIT_STATUS_INDEX_RENAMED:
1052 5c860e29 2018-03-12 stsp printf("m %s -> %s\n",
1053 5c860e29 2018-03-12 stsp se->head_to_index->old_file.path,
1054 5c860e29 2018-03-12 stsp se->head_to_index->new_file.path);
1055 5c860e29 2018-03-12 stsp break;
1056 5c860e29 2018-03-12 stsp case GIT_STATUS_INDEX_TYPECHANGE:
1057 5c860e29 2018-03-12 stsp printf("t %s\n", se->head_to_index->old_file.path);
1058 5c860e29 2018-03-12 stsp break;
1059 5c860e29 2018-03-12 stsp case GIT_STATUS_CURRENT:
1060 5c860e29 2018-03-12 stsp default:
1061 5c860e29 2018-03-12 stsp break;
1062 5c860e29 2018-03-12 stsp }
1063 5c860e29 2018-03-12 stsp }
1064 5c860e29 2018-03-12 stsp
1065 5c860e29 2018-03-12 stsp git_status_list_free(status);
1066 5c860e29 2018-03-12 stsp git_repository_free(repo);
1067 5c860e29 2018-03-12 stsp git_libgit2_shutdown();
1068 5c860e29 2018-03-12 stsp
1069 5c860e29 2018-03-12 stsp return 0;
1070 5c860e29 2018-03-12 stsp }
1071 f42b1b34 2018-03-12 stsp #endif