blob: f20b340eac1498570f872711e98aa6614e6033b2 [file] [log] [blame]
Eric Andersenf6be9441999-10-13 21:12:06 +00001/*
2 * Mini ln implementation for busybox
3 *
Eric Andersenc4996011999-10-20 22:08:37 +00004 *
5 * Copyright (C) 1999 by Lineo, inc.
6 * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org>
Eric Andersenf6be9441999-10-13 21:12:06 +00007 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 *
22 */
23
Eric Andersencc8ed391999-10-05 16:24:54 +000024#include "internal.h"
Erik Andersenfac10d72000-02-07 05:29:42 +000025#define BB_DECLARE_EXTERN
26#define bb_need_name_too_long
27#define bb_need_not_a_directory
28#include "messages.c"
29
Eric Andersencc8ed391999-10-05 16:24:54 +000030#include <stdio.h>
Eric Andersenf6be9441999-10-13 21:12:06 +000031#include <dirent.h>
Eric Andersencc8ed391999-10-05 16:24:54 +000032#include <errno.h>
Erik Andersenfac10d72000-02-07 05:29:42 +000033#include <sys/param.h> /* for PATH_MAX */
Eric Andersencc8ed391999-10-05 16:24:54 +000034
Erik Andersenfac10d72000-02-07 05:29:42 +000035static const char ln_usage[] =
36 "ln [OPTION] TARGET... LINK_NAME|DIRECTORY\n\n"
37 "Create a link named LINK_NAME or DIRECTORY to the specified TARGET\n\n"
38 "Options:\n"
39 "\t-s\tmake symbolic links instead of hard links\n"
40 "\t-f\tremove existing destination files\n"
41 "\t-n\tno dereference symlinks - treat like normal file\n";
Eric Andersenf6be9441999-10-13 21:12:06 +000042
43static int symlinkFlag = FALSE;
44static int removeoldFlag = FALSE;
Erik Andersenfac10d72000-02-07 05:29:42 +000045static int followLinks = TRUE;
Eric Andersenf6be9441999-10-13 21:12:06 +000046
47extern int ln_main(int argc, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +000048{
Erik Andersenfac10d72000-02-07 05:29:42 +000049 char *linkName;
50 int linkIntoDirFlag;
Eric Andersencc8ed391999-10-05 16:24:54 +000051
Eric Andersenf6be9441999-10-13 21:12:06 +000052 if (argc < 3) {
Eric Andersenb0e9a701999-10-18 22:28:26 +000053 usage (ln_usage);
Eric Andersenf6be9441999-10-13 21:12:06 +000054 }
55 argc--;
56 argv++;
57
58 /* Parse any options */
59 while (**argv == '-') {
60 while (*++(*argv))
61 switch (**argv) {
62 case 's':
63 symlinkFlag = TRUE;
64 break;
65 case 'f':
66 removeoldFlag = TRUE;
67 break;
Erik Andersenfac10d72000-02-07 05:29:42 +000068 case 'n':
69 followLinks = FALSE;
70 break;
Eric Andersenf6be9441999-10-13 21:12:06 +000071 default:
Eric Andersenb0e9a701999-10-18 22:28:26 +000072 usage (ln_usage);
Eric Andersenf6be9441999-10-13 21:12:06 +000073 }
74 argc--;
75 argv++;
76 }
77
Eric Andersen29d2e361999-11-06 06:07:27 +000078 linkName = argv[argc - 1];
Eric Andersenf6be9441999-10-13 21:12:06 +000079
Erik Andersenfac10d72000-02-07 05:29:42 +000080 if (strlen(linkName) > PATH_MAX) {
81 fprintf(stderr, name_too_long, "ln");
82 exit FALSE;
83 }
84
85 linkIntoDirFlag = isDirectory(linkName, TRUE);
86
87 if ((argc > 3) && !linkIntoDirFlag) {
88 fprintf(stderr, not_a_directory, "ln", linkName);
89 exit FALSE;
Eric Andersenf6be9441999-10-13 21:12:06 +000090 }
91
92 while (argc-- >= 2) {
Erik Andersenfac10d72000-02-07 05:29:42 +000093 char srcName[PATH_MAX + 1];
94 int nChars, status;
95
96 if (strlen(*argv) > PATH_MAX) {
97 fprintf(stderr, name_too_long, "ln");
98 exit FALSE;
99 }
100
101 if (followLinks == FALSE) {
102 strcpy(srcName, *argv);
103 } else {
104 /* Warning! This can silently truncate if > PATH_MAX, but
105 I don't think that there can be one > PATH_MAX anyway. */
106 nChars = readlink(*argv, srcName, PATH_MAX);
107 srcName[nChars] = '\0';
108 }
109
110 if (removeoldFlag == TRUE) {
Eric Andersen29d2e361999-11-06 06:07:27 +0000111 status = ( unlink(linkName) && errno != ENOENT );
Erik Andersenfac10d72000-02-07 05:29:42 +0000112 if (status != 0) {
Eric Andersen29d2e361999-11-06 06:07:27 +0000113 perror(linkName);
Erik Andersenfac10d72000-02-07 05:29:42 +0000114 exit FALSE;
Eric Andersenf6be9441999-10-13 21:12:06 +0000115 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000116 }
Erik Andersenfac10d72000-02-07 05:29:42 +0000117
118 if (symlinkFlag == TRUE)
119 status = symlink(*argv, linkName);
Eric Andersencc8ed391999-10-05 16:24:54 +0000120 else
Erik Andersenfac10d72000-02-07 05:29:42 +0000121 status = link(*argv, linkName);
122 if (status != 0) {
Eric Andersen29d2e361999-11-06 06:07:27 +0000123 perror(linkName);
Erik Andersenfac10d72000-02-07 05:29:42 +0000124 exit FALSE;
Eric Andersenf6be9441999-10-13 21:12:06 +0000125 }
126 }
Erik Andersenfac10d72000-02-07 05:29:42 +0000127 exit TRUE;
Eric Andersencc8ed391999-10-05 16:24:54 +0000128}