1*8d741a5dSApple OSS Distributions #include <errno.h>
2*8d741a5dSApple OSS Distributions #include <stdlib.h>
3*8d741a5dSApple OSS Distributions #include <libgen.h>
4*8d741a5dSApple OSS Distributions #include <limits.h>
5*8d741a5dSApple OSS Distributions #include <mach-o/dyld.h>
6*8d741a5dSApple OSS Distributions #include <sys/types.h>
7*8d741a5dSApple OSS Distributions #include <sys/sysctl.h>
8*8d741a5dSApple OSS Distributions #include <xlocale.h>
9*8d741a5dSApple OSS Distributions
10*8d741a5dSApple OSS Distributions #include <darwintest.h>
11*8d741a5dSApple OSS Distributions #include <darwintest_utils.h>
12*8d741a5dSApple OSS Distributions
13*8d741a5dSApple OSS Distributions #include "drop_priv.h"
14*8d741a5dSApple OSS Distributions #include "test_utils.h"
15*8d741a5dSApple OSS Distributions
16*8d741a5dSApple OSS Distributions #if ENTITLED
17*8d741a5dSApple OSS Distributions #define SET_TREATMENT_ID set_treatment_id_entitled
18*8d741a5dSApple OSS Distributions #define SET_TREATMENT_ID_DESCR "Can set treatment id with entitlement"
19*8d741a5dSApple OSS Distributions #else /* ENTITLED */
20*8d741a5dSApple OSS Distributions #define SET_TREATMENT_ID set_treatment_id_unentitled
21*8d741a5dSApple OSS Distributions #define SET_TREATMENT_ID_DESCR "Can't set treatment id without entitlement"
22*8d741a5dSApple OSS Distributions #endif /* ENTITLED */
23*8d741a5dSApple OSS Distributions
24*8d741a5dSApple OSS Distributions T_DECL(SET_TREATMENT_ID, "Verifies that EXPERIMENT sysctls can only be set with the entitlement", T_META_ASROOT(false))
25*8d741a5dSApple OSS Distributions {
26*8d741a5dSApple OSS Distributions #define TEST_STR "testing"
27*8d741a5dSApple OSS Distributions #define IDENTIFIER_LENGTH 36
28*8d741a5dSApple OSS Distributions
29*8d741a5dSApple OSS Distributions int ret;
30*8d741a5dSApple OSS Distributions errno_t err;
31*8d741a5dSApple OSS Distributions char val[IDENTIFIER_LENGTH + 1] = {0};
32*8d741a5dSApple OSS Distributions size_t len = sizeof(val);
33*8d741a5dSApple OSS Distributions char new_val[IDENTIFIER_LENGTH + 1] = {0};
34*8d741a5dSApple OSS Distributions
35*8d741a5dSApple OSS Distributions if (!is_development_kernel()) {
36*8d741a5dSApple OSS Distributions T_SKIP("skipping test on release kernel");
37*8d741a5dSApple OSS Distributions }
38*8d741a5dSApple OSS Distributions
39*8d741a5dSApple OSS Distributions strlcpy(new_val, TEST_STR, sizeof(new_val));
40*8d741a5dSApple OSS Distributions if (running_as_root()) {
41*8d741a5dSApple OSS Distributions drop_priv();
42*8d741a5dSApple OSS Distributions }
43*8d741a5dSApple OSS Distributions
44*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.trial_treatment_id", val, &len, new_val, strlen(new_val));
45*8d741a5dSApple OSS Distributions err = errno;
46*8d741a5dSApple OSS Distributions #if ENTITLED
47*8d741a5dSApple OSS Distributions len = sizeof(val);
48*8d741a5dSApple OSS Distributions memset(new_val, 0, sizeof(new_val));
49*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "set kern.trial_treatment_id");
50*8d741a5dSApple OSS Distributions /* Cleanup. Set it back to the empty string. */
51*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.trial_treatment_id", val, &len, new_val, 1);
52*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(ret, "reset kern.trial_treatment_id");
53*8d741a5dSApple OSS Distributions #else
54*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_FAILURE(ret, EPERM, "set kern.trial_treatment_id");
55*8d741a5dSApple OSS Distributions #endif /* ENTITLED */
56*8d741a5dSApple OSS Distributions }
57*8d741a5dSApple OSS Distributions
58*8d741a5dSApple OSS Distributions #if ENTITLED
59*8d741a5dSApple OSS Distributions /* Check min and max value limits on numeric factors */
60*8d741a5dSApple OSS Distributions T_DECL(experiment_factor_numeric_limits,
61*8d741a5dSApple OSS Distributions "Can only set factors within the legal range.",
62*8d741a5dSApple OSS Distributions T_META_ASROOT(false))
63*8d741a5dSApple OSS Distributions {
64*8d741a5dSApple OSS Distributions #define kMinVal 5 /* The min value allowed for the testing factor. */
65*8d741a5dSApple OSS Distributions #define kMaxVal 10 /* The max value allowed for the testing factor. */
66*8d741a5dSApple OSS Distributions errno_t err;
67*8d741a5dSApple OSS Distributions int ret;
68*8d741a5dSApple OSS Distributions unsigned int current_val;
69*8d741a5dSApple OSS Distributions size_t len = sizeof(current_val);
70*8d741a5dSApple OSS Distributions unsigned int new_val;
71*8d741a5dSApple OSS Distributions
72*8d741a5dSApple OSS Distributions if (running_as_root()) {
73*8d741a5dSApple OSS Distributions drop_priv();
74*8d741a5dSApple OSS Distributions }
75*8d741a5dSApple OSS Distributions new_val = kMinVal - 1;
76*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.testing_experiment_factor", ¤t_val, &len, &new_val, sizeof(new_val));
77*8d741a5dSApple OSS Distributions err = errno;
78*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_FAILURE(ret, EINVAL, "set kern.testing_experiment_factor below range.");
79*8d741a5dSApple OSS Distributions
80*8d741a5dSApple OSS Distributions new_val = kMaxVal + 1;
81*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.testing_experiment_factor", ¤t_val, &len, &new_val, sizeof(new_val));
82*8d741a5dSApple OSS Distributions err = errno;
83*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_FAILURE(ret, EINVAL, "set kern.testing_experiment_factor above range.");
84*8d741a5dSApple OSS Distributions
85*8d741a5dSApple OSS Distributions new_val = kMaxVal;
86*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.testing_experiment_factor", ¤t_val, &len, &new_val, sizeof(new_val));
87*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "set kern.testing_experiment_factor at top of range.");
88*8d741a5dSApple OSS Distributions
89*8d741a5dSApple OSS Distributions new_val = kMinVal;
90*8d741a5dSApple OSS Distributions ret = sysctlbyname("kern.testing_experiment_factor", ¤t_val, &len, &new_val, sizeof(new_val));
91*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "set kern.testing_experiment_factor at bottom of range.");
92*8d741a5dSApple OSS Distributions }
93*8d741a5dSApple OSS Distributions
94*8d741a5dSApple OSS Distributions static uint64_t original_libmalloc_experiment_value = 0;
95*8d741a5dSApple OSS Distributions
96*8d741a5dSApple OSS Distributions static void
reset_libmalloc_experiment(void)97*8d741a5dSApple OSS Distributions reset_libmalloc_experiment(void)
98*8d741a5dSApple OSS Distributions {
99*8d741a5dSApple OSS Distributions int ret = sysctlbyname("kern.libmalloc_experiments", NULL, NULL, &original_libmalloc_experiment_value, sizeof(original_libmalloc_experiment_value));
100*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "reset kern.libmalloc_experiments");
101*8d741a5dSApple OSS Distributions }
102*8d741a5dSApple OSS Distributions
103*8d741a5dSApple OSS Distributions static void
set_libmalloc_experiment(uint64_t val)104*8d741a5dSApple OSS Distributions set_libmalloc_experiment(uint64_t val)
105*8d741a5dSApple OSS Distributions {
106*8d741a5dSApple OSS Distributions T_LOG("Setting kern.libmalloc_experiments to %llu", val);
107*8d741a5dSApple OSS Distributions size_t len = sizeof(original_libmalloc_experiment_value);
108*8d741a5dSApple OSS Distributions int ret = sysctlbyname("kern.libmalloc_experiments", &original_libmalloc_experiment_value, &len, &val, sizeof(val));
109*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "set kern.libmalloc_experiments");
110*8d741a5dSApple OSS Distributions T_ATEND(reset_libmalloc_experiment);
111*8d741a5dSApple OSS Distributions }
112*8d741a5dSApple OSS Distributions
113*8d741a5dSApple OSS Distributions #define PRINT_APPLE_ARRAY_TOOL "tools/print_apple_array"
114*8d741a5dSApple OSS Distributions /*
115*8d741a5dSApple OSS Distributions * Spawns a new binary and returns the contents of its apple array
116*8d741a5dSApple OSS Distributions * (after libsystem initialization).
117*8d741a5dSApple OSS Distributions */
118*8d741a5dSApple OSS Distributions static char **
get_apple_array(size_t * num_array_entries,const char * filename)119*8d741a5dSApple OSS Distributions get_apple_array(size_t *num_array_entries, const char * filename)
120*8d741a5dSApple OSS Distributions {
121*8d741a5dSApple OSS Distributions if (filename == NULL) {
122*8d741a5dSApple OSS Distributions filename = PRINT_APPLE_ARRAY_TOOL;
123*8d741a5dSApple OSS Distributions }
124*8d741a5dSApple OSS Distributions int ret;
125*8d741a5dSApple OSS Distributions char stdout_path[MAXPATHLEN] = "apple_array.txt";
126*8d741a5dSApple OSS Distributions dt_resultfile(stdout_path, MAXPATHLEN);
127*8d741a5dSApple OSS Distributions int exit_status = 0, signum = 0;
128*8d741a5dSApple OSS Distributions char binary_path[MAXPATHLEN], binary_dir[MAXPATHLEN];
129*8d741a5dSApple OSS Distributions char *char_ret;
130*8d741a5dSApple OSS Distributions const static size_t kMaxNumArguments = 256;
131*8d741a5dSApple OSS Distributions size_t linecap = 0;
132*8d741a5dSApple OSS Distributions ssize_t linelen = 0;
133*8d741a5dSApple OSS Distributions char **apple_array;
134*8d741a5dSApple OSS Distributions char **line = NULL;
135*8d741a5dSApple OSS Distributions size_t num_lines = 0;
136*8d741a5dSApple OSS Distributions FILE *stdout_f = NULL;
137*8d741a5dSApple OSS Distributions uint32_t name_size = MAXPATHLEN;
138*8d741a5dSApple OSS Distributions
139*8d741a5dSApple OSS Distributions ret = _NSGetExecutablePath(binary_path, &name_size);
140*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_EQ(ret, 0, "_NSGetExecutablePath");
141*8d741a5dSApple OSS Distributions char_ret = dirname_r(binary_path, binary_dir);
142*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_TRUE(char_ret != NULL, "dirname_r");
143*8d741a5dSApple OSS Distributions snprintf(binary_path, MAXPATHLEN, "%s/%s", binary_dir, filename);
144*8d741a5dSApple OSS Distributions
145*8d741a5dSApple OSS Distributions char *launch_tool_args[] = {
146*8d741a5dSApple OSS Distributions binary_path,
147*8d741a5dSApple OSS Distributions NULL
148*8d741a5dSApple OSS Distributions };
149*8d741a5dSApple OSS Distributions pid_t child_pid;
150*8d741a5dSApple OSS Distributions ret = dt_launch_tool(&child_pid, launch_tool_args, false, stdout_path, NULL);
151*8d741a5dSApple OSS Distributions T_WITH_ERRNO; T_ASSERT_EQ(ret, 0, "dt_launch_tool: %s", binary_path);
152*8d741a5dSApple OSS Distributions
153*8d741a5dSApple OSS Distributions ret = dt_waitpid(child_pid, &exit_status, &signum, 60 * 5);
154*8d741a5dSApple OSS Distributions T_ASSERT_EQ(ret, 1, "dt_waitpid");
155*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_EQ(exit_status, 0, "dt_waitpid: exit_status");
156*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_EQ(signum, 0, "dt_waitpid: signum");
157*8d741a5dSApple OSS Distributions
158*8d741a5dSApple OSS Distributions stdout_f = fopen(stdout_path, "r");
159*8d741a5dSApple OSS Distributions T_WITH_ERRNO; T_ASSERT_NOTNULL(stdout_f, "open(%s)", stdout_path);
160*8d741a5dSApple OSS Distributions apple_array = calloc(kMaxNumArguments, sizeof(char *));
161*8d741a5dSApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(apple_array, "calloc: %lu\n", sizeof(char *) * kMaxNumArguments);
162*8d741a5dSApple OSS Distributions while (num_lines < kMaxNumArguments) {
163*8d741a5dSApple OSS Distributions line = &(apple_array[num_lines++]);
164*8d741a5dSApple OSS Distributions linecap = 0;
165*8d741a5dSApple OSS Distributions linelen = getline(line, &linecap, stdout_f);
166*8d741a5dSApple OSS Distributions if (linelen == -1) {
167*8d741a5dSApple OSS Distributions break;
168*8d741a5dSApple OSS Distributions }
169*8d741a5dSApple OSS Distributions }
170*8d741a5dSApple OSS Distributions *num_array_entries = num_lines - 1;
171*8d741a5dSApple OSS Distributions
172*8d741a5dSApple OSS Distributions ret = fclose(stdout_f);
173*8d741a5dSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ret, "fclose(%s)", stdout_path);
174*8d741a5dSApple OSS Distributions
175*8d741a5dSApple OSS Distributions return apple_array;
176*8d741a5dSApple OSS Distributions }
177*8d741a5dSApple OSS Distributions
178*8d741a5dSApple OSS Distributions #define LIBMALLOC_EXPERIMENT_FACTORS_KEY "MallocExperiment="
179*8d741a5dSApple OSS Distributions
180*8d741a5dSApple OSS Distributions #define HARDENED_RUNTIME_KEY "HardenedRuntime="
181*8d741a5dSApple OSS Distributions
182*8d741a5dSApple OSS Distributions
183*8d741a5dSApple OSS Distributions /*
184*8d741a5dSApple OSS Distributions * Get the value of the key in the apple array.
185*8d741a5dSApple OSS Distributions * Returns true iff the key is present.
186*8d741a5dSApple OSS Distributions */
187*8d741a5dSApple OSS Distributions static bool
get_apple_array_key(char ** apple_array,size_t num_array_entries,uint64_t * factors,const char * key)188*8d741a5dSApple OSS Distributions get_apple_array_key(char **apple_array, size_t num_array_entries, uint64_t *factors, const char *key)
189*8d741a5dSApple OSS Distributions {
190*8d741a5dSApple OSS Distributions bool found = false;
191*8d741a5dSApple OSS Distributions for (size_t i = 0; i < num_array_entries; i++) {
192*8d741a5dSApple OSS Distributions char *str = apple_array[i];
193*8d741a5dSApple OSS Distributions if (strstr(str, key)) {
194*8d741a5dSApple OSS Distributions found = true;
195*8d741a5dSApple OSS Distributions if (factors != NULL) {
196*8d741a5dSApple OSS Distributions str = strchr(str, '=');
197*8d741a5dSApple OSS Distributions T_ASSERT_NOTNULL(str, "skip over =");
198*8d741a5dSApple OSS Distributions ++str;
199*8d741a5dSApple OSS Distributions *factors = strtoull_l(str, NULL, 16, NULL);
200*8d741a5dSApple OSS Distributions }
201*8d741a5dSApple OSS Distributions break;
202*8d741a5dSApple OSS Distributions }
203*8d741a5dSApple OSS Distributions }
204*8d741a5dSApple OSS Distributions return found;
205*8d741a5dSApple OSS Distributions }
206*8d741a5dSApple OSS Distributions
207*8d741a5dSApple OSS Distributions /* libmalloc relies on these values not changing. If they change,
208*8d741a5dSApple OSS Distributions * you need to update the values in that project as well */
209*8d741a5dSApple OSS Distributions __options_decl(HR_flags_t, uint32_t, {
210*8d741a5dSApple OSS Distributions BrowserHostEntitlementMask = 0x01,
211*8d741a5dSApple OSS Distributions BrowserGPUEntitlementMask = 0x02,
212*8d741a5dSApple OSS Distributions BrowserNetworkEntitlementMask = 0x04,
213*8d741a5dSApple OSS Distributions BrowserWebContentEntitlementMask = 0x08,
214*8d741a5dSApple OSS Distributions });
215*8d741a5dSApple OSS Distributions
216*8d741a5dSApple OSS Distributions T_DECL(libmalloc_hardened_binary_present,
217*8d741a5dSApple OSS Distributions "hardened binary flags show up in apple array",
218*8d741a5dSApple OSS Distributions T_META_ASROOT(false))
219*8d741a5dSApple OSS Distributions {
220*8d741a5dSApple OSS Distributions uint64_t apple_array_val = 0;
221*8d741a5dSApple OSS Distributions size_t num_array_entries = 0;
222*8d741a5dSApple OSS Distributions char **apple_array;
223*8d741a5dSApple OSS Distributions bool found = false;
224*8d741a5dSApple OSS Distributions
225*8d741a5dSApple OSS Distributions /* These are the entitlements on the HR1 binary */
226*8d741a5dSApple OSS Distributions uint32_t mask_val = BrowserHostEntitlementMask | BrowserGPUEntitlementMask | BrowserWebContentEntitlementMask;
227*8d741a5dSApple OSS Distributions apple_array = get_apple_array(&num_array_entries, "tools/print_apple_array_HR1");
228*8d741a5dSApple OSS Distributions found = get_apple_array_key(apple_array, num_array_entries, &apple_array_val, HARDENED_RUNTIME_KEY);
229*8d741a5dSApple OSS Distributions T_ASSERT_TRUE(found, "Found " HARDENED_RUNTIME_KEY " in apple array");
230*8d741a5dSApple OSS Distributions T_ASSERT_EQ(apple_array_val, mask_val, "Bitmask value matches");
231*8d741a5dSApple OSS Distributions
232*8d741a5dSApple OSS Distributions /* These are the entitlements on the HR2 binary */
233*8d741a5dSApple OSS Distributions mask_val = BrowserGPUEntitlementMask | BrowserNetworkEntitlementMask;
234*8d741a5dSApple OSS Distributions apple_array = get_apple_array(&num_array_entries, "tools/print_apple_array_HR2");
235*8d741a5dSApple OSS Distributions found = get_apple_array_key(apple_array, num_array_entries, &apple_array_val, HARDENED_RUNTIME_KEY);
236*8d741a5dSApple OSS Distributions T_ASSERT_TRUE(found, "Found " HARDENED_RUNTIME_KEY " in apple array");
237*8d741a5dSApple OSS Distributions T_ASSERT_EQ(apple_array_val, mask_val, "Bitmask value matches");
238*8d741a5dSApple OSS Distributions free(apple_array);
239*8d741a5dSApple OSS Distributions }
240*8d741a5dSApple OSS Distributions
241*8d741a5dSApple OSS Distributions
242*8d741a5dSApple OSS Distributions T_DECL(libmalloc_hardened_binary_absent,
243*8d741a5dSApple OSS Distributions "hardened binary flags do not show up in apple array for normal third party processes",
244*8d741a5dSApple OSS Distributions T_META_ASROOT(false))
245*8d741a5dSApple OSS Distributions {
246*8d741a5dSApple OSS Distributions uint64_t new_val, apple_array_val = 0;
247*8d741a5dSApple OSS Distributions size_t num_array_entries = 0;
248*8d741a5dSApple OSS Distributions char **apple_array;
249*8d741a5dSApple OSS Distributions bool found = false;
250*8d741a5dSApple OSS Distributions apple_array = get_apple_array(&num_array_entries, NULL); // todo apple_array_3p?
251*8d741a5dSApple OSS Distributions found = get_apple_array_key(apple_array, num_array_entries, &apple_array_val, HARDENED_RUNTIME_KEY);
252*8d741a5dSApple OSS Distributions T_ASSERT_TRUE(!found, "Did not find " HARDENED_RUNTIME_KEY " in apple array");
253*8d741a5dSApple OSS Distributions free(apple_array);
254*8d741a5dSApple OSS Distributions }
255*8d741a5dSApple OSS Distributions
256*8d741a5dSApple OSS Distributions T_DECL(libmalloc_experiment,
257*8d741a5dSApple OSS Distributions "libmalloc experiment flags show up in apple array if we're doing an experiment",
258*8d741a5dSApple OSS Distributions T_META_ASROOT(false))
259*8d741a5dSApple OSS Distributions {
260*8d741a5dSApple OSS Distributions uint64_t new_val, apple_array_val = 0;
261*8d741a5dSApple OSS Distributions size_t num_array_entries = 0;
262*8d741a5dSApple OSS Distributions char **apple_array;
263*8d741a5dSApple OSS Distributions bool found = false;
264*8d741a5dSApple OSS Distributions
265*8d741a5dSApple OSS Distributions if (running_as_root()) {
266*8d741a5dSApple OSS Distributions drop_priv();
267*8d741a5dSApple OSS Distributions }
268*8d741a5dSApple OSS Distributions new_val = (1ULL << 63) - 1;
269*8d741a5dSApple OSS Distributions set_libmalloc_experiment(new_val);
270*8d741a5dSApple OSS Distributions
271*8d741a5dSApple OSS Distributions apple_array = get_apple_array(&num_array_entries, NULL);
272*8d741a5dSApple OSS Distributions found = get_apple_array_key(apple_array, num_array_entries, &apple_array_val, LIBMALLOC_EXPERIMENT_FACTORS_KEY);
273*8d741a5dSApple OSS Distributions T_ASSERT_TRUE(found, "Found " LIBMALLOC_EXPERIMENT_FACTORS_KEY " in apple array");
274*8d741a5dSApple OSS Distributions T_ASSERT_EQ(apple_array_val, new_val, "Experiment value matches");
275*8d741a5dSApple OSS Distributions free(apple_array);
276*8d741a5dSApple OSS Distributions }
277*8d741a5dSApple OSS Distributions
278*8d741a5dSApple OSS Distributions T_DECL(libmalloc_experiment_not_in_array,
279*8d741a5dSApple OSS Distributions "libmalloc experiment flags do not show up in apple array if we're not doing an experiment",
280*8d741a5dSApple OSS Distributions T_META_ASROOT(false))
281*8d741a5dSApple OSS Distributions {
282*8d741a5dSApple OSS Distributions size_t num_array_entries = 0;
283*8d741a5dSApple OSS Distributions char **apple_array;
284*8d741a5dSApple OSS Distributions bool found = false;
285*8d741a5dSApple OSS Distributions
286*8d741a5dSApple OSS Distributions if (running_as_root()) {
287*8d741a5dSApple OSS Distributions drop_priv();
288*8d741a5dSApple OSS Distributions }
289*8d741a5dSApple OSS Distributions set_libmalloc_experiment(0);
290*8d741a5dSApple OSS Distributions
291*8d741a5dSApple OSS Distributions apple_array = get_apple_array(&num_array_entries, NULL);
292*8d741a5dSApple OSS Distributions found = get_apple_array_key(apple_array, num_array_entries, NULL, LIBMALLOC_EXPERIMENT_FACTORS_KEY);
293*8d741a5dSApple OSS Distributions T_ASSERT_TRUE(!found, "Did not find " LIBMALLOC_EXPERIMENT_FACTORS_KEY " in apple array");
294*8d741a5dSApple OSS Distributions free(apple_array);
295*8d741a5dSApple OSS Distributions }
296*8d741a5dSApple OSS Distributions #endif /* ENTITLED */
297