git
/
statinfo.c
130 строк · 3.0 Кб
1#include "git-compat-util.h"
2#include "environment.h"
3#include "statinfo.h"
4
5/*
6* Munge st_size into an unsigned int.
7*/
8static unsigned int munge_st_size(off_t st_size) {
9unsigned int sd_size = st_size;
10
11/*
12* If the file is an exact multiple of 4 GiB, modify the value so it
13* doesn't get marked as racily clean (zero).
14*/
15if (!sd_size && st_size)
16return 0x80000000;
17else
18return sd_size;
19}
20
21void fill_stat_data(struct stat_data *sd, struct stat *st)
22{
23sd->sd_ctime.sec = (unsigned int)st->st_ctime;
24sd->sd_mtime.sec = (unsigned int)st->st_mtime;
25sd->sd_ctime.nsec = ST_CTIME_NSEC(*st);
26sd->sd_mtime.nsec = ST_MTIME_NSEC(*st);
27sd->sd_dev = st->st_dev;
28sd->sd_ino = st->st_ino;
29sd->sd_uid = st->st_uid;
30sd->sd_gid = st->st_gid;
31sd->sd_size = munge_st_size(st->st_size);
32}
33
34static void set_times(struct stat *st, const struct stat_data *sd)
35{
36st->st_ctime = sd->sd_ctime.sec;
37st->st_mtime = sd->sd_mtime.sec;
38#ifdef NO_NSEC
39; /* nothing */
40#else
41#ifdef USE_ST_TIMESPEC
42st->st_ctimespec.tv_nsec = sd->sd_ctime.nsec;
43st->st_mtimespec.tv_nsec = sd->sd_mtime.nsec;
44#else
45st->st_ctim.tv_nsec = sd->sd_ctime.nsec;
46st->st_mtim.tv_nsec = sd->sd_mtime.nsec;
47#endif
48#endif
49}
50
51void fake_lstat_data(const struct stat_data *sd, struct stat *st)
52{
53set_times(st, sd);
54st->st_dev = sd->sd_dev;
55st->st_ino = sd->sd_ino;
56st->st_uid = sd->sd_uid;
57st->st_gid = sd->sd_gid;
58st->st_size = sd->sd_size;
59}
60
61int match_stat_data(const struct stat_data *sd, struct stat *st)
62{
63int changed = 0;
64
65if (sd->sd_mtime.sec != (unsigned int)st->st_mtime)
66changed |= MTIME_CHANGED;
67if (trust_ctime && check_stat &&
68sd->sd_ctime.sec != (unsigned int)st->st_ctime)
69changed |= CTIME_CHANGED;
70
71#ifdef USE_NSEC
72if (check_stat && sd->sd_mtime.nsec != ST_MTIME_NSEC(*st))
73changed |= MTIME_CHANGED;
74if (trust_ctime && check_stat &&
75sd->sd_ctime.nsec != ST_CTIME_NSEC(*st))
76changed |= CTIME_CHANGED;
77#endif
78
79if (check_stat) {
80if (sd->sd_uid != (unsigned int) st->st_uid ||
81sd->sd_gid != (unsigned int) st->st_gid)
82changed |= OWNER_CHANGED;
83if (sd->sd_ino != (unsigned int) st->st_ino)
84changed |= INODE_CHANGED;
85}
86
87#ifdef USE_STDEV
88/*
89* st_dev breaks on network filesystems where different
90* clients will have different views of what "device"
91* the filesystem is on
92*/
93if (check_stat && sd->sd_dev != (unsigned int) st->st_dev)
94changed |= INODE_CHANGED;
95#endif
96
97if (sd->sd_size != munge_st_size(st->st_size))
98changed |= DATA_CHANGED;
99
100return changed;
101}
102
103void stat_validity_clear(struct stat_validity *sv)
104{
105FREE_AND_NULL(sv->sd);
106}
107
108int stat_validity_check(struct stat_validity *sv, const char *path)
109{
110struct stat st;
111
112if (stat(path, &st) < 0)
113return sv->sd == NULL;
114if (!sv->sd)
115return 0;
116return S_ISREG(st.st_mode) && !match_stat_data(sv->sd, &st);
117}
118
119void stat_validity_update(struct stat_validity *sv, int fd)
120{
121struct stat st;
122
123if (fstat(fd, &st) < 0 || !S_ISREG(st.st_mode))
124stat_validity_clear(sv);
125else {
126if (!sv->sd)
127CALLOC_ARRAY(sv->sd, 1);
128fill_stat_data(sv->sd, &st);
129}
130}
131