]> git.saurik.com Git - apple/libc.git/blame_incremental - libdarwin/bsd.c
Libc-1353.41.1.tar.gz
[apple/libc.git] / libdarwin / bsd.c
... / ...
CommitLineData
1/*
2 * Copyright (c) 2018 Apple Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
11 * file.
12 *
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
20 *
21 * @APPLE_LICENSE_HEADER_END@
22 */
23#include "internal.h"
24
25#pragma mark Utilities
26
27/*
28 * Factored out from _get_parse_boot_arg_value for unit testing purposes
29 */
30static bool
31_parse_boot_arg_value(char *argsbuff, const char *which, char *where, size_t max)
32{
33 bool found = false;
34
35 char *token = NULL;
36 char *argsstr = argsbuff;
37 static const char seps[] = { ' ', '\t', };
38 while ((token = strsep(&argsstr, seps)) != NULL) {
39 bool is_boolean = false;
40
41 char *value = NULL;
42 char *equals = strchr(token, '=');
43 if (token[0] == '-') {
44 /*
45 * Arguments whose names begins with "-" are booleans, so don't get
46 * key=value splitting. Though I'd still discourage you from
47 * naming your option "-edge=case".
48 */
49 is_boolean = true;
50 } else if (equals) {
51 equals[0] = '\0';
52 value = &equals[1];
53 } else {
54 is_boolean = true;
55 }
56
57 if (strcmp(which, token) == 0) {
58 /*
59 * Found it! Copy out the value as required.
60 */
61 found = true;
62
63 if (!where) {
64 // Caller just wants to know whether the boot-arg exists.
65 } else if (is_boolean || value == NULL) {
66 strlcpy(where, "", max);
67 } else {
68 strlcpy(where, value, max);
69 }
70
71 break;
72 }
73 }
74
75 return found;
76}
77
78/*
79 * This is (very) loosely based on the implementation of
80 * PE_parse_boot_argn() (or at least the parts where I was able to easily
81 * decipher the policy).
82 */
83static bool
84_get_boot_arg_value(const char *which, char *where, size_t max)
85{
86 bool found = false;
87 __os_free char *argsbuff = NULL;
88 size_t argsbuff_len = 0;
89 errno_t error = sysctlbyname_get_data_np("kern.bootargs",
90 (void **)&argsbuff, &argsbuff_len);
91
92 if (!error) {
93 found = _parse_boot_arg_value(argsbuff, which, where, max);
94 }
95
96 return found;
97}
98
99#pragma mark API
100errno_t
101sysctl_get_data_np(int mib[4], size_t mib_cnt, void **buff, size_t *buff_len)
102{
103 errno_t error = 0;
104 int ret = 0;
105 size_t needed = 0;
106 void *mybuff = NULL;
107
108 // We need to get the length of the parameter so we can allocate a buffer
109 // that's large enough.
110 ret = sysctl(mib, (unsigned int)mib_cnt, NULL, &needed, NULL, 0);
111 if (ret) {
112 error = errno;
113 goto __out;
114 }
115
116 mybuff = malloc(needed);
117 if (!mybuff) {
118 error = errno;
119 goto __out;
120 }
121
122 ret = sysctl(mib, (unsigned int)mib_cnt, mybuff, &needed, NULL, 0);
123 if (ret) {
124 // It's conceivable that some other process came along within this
125 // window and modified the variable to be even larger than we'd
126 // previously been told, but if that's the case, just give up.
127 error = errno;
128 goto __out;
129 }
130
131 *buff = mybuff;
132 *buff_len = needed;
133
134__out:
135 if (error) {
136 free(mybuff);
137 }
138 return error;
139}
140
141errno_t
142sysctlbyname_get_data_np(const char *mibdesc, void **buff, size_t *buff_len)
143{
144 int ret = -1;
145 int error = -1;
146 int mib[4];
147 size_t mib_cnt = countof(mib);
148
149 ret = sysctlnametomib(mibdesc, mib, &mib_cnt);
150 if (ret) {
151 error = errno;
152 goto __out;
153 }
154
155 error = sysctl_get_data_np(mib, mib_cnt, buff, buff_len);
156
157__out:
158 return error;
159}
160
161bool
162os_parse_boot_arg_int(const char *which, int64_t *where)
163{
164 bool found = false;
165 char buff[24] = {0};
166 char *endptr = NULL;
167 int64_t val = 0;
168
169 found = _get_boot_arg_value(which, buff, sizeof(buff));
170 if (!found || !where) {
171 goto __out;
172 }
173
174 // A base of zero handles bases 8, 10, and 16.
175 val = strtoll(buff, &endptr, 0);
176 if (*endptr == 0) {
177 *where = val;
178 } else {
179 // The boot-arg value was invalid, so say we didn't find it.
180 found = false;
181 }
182
183__out:
184 return found;
185}
186
187bool
188os_parse_boot_arg_string(const char *which, char *where, size_t maxlen)
189{
190 return _get_boot_arg_value(which, where, maxlen);
191}