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
98
99
100
101
102
103
104
105
106
107
108
109
|
/* getprop.c - Get an Android system property
*
* Copyright 2015 The Android Open Source Project
USE_GETPROP(NEWTOY(getprop, ">2Z", TOYFLAG_USR|TOYFLAG_SBIN))
config GETPROP
bool "getprop"
default y
depends on TOYBOX_ON_ANDROID
help
usage: getprop [NAME [DEFAULT]]
Gets an Android system property, or lists them all.
*/
#define FOR_getprop
#include "toys.h"
#include <sys/system_properties.h>
#include <selinux/android.h>
#include <selinux/label.h>
#include <selinux/selinux.h>
GLOBALS(
size_t size;
char **nv; // name/value pairs: even=name, odd=value
struct selabel_handle *handle;
)
static char *get_property_context(char *property)
{
char *context = NULL;
if (selabel_lookup(TT.handle, &context, property, 1)) {
perror_exit("unable to lookup label for \"%s\"", property);
}
return context;
}
static void add_property(const prop_info *pi, void *unused)
{
char name[PROP_NAME_MAX];
char value[PROP_VALUE_MAX];
__system_property_read(pi, name, value);
if (!(TT.size&31)) TT.nv = xrealloc(TT.nv, (TT.size+32)*2*sizeof(char *));
TT.nv[2*TT.size] = xstrdup(name);
if (toys.optflags & FLAG_Z) {
TT.nv[1+2*TT.size++] = get_property_context(name);
} else {
TT.nv[1+2*TT.size++] = xstrdup(value);
}
}
// Needed to supress extraneous "Loaded property_contexts from" message
static int selinux_log_callback_local(int type, const char *fmt, ...)
{
va_list ap;
if (type == SELINUX_INFO) return 0;
va_start(ap, fmt);
verror_msg((char *)fmt, 0, ap);
va_end(ap);
return 0;
}
void getprop_main(void)
{
if (toys.optflags & FLAG_Z) {
union selinux_callback cb;
cb.func_log = selinux_log_callback_local;
selinux_set_callback(SELINUX_CB_LOG, cb);
TT.handle = selinux_android_prop_context_handle();
if (!TT.handle) error_exit("unable to get selinux property context handle");
}
if (*toys.optargs) {
if (toys.optflags & FLAG_Z) {
char *context = get_property_context(*toys.optargs);
puts(context);
if (CFG_TOYBOX_FREE) free(context);
} else {
if (__system_property_get(*toys.optargs, toybuf) <= 0)
strcpy(toybuf, toys.optargs[1] ? toys.optargs[1] : "");
puts(toybuf);
}
} else {
size_t i;
if (__system_property_foreach(add_property, NULL))
error_exit("property_list");
qsort(TT.nv, TT.size, 2*sizeof(char *), qstrcmp);
for (i = 0; i<TT.size; i++) printf("[%s]: [%s]\n", TT.nv[i*2],TT.nv[1+i*2]);
if (CFG_TOYBOX_FREE) {
for (i = 0; i<TT.size; i++) {
free(TT.nv[i*2]);
free(TT.nv[1+i*2]);
}
free(TT.nv);
}
}
if (CFG_TOYBOX_FREE && (toys.optflags & FLAG_Z)) selabel_close(TT.handle);
}
|