blob: beaa58fac52c5f010bc8e4b080c47c1c78101951 [file] [log] [blame]
Erik Andersene49d5ec2000-02-08 19:58:47 +00001/* vi: set sw=4 ts=4: */
Eric Andersenf6be9441999-10-13 21:12:06 +00002/*
3 * Mini ln implementation for busybox
4 *
Erik Andersen61677fe2000-04-13 01:18:56 +00005 * Copyright (C) 1999,2000 by Lineo, inc.
Eric Andersenc4996011999-10-20 22:08:37 +00006 * 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>
33
Erik Andersenfac10d72000-02-07 05:29:42 +000034static const char ln_usage[] =
Erik Andersen7ab9c7e2000-05-12 19:41:47 +000035 "ln [OPTION] TARGET... LINK_NAME|DIRECTORY\n"
36#ifndef BB_FEATURE_TRIVIAL_HELP
Eric Andersenaddc9c22000-06-21 22:53:24 +000037 "\nCreate a link named LINK_NAME or DIRECTORY to the specified TARGET\n"
38 "\nYou may use '--' to indicate that all following arguments are non-options.\n\n"
Erik Andersene49d5ec2000-02-08 19:58:47 +000039 "Options:\n"
40 "\t-s\tmake symbolic links instead of hard links\n"
41
42 "\t-f\tremove existing destination files\n"
Eric Andersen6f9985e2000-06-22 00:02:56 +000043#if 0
Erik Andersen7ab9c7e2000-05-12 19:41:47 +000044 "\t-n\tno dereference symlinks - treat like normal file\n"
45#endif
Eric Andersen6f9985e2000-06-22 00:02:56 +000046#endif
Erik Andersen7ab9c7e2000-05-12 19:41:47 +000047 ;
Eric Andersenf6be9441999-10-13 21:12:06 +000048
49static int symlinkFlag = FALSE;
50static int removeoldFlag = FALSE;
Erik Andersenfac10d72000-02-07 05:29:42 +000051static int followLinks = TRUE;
Eric Andersenf6be9441999-10-13 21:12:06 +000052
53extern int ln_main(int argc, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +000054{
Eric Andersen0b4551f2000-07-10 16:44:03 +000055 char *linkName, *dirName;
Erik Andersene49d5ec2000-02-08 19:58:47 +000056 int linkIntoDirFlag;
Pavel Roskin2e965232000-06-15 18:04:40 +000057 int stopIt = FALSE;
Eric Andersencc8ed391999-10-05 16:24:54 +000058
Eric Andersenf6be9441999-10-13 21:12:06 +000059 argc--;
60 argv++;
Eric Andersenf6be9441999-10-13 21:12:06 +000061
Erik Andersene49d5ec2000-02-08 19:58:47 +000062 /* Parse any options */
Pavel Roskin2e965232000-06-15 18:04:40 +000063 while (argc > 0 && stopIt == FALSE) {
64 if (**argv == '-') {
Eric Andersen815e9042000-06-06 16:15:23 +000065 while (*++(*argv))
66 switch (**argv) {
67 case 's':
68 symlinkFlag = TRUE;
69 break;
70 case 'f':
71 removeoldFlag = TRUE;
72 break;
73 case 'n':
74 followLinks = FALSE;
75 break;
Pavel Roskin2e965232000-06-15 18:04:40 +000076 case '-':
77 stopIt = TRUE;
78 break;
Eric Andersen815e9042000-06-06 16:15:23 +000079 default:
80 usage(ln_usage);
81 }
Pavel Roskin2e965232000-06-15 18:04:40 +000082 argc--;
83 argv++;
Eric Andersen815e9042000-06-06 16:15:23 +000084 }
Pavel Roskin2e965232000-06-15 18:04:40 +000085 else
86 break;
Erik Andersenfac10d72000-02-07 05:29:42 +000087 }
88
Pavel Roskin2e965232000-06-15 18:04:40 +000089 if (argc < 2) {
90 usage(ln_usage);
Eric Andersen815e9042000-06-06 16:15:23 +000091 }
92
Erik Andersene49d5ec2000-02-08 19:58:47 +000093 linkName = argv[argc - 1];
Erik Andersenfac10d72000-02-07 05:29:42 +000094
Erik Andersen4f3f7572000-04-28 00:18:56 +000095 if (strlen(linkName) > BUFSIZ) {
Erik Andersene49d5ec2000-02-08 19:58:47 +000096 fprintf(stderr, name_too_long, "ln");
Erik Andersenfac10d72000-02-07 05:29:42 +000097 exit FALSE;
Eric Andersencc8ed391999-10-05 16:24:54 +000098 }
Erik Andersenfac10d72000-02-07 05:29:42 +000099
Erik Andersen9ffdaa62000-02-11 21:55:04 +0000100 linkIntoDirFlag = isDirectory(linkName, TRUE, NULL);
Erik Andersene49d5ec2000-02-08 19:58:47 +0000101
Pavel Roskin110fc0c2000-06-18 00:02:24 +0000102 if ((argc >= 3) && linkIntoDirFlag == FALSE) {
Erik Andersene49d5ec2000-02-08 19:58:47 +0000103 fprintf(stderr, not_a_directory, "ln", linkName);
104 exit FALSE;
Eric Andersenf6be9441999-10-13 21:12:06 +0000105 }
Erik Andersene49d5ec2000-02-08 19:58:47 +0000106
Eric Andersen0b4551f2000-07-10 16:44:03 +0000107 if (linkIntoDirFlag == TRUE)
108 dirName = linkName;
109
Erik Andersene49d5ec2000-02-08 19:58:47 +0000110 while (argc-- >= 2) {
Eric Andersen6f9985e2000-06-22 00:02:56 +0000111#if 0
Erik Andersen4f3f7572000-04-28 00:18:56 +0000112 char srcName[BUFSIZ + 1];
Eric Andersen6f9985e2000-06-22 00:02:56 +0000113 int nChars;
114#endif
115 int status;
Erik Andersene49d5ec2000-02-08 19:58:47 +0000116
Erik Andersen4f3f7572000-04-28 00:18:56 +0000117 if (strlen(*argv) > BUFSIZ) {
Erik Andersene49d5ec2000-02-08 19:58:47 +0000118 fprintf(stderr, name_too_long, "ln");
119 exit FALSE;
120 }
121
Eric Andersen6f9985e2000-06-22 00:02:56 +0000122#if 0
Erik Andersene49d5ec2000-02-08 19:58:47 +0000123 if (followLinks == FALSE) {
124 strcpy(srcName, *argv);
125 } else {
Erik Andersen4f3f7572000-04-28 00:18:56 +0000126 /* Warning! This can silently truncate if > BUFSIZ, but
127 I don't think that there can be one > BUFSIZ anyway. */
128 nChars = readlink(*argv, srcName, BUFSIZ);
Erik Andersene49d5ec2000-02-08 19:58:47 +0000129 srcName[nChars] = '\0';
130 }
Eric Andersen6f9985e2000-06-22 00:02:56 +0000131#endif
Eric Andersen0b4551f2000-07-10 16:44:03 +0000132 if (linkIntoDirFlag == TRUE) {
133 char *baseName = get_last_path_component(*argv);
134 linkName = (char *)malloc(strlen(dirName)+strlen(baseName)+2);
135 strcpy(linkName, dirName);
136 if(dirName[strlen(dirName)-1] != '/')
137 strcat(linkName, "/");
138 strcat(linkName,baseName);
139 }
Erik Andersene49d5ec2000-02-08 19:58:47 +0000140
141 if (removeoldFlag == TRUE) {
142 status = (unlink(linkName) && errno != ENOENT);
143 if (status != 0) {
144 perror(linkName);
145 exit FALSE;
146 }
147 }
148
149 if (symlinkFlag == TRUE)
150 status = symlink(*argv, linkName);
151 else
152 status = link(*argv, linkName);
153 if (status != 0) {
154 perror(linkName);
155 exit FALSE;
156 }
Eric Andersen0b4551f2000-07-10 16:44:03 +0000157
158 if (linkIntoDirFlag)
159 free(linkName);
160
161 argv++;
Erik Andersene49d5ec2000-02-08 19:58:47 +0000162 }
Eric Andersenb6106152000-06-19 17:25:40 +0000163 return( TRUE);
Eric Andersencc8ed391999-10-05 16:24:54 +0000164}
Erik Andersen029011b2000-03-04 21:19:32 +0000165
166/*
167Local Variables:
168c-file-style: "linux"
169c-basic-offset: 4
170tab-width: 4
171End:
172*/