aboutsummaryrefslogtreecommitdiff
path: root/core/ln.c
blob: 301ba7d98dea71e732bb95588f25d4caedcbcfc8 (plain) (blame)
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
/* SPDX-License-Identifier: BSD-3-Clause */
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>

#define REQ_PRINT_USAGE /* Require print_usage() from ../common/common.h */
#define REQ_ERRPRINT /* Require errprint() from ../common/common.h */
#define DESCRIPTION "Link files."
#define OPERANDS    "[-fs] [-P|-L] source_file target_file"
#include "../common/common.h"

int  getopt(int argc, char *const argv[], const char *optstring);

int main(int argc, char *const argv[]) {
	int argument;
	char param[256], *buffer = NULL, *argv0 = strdup(argv[0]);

	if (argc == 1) {
		print_usage(argv[0], DESCRIPTION, OPERANDS, VERSION);
		return 1;
	}
	while ((argument = getopt(argc, argv, "fsPL")) != -1) {
		if (argument == '?') {
			print_usage(argv[0], DESCRIPTION, OPERANDS, VERSION);
			return 1;
		}
		param[argument] = argument;
	} argc -= optind; argv += optind;
	for (int i = 0; i < (argc - 1); i++) {
		if (param['f']) remove(argv[argc - 1]);
		if (errno && errno != ENOENT) return errprint(argv0, argv[i], errno);
		errno = 0; /* Not reached if errno == ENOENT (no such file) */
		if (param['s']) symlink(argv[i], argv[argc - 1]);
		/* The -P option is the default behavior (at least on musl), 
		 * so no if statement. 
		 */
		else if (param['L']) {
			readlink(argv[i], buffer, strlen(buffer)); /* Read the link */
			if (errno) return errprint(argv0, argv[i], errno);
			link(buffer, argv[argc - 1]);
		}
		else link(argv[i], argv[argc - 1]);
		if (errno) return errprint(argv0, argv[i], errno);
	}
	return 0;
}