forked from git/git
-
Notifications
You must be signed in to change notification settings - Fork 136
/
chdir-notify.c
96 lines (80 loc) · 1.88 KB
/
chdir-notify.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#include "git-compat-util.h"
#include "abspath.h"
#include "chdir-notify.h"
#include "list.h"
#include "path.h"
#include "strbuf.h"
#include "trace.h"
struct chdir_notify_entry {
const char *name;
chdir_notify_callback cb;
void *data;
struct list_head list;
};
static LIST_HEAD(chdir_notify_entries);
void chdir_notify_register(const char *name,
chdir_notify_callback cb,
void *data)
{
struct chdir_notify_entry *e = xmalloc(sizeof(*e));
e->name = name;
e->cb = cb;
e->data = data;
list_add_tail(&e->list, &chdir_notify_entries);
}
static void reparent_cb(const char *name,
const char *old_cwd,
const char *new_cwd,
void *data)
{
char **path = data;
char *tmp = *path;
if (!tmp)
return;
*path = reparent_relative_path(old_cwd, new_cwd, tmp);
free(tmp);
if (name) {
trace_printf_key(&trace_setup_key,
"setup: reparent %s to '%s'",
name, *path);
}
}
void chdir_notify_reparent(const char *name, char **path)
{
chdir_notify_register(name, reparent_cb, path);
}
int chdir_notify(const char *new_cwd)
{
struct strbuf old_cwd = STRBUF_INIT;
struct list_head *pos;
if (strbuf_getcwd(&old_cwd) < 0)
return -1;
if (chdir(new_cwd) < 0) {
int saved_errno = errno;
strbuf_release(&old_cwd);
errno = saved_errno;
return -1;
}
trace_printf_key(&trace_setup_key,
"setup: chdir from '%s' to '%s'",
old_cwd.buf, new_cwd);
list_for_each(pos, &chdir_notify_entries) {
struct chdir_notify_entry *e =
list_entry(pos, struct chdir_notify_entry, list);
e->cb(e->name, old_cwd.buf, new_cwd, e->data);
}
strbuf_release(&old_cwd);
return 0;
}
char *reparent_relative_path(const char *old_cwd,
const char *new_cwd,
const char *path)
{
char *ret, *full;
if (is_absolute_path(path))
return xstrdup(path);
full = xstrfmt("%s/%s", old_cwd, path);
ret = xstrdup(remove_leading_path(full, new_cwd));
free(full);
return ret;
}