xref: /aosp_15_r20/external/toybox/toys/posix/ln.c (revision cf5a6c84e2b8763fc1a7db14496fd4742913b199)
1*cf5a6c84SAndroid Build Coastguard Worker /* ln.c - Create filesystem links
2*cf5a6c84SAndroid Build Coastguard Worker  *
3*cf5a6c84SAndroid Build Coastguard Worker  * Copyright 2012 Andre Renaud <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker  *
5*cf5a6c84SAndroid Build Coastguard Worker  * See http://opengroup.org/onlinepubs/9699919799/utilities/ln.html
6*cf5a6c84SAndroid Build Coastguard Worker 
7*cf5a6c84SAndroid Build Coastguard Worker USE_LN(NEWTOY(ln, "<1rt:Tvnfs", TOYFLAG_BIN))
8*cf5a6c84SAndroid Build Coastguard Worker 
9*cf5a6c84SAndroid Build Coastguard Worker config LN
10*cf5a6c84SAndroid Build Coastguard Worker   bool "ln"
11*cf5a6c84SAndroid Build Coastguard Worker   default y
12*cf5a6c84SAndroid Build Coastguard Worker   help
13*cf5a6c84SAndroid Build Coastguard Worker     usage: ln [-fnrsTv] [-t DIR] [FROM...] TO
14*cf5a6c84SAndroid Build Coastguard Worker 
15*cf5a6c84SAndroid Build Coastguard Worker     Create a link between FROM and TO.
16*cf5a6c84SAndroid Build Coastguard Worker     One/two/many arguments work like "mv" or "cp".
17*cf5a6c84SAndroid Build Coastguard Worker 
18*cf5a6c84SAndroid Build Coastguard Worker     -f	Force the creation of the link, even if TO already exists
19*cf5a6c84SAndroid Build Coastguard Worker     -n	Symlink at TO treated as file
20*cf5a6c84SAndroid Build Coastguard Worker     -r	Create relative symlink from -> to
21*cf5a6c84SAndroid Build Coastguard Worker     -s	Create a symbolic link
22*cf5a6c84SAndroid Build Coastguard Worker     -t	Create links in DIR
23*cf5a6c84SAndroid Build Coastguard Worker     -T	TO always treated as file, max 2 arguments
24*cf5a6c84SAndroid Build Coastguard Worker     -v	Verbose
25*cf5a6c84SAndroid Build Coastguard Worker */
26*cf5a6c84SAndroid Build Coastguard Worker 
27*cf5a6c84SAndroid Build Coastguard Worker #define FOR_ln
28*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
29*cf5a6c84SAndroid Build Coastguard Worker 
GLOBALS(char * t;)30*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
31*cf5a6c84SAndroid Build Coastguard Worker   char *t;
32*cf5a6c84SAndroid Build Coastguard Worker )
33*cf5a6c84SAndroid Build Coastguard Worker 
34*cf5a6c84SAndroid Build Coastguard Worker void ln_main(void)
35*cf5a6c84SAndroid Build Coastguard Worker {
36*cf5a6c84SAndroid Build Coastguard Worker   char *dest = TT.t ? TT.t : toys.optargs[--toys.optc], *new;
37*cf5a6c84SAndroid Build Coastguard Worker   struct stat buf;
38*cf5a6c84SAndroid Build Coastguard Worker   int i, rc;
39*cf5a6c84SAndroid Build Coastguard Worker 
40*cf5a6c84SAndroid Build Coastguard Worker   if (FLAG(T) && toys.optc>1) help_exit("Max 2 arguments");
41*cf5a6c84SAndroid Build Coastguard Worker 
42*cf5a6c84SAndroid Build Coastguard Worker   // With one argument, create link in current directory.
43*cf5a6c84SAndroid Build Coastguard Worker   if (!toys.optc) {
44*cf5a6c84SAndroid Build Coastguard Worker     toys.optc++;
45*cf5a6c84SAndroid Build Coastguard Worker     dest = ".";
46*cf5a6c84SAndroid Build Coastguard Worker   }
47*cf5a6c84SAndroid Build Coastguard Worker 
48*cf5a6c84SAndroid Build Coastguard Worker   // Is destination a directory?
49*cf5a6c84SAndroid Build Coastguard Worker   if (!((FLAG(n)||FLAG(T)) ? lstat : stat)(dest, &buf)) {
50*cf5a6c84SAndroid Build Coastguard Worker     if ((i = S_ISDIR(buf.st_mode)) ? FLAG(T) : (toys.optc>1 || TT.t))
51*cf5a6c84SAndroid Build Coastguard Worker       error_exit("'%s' %s a directory", dest, i ? "is" : "not");
52*cf5a6c84SAndroid Build Coastguard Worker   } else buf.st_mode = 0;
53*cf5a6c84SAndroid Build Coastguard Worker 
54*cf5a6c84SAndroid Build Coastguard Worker   for (i=0; i<toys.optc; i++) {
55*cf5a6c84SAndroid Build Coastguard Worker     char *oldnew = 0, *try = toys.optargs[i], *ss;
56*cf5a6c84SAndroid Build Coastguard Worker 
57*cf5a6c84SAndroid Build Coastguard Worker     if (S_ISDIR(buf.st_mode)) new = xmprintf("%s/%s", dest, basename(try));
58*cf5a6c84SAndroid Build Coastguard Worker     else new = dest;
59*cf5a6c84SAndroid Build Coastguard Worker 
60*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(r)) {
61*cf5a6c84SAndroid Build Coastguard Worker       ss = xstrdup(new);
62*cf5a6c84SAndroid Build Coastguard Worker       try = relative_path(dirname(ss), try, 1);
63*cf5a6c84SAndroid Build Coastguard Worker       free(ss);
64*cf5a6c84SAndroid Build Coastguard Worker       if (!try) {
65*cf5a6c84SAndroid Build Coastguard Worker         if (new != dest) free(new);
66*cf5a6c84SAndroid Build Coastguard Worker         continue;
67*cf5a6c84SAndroid Build Coastguard Worker       }
68*cf5a6c84SAndroid Build Coastguard Worker       toys.optflags |= FLAG_s;
69*cf5a6c84SAndroid Build Coastguard Worker     }
70*cf5a6c84SAndroid Build Coastguard Worker 
71*cf5a6c84SAndroid Build Coastguard Worker     // Force needs to unlink the existing target (if any). Do that by creating
72*cf5a6c84SAndroid Build Coastguard Worker     // a temp version and renaming it over the old one, so we can retain the
73*cf5a6c84SAndroid Build Coastguard Worker     // old file in cases we can't replace it (such as hardlink between mounts).
74*cf5a6c84SAndroid Build Coastguard Worker     oldnew = new;
75*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(f)) {
76*cf5a6c84SAndroid Build Coastguard Worker       new = xmprintf("%s_XXXXXX", new);
77*cf5a6c84SAndroid Build Coastguard Worker       rc = mkstemp(new);
78*cf5a6c84SAndroid Build Coastguard Worker       if (rc >= 0) {
79*cf5a6c84SAndroid Build Coastguard Worker         close(rc);
80*cf5a6c84SAndroid Build Coastguard Worker         if (unlink(new)) perror_msg("unlink '%s'", new);
81*cf5a6c84SAndroid Build Coastguard Worker       }
82*cf5a6c84SAndroid Build Coastguard Worker     }
83*cf5a6c84SAndroid Build Coastguard Worker 
84*cf5a6c84SAndroid Build Coastguard Worker     rc = FLAG(s) ? symlink(try, new) : link(try, new);
85*cf5a6c84SAndroid Build Coastguard Worker     if (FLAG(f)) {
86*cf5a6c84SAndroid Build Coastguard Worker       if (!rc) {
87*cf5a6c84SAndroid Build Coastguard Worker         int temp;
88*cf5a6c84SAndroid Build Coastguard Worker 
89*cf5a6c84SAndroid Build Coastguard Worker         rc = rename(new, oldnew);
90*cf5a6c84SAndroid Build Coastguard Worker         temp = errno;
91*cf5a6c84SAndroid Build Coastguard Worker         if (rc && unlink(new)) perror_msg("unlink '%s'", new);
92*cf5a6c84SAndroid Build Coastguard Worker         errno = temp;
93*cf5a6c84SAndroid Build Coastguard Worker       }
94*cf5a6c84SAndroid Build Coastguard Worker       free(new);
95*cf5a6c84SAndroid Build Coastguard Worker       new = oldnew;
96*cf5a6c84SAndroid Build Coastguard Worker     }
97*cf5a6c84SAndroid Build Coastguard Worker     if (rc) perror_msg("cannot create %s link from '%s' to '%s'",
98*cf5a6c84SAndroid Build Coastguard Worker                        FLAG(s) ? "symbolic" : "hard", try, new);
99*cf5a6c84SAndroid Build Coastguard Worker     else if (FLAG(v)) fprintf(stderr, "'%s' -> '%s'\n", new, try);
100*cf5a6c84SAndroid Build Coastguard Worker 
101*cf5a6c84SAndroid Build Coastguard Worker     if (try != toys.optargs[i]) free(try);
102*cf5a6c84SAndroid Build Coastguard Worker     if (new != dest) free(new);
103*cf5a6c84SAndroid Build Coastguard Worker   }
104*cf5a6c84SAndroid Build Coastguard Worker }
105