summaryrefslogtreecommitdiffstats
path: root/tests/bugs/access-control/bug-1051896.c
blob: 27aa15594538cf74a88f16fb01f720d2747d91fc (plain)
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
#include <sys/param.h>
#include <sys/stat.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <grp.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <assert.h>
#include <sys/types.h>
#include <utime.h>
#include <sys/acl.h>

int do_setfacl(const char *path, const char *options, const char *textacl)
{
	int r;
	int type;
	acl_t acl;
	int dob;
	int dok;
	int dom;
	struct stat st;
	char textmode[30];

	r = 0;
	dob = strchr(options, 'b') != (char *)NULL;
	dok = strchr(options, 'k') != (char *)NULL;
	dom = strchr(options, 'm') != (char *)NULL;
	if ((dom && !textacl)
	    || (!dom && (textacl || (!dok && !dob) ||
                                    strchr(options, 'd')))) {
		errno = EBADRQC; /* "bad request" */
		r = -1;
	} else {
		if (dob || dok) {
			r = acl_delete_def_file(path);
		}
		if (dob && !r) {
			if (!stat(path, &st)) {
				sprintf(textmode,
                                        "u::%c%c%c,g::%c%c%c,o::%c%c%c",
					(st.st_mode & 0400 ? 'r' : '-'),
					(st.st_mode & 0200 ? 'w' : '-'),
					(st.st_mode & 0100 ? 'x' : '-'),
					(st.st_mode & 0040 ? 'r' : '-'),
					(st.st_mode & 0020 ? 'w' : '-'),
					(st.st_mode & 0010 ? 'x' : '-'),
					(st.st_mode & 004 ? 'r' : '-'),
					(st.st_mode & 002 ? 'w' : '-'),
					(st.st_mode & 001 ? 'x' : '-'));
				acl = acl_from_text(textmode);
				if (acl) {
					r = acl_set_file(path,
                                                         ACL_TYPE_ACCESS, acl);
					acl_free(acl);
				} else
					r = -1;
			} else
				r = -1;
		}
		if (!r && dom) {
			if (strchr(options, 'd'))
				type = ACL_TYPE_DEFAULT;
			else
				type = ACL_TYPE_ACCESS;
			acl = acl_from_text(textacl);
			if (acl) {
				r = acl_set_file(path, type, acl);
				acl_free(acl);
			} else
				r = -1;
		}
	}
	if (r)
		r = -errno;
	return r;
}


int main(int argc, char *argv[])
{
	int rc = 0;

	if (argc != 4) {
		fprintf(stderr,
                        "usage: ./setfacl_test <path> <options> <textacl>\n");
		return 0;
	}
	rc = do_setfacl(argv[1], argv[2], argv[3]);
	if (rc != 0) {
		fprintf(stderr, "do_setfacl failed: %s\n", strerror(errno));
		return rc;
	}
	return 0;
}