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
97
98
99
100
101
|
#include "lib.h"
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include "sys9.h"
#include "dir.h"
int
rename(const char *from, const char *to)
{
char buf[8192], *f, *t;
Dir *s, *d, nd;
int n, ffd, tfd;
f = strrchr(from, '/');
t = strrchr(to, '/');
f = f != nil ? f+1 : (char*)from;
t = t != nil ? t+1 : (char*)to;
if(*f == '\0' || strcmp(f, ".") == 0 || strcmp(f, "..") == 0
|| *t == '\0' || strcmp(t, ".") == 0 || strcmp(t, "..") == 0){
errno = EINVAL;
return -1;
}
if((s = _dirstat(from)) == nil){
_syserrno();
return -1;
}
if((d = _dirstat(to)) != nil){
if(d->qid.type == s->qid.type
&& d->qid.path == s->qid.path
&& d->qid.vers == s->qid.vers
&& d->type == s->type
&& d->dev == s->dev)
goto out; /* same file */
if((d->mode ^ s->mode) & DMDIR){
errno = (d->mode & DMDIR) ? EISDIR : ENOTDIR;
goto err;
}
}
/* from and to are in same directory (we miss some cases) */
if(f-from==t-to && strncmp(from, to, f-from)==0){
if(d != nil && _REMOVE(to) < 0){
_syserrno();
goto err;
}
_nulldir(&nd);
nd.name = t;
if(_dirwstat(from, &nd) < 0){
_syserrno();
goto err;
}
goto out;
}
/* different directories: have to copy */
if((ffd = _OPEN(from, OREAD)) < 0){
_syserrno();
goto err;
}
if((s->mode & DMDIR) != 0 && _READ(ffd, buf, sizeof(buf)) > 0){
/* cannot copy non-empty directories */
errno = EXDEV;
_CLOSE(ffd);
goto err;
}
if(d != nil && _REMOVE(to) < 0){
_syserrno();
_CLOSE(ffd);
goto err;
}
if((tfd = _CREATE(to, OWRITE, s->mode)) < 0){
_syserrno();
_CLOSE(ffd);
goto err;
}
while((n = _READ(ffd, buf, sizeof(buf))) > 0){
if(_WRITE(tfd, buf, n) != n)
break;
}
_CLOSE(ffd);
_CLOSE(tfd);
if(n != 0 || _REMOVE(from) < 0){
_syserrno();
_REMOVE(to); /* cleanup */
goto err;
}
out:
free(s);
free(d);
return 0;
err:
free(s);
free(d);
return -1;
}
|