1 #include <unistd.h>
2 #include <fcntl.h>
3 #include <string.h>
4 #include <stdlib.h>
5 #include <errno.h>
6 #include <sys/xattr.h>
7 #include "selinux_internal.h"
8 #include "policy.h"
9 
fgetfilecon_raw(int fd,char ** context)10 int fgetfilecon_raw(int fd, char ** context)
11 {
12 	char *buf;
13 	ssize_t size;
14 	ssize_t ret;
15 
16 	size = INITCONTEXTLEN + 1;
17 	buf = malloc(size);
18 	if (!buf)
19 		return -1;
20 	memset(buf, 0, size);
21 
22 	ret = fgetxattr(fd, XATTR_NAME_SELINUX, buf, size - 1);
23 	if (ret < 0 && errno == ERANGE) {
24 		char *newbuf;
25 
26 		size = fgetxattr(fd, XATTR_NAME_SELINUX, NULL, 0);
27 		if (size < 0)
28 			goto out;
29 
30 		size++;
31 		newbuf = realloc(buf, size);
32 		if (!newbuf)
33 			goto out;
34 
35 		buf = newbuf;
36 		memset(buf, 0, size);
37 		ret = fgetxattr(fd, XATTR_NAME_SELINUX, buf, size - 1);
38 	}
39       out:
40 	if (ret == 0) {
41 		/* Re-map empty attribute values to errors. */
42 		errno = ENOTSUP;
43 		ret = -1;
44 	}
45 	if (ret < 0)
46 		free(buf);
47 	else
48 		*context = buf;
49 	return ret;
50 }
51 
hidden_def(fgetfilecon_raw)52 hidden_def(fgetfilecon_raw)
53 
54 int fgetfilecon(int fd, char ** context)
55 {
56 	char * rcontext = NULL;
57 	int ret;
58 
59 	*context = NULL;
60 
61 	ret = fgetfilecon_raw(fd, &rcontext);
62 
63 	if (ret > 0) {
64 		ret = selinux_raw_to_trans_context(rcontext, context);
65 		freecon(rcontext);
66 	}
67 
68 	if (ret >= 0 && *context)
69 		return strlen(*context) + 1;
70 
71 	return ret;
72 }
73