xref: /titanic_51/usr/src/uts/common/syscall/auditsys.c (revision cb49a9fdee4ad0ac3977d55d26fad4f61caad085)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <sys/systm.h>
27 #include <sys/errno.h>
28 #include <sys/policy.h>
29 
30 #include <c2/audit.h>
31 #include <c2/audit_kernel.h>
32 #include <c2/audit_record.h>
33 
34 #define	CLEAR_VAL -1
35 
36 extern kmutex_t pidlock;
37 
38 int audit_policy; /* global audit policies in force */
39 
40 
41 /*ARGSUSED1*/
42 int
43 auditsys(struct auditcalls *uap, rval_t *rvp)
44 {
45 	int err;
46 	int result = 0;
47 
48 	if (audit_active == C2AUDIT_DISABLED)
49 		return (ENOTSUP);
50 
51 	switch (uap->code) {
52 	case BSM_GETAUID:
53 		result = getauid((caddr_t)uap->a1);
54 		break;
55 	case BSM_SETAUID:
56 		result = setauid((caddr_t)uap->a1);
57 		break;
58 	case BSM_GETAUDIT:
59 		result = getaudit((caddr_t)uap->a1);
60 		break;
61 	case BSM_GETAUDIT_ADDR:
62 		result = getaudit_addr((caddr_t)uap->a1, (int)uap->a2);
63 		break;
64 	case BSM_SETAUDIT:
65 		result = setaudit((caddr_t)uap->a1);
66 		break;
67 	case BSM_SETAUDIT_ADDR:
68 		result = setaudit_addr((caddr_t)uap->a1, (int)uap->a2);
69 		break;
70 	case BSM_AUDITCTL:
71 		result = auditctl((int)uap->a1, (caddr_t)uap->a2, (int)uap->a3);
72 		break;
73 	case BSM_AUDIT:
74 		if (audit_active == C2AUDIT_UNLOADED)
75 			return (0);
76 		result = audit((caddr_t)uap->a1, (int)uap->a2);
77 		break;
78 	case BSM_AUDITDOOR:
79 		if (audit_active == C2AUDIT_LOADED) {
80 			result = auditdoor((int)uap->a1);
81 			break;
82 		}
83 	default:
84 		if (audit_active == C2AUDIT_LOADED) {
85 			result = EINVAL;
86 			break;
87 		}
88 		/* Return a different error when not privileged */
89 		err = secpolicy_audit_config(CRED());
90 		if (err == 0)
91 			return (EINVAL);
92 		else
93 			return (err);
94 	}
95 	rvp->r_vals = result;
96 	return (result);
97 }
98 
99 /*
100  * Return the audit user ID for the current process.  Currently only
101  * the privileged processes may see the audit id.  That may change.
102  * If copyout is unsucessful return EFAULT.
103  */
104 int
105 getauid(caddr_t auid_p)
106 {
107 	const auditinfo_addr_t	*ainfo;
108 
109 	if (secpolicy_audit_getattr(CRED()) != 0)
110 		return (EPERM);
111 
112 	ainfo = crgetauinfo(CRED());
113 	if (ainfo == NULL)
114 		return (EINVAL);
115 
116 	if (copyout(&ainfo->ai_auid, auid_p, sizeof (au_id_t)))
117 		return (EFAULT);
118 
119 	return (0);
120 }
121 
122 /*
123  * Set the audit userid, for a process.  This can only be changed by
124  * privileged processes.  The audit userid is inherited across forks & execs.
125  * Passed in is a pointer to the au_id_t; if copyin unsuccessful return EFAULT.
126  */
127 int
128 setauid(caddr_t auid_p)
129 {
130 	proc_t *p;
131 	au_id_t	auid;
132 	cred_t *newcred;
133 	auditinfo_addr_t *auinfo;
134 
135 	if (secpolicy_audit_config(CRED()) != 0)
136 		return (EPERM);
137 
138 	if (copyin(auid_p, &auid, sizeof (au_id_t))) {
139 		return (EFAULT);
140 	}
141 
142 	newcred = cralloc();
143 	if ((auinfo = crgetauinfo_modifiable(newcred)) == NULL) {
144 		crfree(newcred);
145 		return (EINVAL);
146 	}
147 
148 	/* grab p_crlock and switch to new cred */
149 	p = curproc;
150 	mutex_enter(&p->p_crlock);
151 	crcopy_to(p->p_cred, newcred);
152 	p->p_cred = newcred;
153 
154 	auinfo->ai_auid = auid;			/* update the auid */
155 
156 	/* unlock and broadcast the cred changes */
157 	mutex_exit(&p->p_crlock);
158 	crset(p, newcred);
159 
160 	return (0);
161 }
162 
163 /*
164  * Get the audit state information from the current process.
165  * Return EFAULT if copyout fails.
166  */
167 int
168 getaudit(caddr_t info_p)
169 {
170 	STRUCT_DECL(auditinfo, info);
171 	const auditinfo_addr_t	*ainfo;
172 	model_t	model;
173 
174 	if (secpolicy_audit_getattr(CRED()) != 0)
175 		return (EPERM);
176 
177 	model = get_udatamodel();
178 	STRUCT_INIT(info, model);
179 
180 	ainfo = crgetauinfo(CRED());
181 	if (ainfo == NULL)
182 		return (EINVAL);
183 
184 	/* trying to read a process with an IPv6 address? */
185 	if (ainfo->ai_termid.at_type == AU_IPv6)
186 		return (EOVERFLOW);
187 
188 	STRUCT_FSET(info, ai_auid, ainfo->ai_auid);
189 	STRUCT_FSET(info, ai_mask, ainfo->ai_mask);
190 #ifdef _LP64
191 	if (model == DATAMODEL_ILP32) {
192 		dev32_t dev;
193 		/* convert internal 64 bit form to 32 bit version */
194 		if (cmpldev(&dev, ainfo->ai_termid.at_port) == 0) {
195 			return (EOVERFLOW);
196 		}
197 		STRUCT_FSET(info, ai_termid.port, dev);
198 	} else
199 		STRUCT_FSET(info, ai_termid.port, ainfo->ai_termid.at_port);
200 #else
201 	STRUCT_FSET(info, ai_termid.port, ainfo->ai_termid.at_port);
202 #endif
203 	STRUCT_FSET(info, ai_termid.machine, ainfo->ai_termid.at_addr[0]);
204 	STRUCT_FSET(info, ai_asid, ainfo->ai_asid);
205 
206 	if (copyout(STRUCT_BUF(info), info_p, STRUCT_SIZE(info)))
207 		return (EFAULT);
208 
209 	return (0);
210 }
211 
212 /*
213  * Get the audit state information from the current process.
214  * Return EFAULT if copyout fails.
215  */
216 int
217 getaudit_addr(caddr_t info_p, int len)
218 {
219 	STRUCT_DECL(auditinfo_addr, info);
220 	const auditinfo_addr_t	*ainfo;
221 	model_t	model;
222 
223 	if (secpolicy_audit_getattr(CRED()) != 0)
224 		return (EPERM);
225 
226 	model = get_udatamodel();
227 	STRUCT_INIT(info, model);
228 
229 	if (len < STRUCT_SIZE(info))
230 		return (EOVERFLOW);
231 
232 	ainfo = crgetauinfo(CRED());
233 
234 	if (ainfo == NULL)
235 		return (EINVAL);
236 
237 	STRUCT_FSET(info, ai_auid, ainfo->ai_auid);
238 	STRUCT_FSET(info, ai_mask, ainfo->ai_mask);
239 #ifdef _LP64
240 	if (model == DATAMODEL_ILP32) {
241 		dev32_t dev;
242 		/* convert internal 64 bit form to 32 bit version */
243 		if (cmpldev(&dev, ainfo->ai_termid.at_port) == 0) {
244 			return (EOVERFLOW);
245 		}
246 		STRUCT_FSET(info, ai_termid.at_port, dev);
247 	} else
248 		STRUCT_FSET(info, ai_termid.at_port, ainfo->ai_termid.at_port);
249 #else
250 	STRUCT_FSET(info, ai_termid.at_port, ainfo->ai_termid.at_port);
251 #endif
252 	STRUCT_FSET(info, ai_termid.at_type, ainfo->ai_termid.at_type);
253 	STRUCT_FSET(info, ai_termid.at_addr[0], ainfo->ai_termid.at_addr[0]);
254 	STRUCT_FSET(info, ai_termid.at_addr[1], ainfo->ai_termid.at_addr[1]);
255 	STRUCT_FSET(info, ai_termid.at_addr[2], ainfo->ai_termid.at_addr[2]);
256 	STRUCT_FSET(info, ai_termid.at_addr[3], ainfo->ai_termid.at_addr[3]);
257 	STRUCT_FSET(info, ai_asid, ainfo->ai_asid);
258 
259 	if (copyout(STRUCT_BUF(info), info_p, STRUCT_SIZE(info)))
260 		return (EFAULT);
261 
262 	return (0);
263 }
264 
265 /*
266  * Set the audit state information for the current process.
267  * Return EFAULT if copyout fails.
268  */
269 int
270 setaudit(caddr_t info_p)
271 {
272 	STRUCT_DECL(auditinfo, info);
273 	proc_t *p;
274 	cred_t	*newcred;
275 	model_t	model;
276 	auditinfo_addr_t *ainfo;
277 
278 	if (secpolicy_audit_config(CRED()) != 0)
279 		return (EPERM);
280 
281 	model = get_udatamodel();
282 	STRUCT_INIT(info, model);
283 
284 	if (copyin(info_p, STRUCT_BUF(info), STRUCT_SIZE(info)))
285 		return (EFAULT);
286 
287 	newcred = cralloc();
288 	if ((ainfo = crgetauinfo_modifiable(newcred)) == NULL) {
289 		crfree(newcred);
290 		return (EINVAL);
291 	}
292 
293 	/* grab p_crlock and switch to new cred */
294 	p = curproc;
295 	mutex_enter(&p->p_crlock);
296 	crcopy_to(p->p_cred, newcred);
297 	p->p_cred = newcred;
298 
299 	/* Set audit mask, id, termid and session id as specified */
300 	ainfo->ai_auid = STRUCT_FGET(info, ai_auid);
301 #ifdef _LP64
302 	/* only convert to 64 bit if coming from a 32 bit binary */
303 	if (model == DATAMODEL_ILP32)
304 		ainfo->ai_termid.at_port =
305 		    DEVEXPL(STRUCT_FGET(info, ai_termid.port));
306 	else
307 		ainfo->ai_termid.at_port = STRUCT_FGET(info, ai_termid.port);
308 #else
309 	ainfo->ai_termid.at_port = STRUCT_FGET(info, ai_termid.port);
310 #endif
311 	ainfo->ai_termid.at_type = AU_IPv4;
312 	ainfo->ai_termid.at_addr[0] = STRUCT_FGET(info, ai_termid.machine);
313 	ainfo->ai_asid = STRUCT_FGET(info, ai_asid);
314 	ainfo->ai_mask = STRUCT_FGET(info, ai_mask);
315 
316 	/* unlock and broadcast the cred changes */
317 	mutex_exit(&p->p_crlock);
318 	crset(p, newcred);
319 
320 	return (0);
321 }
322 
323 /*
324  * Set the audit state information for the current process.
325  * Return EFAULT if copyin fails.
326  */
327 int
328 setaudit_addr(caddr_t info_p, int len)
329 {
330 	STRUCT_DECL(auditinfo_addr, info);
331 	proc_t *p;
332 	cred_t	*newcred;
333 	model_t	model;
334 	int i;
335 	int type;
336 	auditinfo_addr_t *ainfo;
337 
338 	if (secpolicy_audit_config(CRED()) != 0)
339 		return (EPERM);
340 
341 	model = get_udatamodel();
342 	STRUCT_INIT(info, model);
343 
344 	if (len < STRUCT_SIZE(info))
345 		return (EOVERFLOW);
346 
347 	if (copyin(info_p, STRUCT_BUF(info), STRUCT_SIZE(info)))
348 		return (EFAULT);
349 
350 	type = STRUCT_FGET(info, ai_termid.at_type);
351 	if ((type != AU_IPv4) && (type != AU_IPv6))
352 		return (EINVAL);
353 
354 	newcred = cralloc();
355 	if ((ainfo = crgetauinfo_modifiable(newcred)) == NULL) {
356 		crfree(newcred);
357 		return (EINVAL);
358 	}
359 
360 	/* grab p_crlock and switch to new cred */
361 	p = curproc;
362 	mutex_enter(&p->p_crlock);
363 	crcopy_to(p->p_cred, newcred);
364 	p->p_cred = newcred;
365 
366 	/* Set audit mask, id, termid and session id as specified */
367 	ainfo->ai_auid = STRUCT_FGET(info, ai_auid);
368 	ainfo->ai_mask = STRUCT_FGET(info, ai_mask);
369 #ifdef _LP64
370 	/* only convert to 64 bit if coming from a 32 bit binary */
371 	if (model == DATAMODEL_ILP32)
372 		ainfo->ai_termid.at_port =
373 		    DEVEXPL(STRUCT_FGET(info, ai_termid.at_port));
374 	else
375 		ainfo->ai_termid.at_port = STRUCT_FGET(info, ai_termid.at_port);
376 #else
377 	ainfo->ai_termid.at_port = STRUCT_FGET(info, ai_termid.at_port);
378 #endif
379 	ainfo->ai_termid.at_type = type;
380 	bzero(&ainfo->ai_termid.at_addr[0], sizeof (ainfo->ai_termid.at_addr));
381 	for (i = 0; i < (type/sizeof (int)); i++)
382 		ainfo->ai_termid.at_addr[i] =
383 		    STRUCT_FGET(info, ai_termid.at_addr[i]);
384 
385 	if (ainfo->ai_termid.at_type == AU_IPv6 &&
386 	    IN6_IS_ADDR_V4MAPPED(((in6_addr_t *)ainfo->ai_termid.at_addr))) {
387 		ainfo->ai_termid.at_type = AU_IPv4;
388 		ainfo->ai_termid.at_addr[0] = ainfo->ai_termid.at_addr[3];
389 		ainfo->ai_termid.at_addr[1] = 0;
390 		ainfo->ai_termid.at_addr[2] = 0;
391 		ainfo->ai_termid.at_addr[3] = 0;
392 	}
393 
394 	ainfo->ai_asid = STRUCT_FGET(info, ai_asid);
395 
396 	/* unlock and broadcast the cred changes */
397 	mutex_exit(&p->p_crlock);
398 	crset(p, newcred);
399 
400 	return (0);
401 }
402 
403 /*
404  * Get the global policy flag
405  */
406 static int
407 getpolicy(caddr_t data)
408 {
409 	int	policy;
410 	au_kcontext_t	*kctx = GET_KCTX_PZ;
411 
412 	policy = audit_policy | kctx->auk_policy;
413 
414 	if (copyout(&policy, data, sizeof (int)))
415 		return (EFAULT);
416 	return (0);
417 }
418 
419 /*
420  * Set the global and local policy flags
421  *
422  * The global flags only make sense from the global zone;
423  * the local flags depend on the AUDIT_PERZONE policy:
424  * if the perzone policy is set, then policy is set separately
425  * per zone, else held only in the global zone.
426  *
427  * The initial value of a local zone's policy flag is determined
428  * by the value of the global zone's flags at the time the
429  * local zone is created.
430  *
431  * While auditconfig(1M) allows setting and unsetting policies one bit
432  * at a time, the mask passed in from auditconfig() is created by a
433  * syscall to getpolicy and then modified based on the auditconfig()
434  * cmd line, so the input policy value is used to replace the existing
435  * policy.
436  */
437 static int
438 setpolicy(caddr_t data)
439 {
440 	int	policy;
441 	au_kcontext_t	*kctx;
442 
443 	if (copyin(data, &policy, sizeof (int)))
444 		return (EFAULT);
445 
446 	kctx = GET_KCTX_NGZ;
447 
448 	if (INGLOBALZONE(curproc)) {
449 		if (policy & ~(AUDIT_GLOBAL | AUDIT_LOCAL))
450 			return (EINVAL);
451 
452 		audit_policy = policy & AUDIT_GLOBAL;
453 	} else {
454 		if (!(audit_policy & AUDIT_PERZONE))
455 			return (EINVAL);
456 
457 		if (policy & ~AUDIT_LOCAL)	/* global bits are a no-no */
458 			return (EINVAL);
459 	}
460 	kctx->auk_policy = policy & AUDIT_LOCAL;
461 
462 	/*
463 	 * auk_current_vp is NULL before auditd starts (or during early
464 	 * auditd starup) or if auditd is halted; in either case,
465 	 * notification of a policy change is not needed, since auditd
466 	 * reads policy as it comes up.  The error return from au_doormsg()
467 	 * is ignored to avoid a race condition -- for example if auditd
468 	 * segv's, the audit state may be "auditing" but the door may
469 	 * be closed.  Returning an error if the door is open makes it
470 	 * impossible for Greenline to restart auditd.
471 	 */
472 	if (kctx->auk_current_vp != NULL)
473 		(void) au_doormsg(kctx, AU_DBUF_POLICY, &policy);
474 
475 	/*
476 	 * Wake up anyone who might have blocked on full audit
477 	 * partitions. audit daemons need to set AUDIT_FULL when no
478 	 * space so we can tell if we should start dropping records.
479 	 */
480 	mutex_enter(&(kctx->auk_queue.lock));
481 
482 	if ((policy & (AUDIT_CNT | AUDIT_SCNT) &&
483 	    (kctx->auk_queue.cnt >= kctx->auk_queue.hiwater)))
484 		cv_broadcast(&(kctx->auk_queue.write_cv));
485 
486 	mutex_exit(&(kctx->auk_queue.lock));
487 
488 	return (0);
489 }
490 
491 static int
492 getkmask(caddr_t data)
493 {
494 	au_kcontext_t	*kctx;
495 
496 	kctx = GET_KCTX_PZ;
497 
498 	if (copyout(&kctx->auk_info.ai_mask, data, sizeof (au_mask_t)))
499 		return (EFAULT);
500 	return (0);
501 }
502 
503 static int
504 setkmask(caddr_t data)
505 {
506 	au_mask_t	mask;
507 	au_kcontext_t	*kctx;
508 
509 	if (!(audit_policy & AUDIT_PERZONE) && !INGLOBALZONE(curproc))
510 		return (EINVAL);
511 
512 	kctx = GET_KCTX_NGZ;
513 
514 	if (copyin(data, &mask, sizeof (au_mask_t)))
515 		return (EFAULT);
516 
517 	kctx->auk_info.ai_mask = mask;
518 	return (0);
519 }
520 
521 static int
522 getkaudit(caddr_t info_p, int len)
523 {
524 	STRUCT_DECL(auditinfo_addr, info);
525 	model_t model;
526 	au_kcontext_t	*kctx = GET_KCTX_PZ;
527 
528 	model = get_udatamodel();
529 	STRUCT_INIT(info, model);
530 
531 	if (len < STRUCT_SIZE(info))
532 		return (EOVERFLOW);
533 
534 	STRUCT_FSET(info, ai_auid, kctx->auk_info.ai_auid);
535 	STRUCT_FSET(info, ai_mask, kctx->auk_info.ai_mask);
536 #ifdef _LP64
537 	if (model == DATAMODEL_ILP32) {
538 		dev32_t dev;
539 		/* convert internal 64 bit form to 32 bit version */
540 		if (cmpldev(&dev, kctx->auk_info.ai_termid.at_port) == 0) {
541 			return (EOVERFLOW);
542 		}
543 		STRUCT_FSET(info, ai_termid.at_port, dev);
544 	} else {
545 		STRUCT_FSET(info, ai_termid.at_port,
546 		    kctx->auk_info.ai_termid.at_port);
547 	}
548 #else
549 	STRUCT_FSET(info, ai_termid.at_port,
550 	    kctx->auk_info.ai_termid.at_port);
551 #endif
552 	STRUCT_FSET(info, ai_termid.at_type,
553 	    kctx->auk_info.ai_termid.at_type);
554 	STRUCT_FSET(info, ai_termid.at_addr[0],
555 	    kctx->auk_info.ai_termid.at_addr[0]);
556 	STRUCT_FSET(info, ai_termid.at_addr[1],
557 	    kctx->auk_info.ai_termid.at_addr[1]);
558 	STRUCT_FSET(info, ai_termid.at_addr[2],
559 	    kctx->auk_info.ai_termid.at_addr[2]);
560 	STRUCT_FSET(info, ai_termid.at_addr[3],
561 	    kctx->auk_info.ai_termid.at_addr[3]);
562 	STRUCT_FSET(info, ai_asid, kctx->auk_info.ai_asid);
563 
564 	if (copyout(STRUCT_BUF(info), info_p, STRUCT_SIZE(info)))
565 		return (EFAULT);
566 
567 	return (0);
568 }
569 
570 /*
571  * the host address for AUDIT_PERZONE == 0 is that of the global
572  * zone and for local zones it is of the current zone.
573  */
574 static int
575 setkaudit(caddr_t info_p, int len)
576 {
577 	STRUCT_DECL(auditinfo_addr, info);
578 	model_t model;
579 	au_kcontext_t	*kctx;
580 
581 	if (!(audit_policy & AUDIT_PERZONE) && !INGLOBALZONE(curproc))
582 		return (EINVAL);
583 
584 	kctx = GET_KCTX_NGZ;
585 
586 	model = get_udatamodel();
587 	STRUCT_INIT(info, model);
588 
589 	if (len < STRUCT_SIZE(info))
590 		return (EOVERFLOW);
591 
592 	if (copyin(info_p, STRUCT_BUF(info), STRUCT_SIZE(info)))
593 		return (EFAULT);
594 
595 	if ((STRUCT_FGET(info, ai_termid.at_type) != AU_IPv4) &&
596 	    (STRUCT_FGET(info, ai_termid.at_type) != AU_IPv6))
597 		return (EINVAL);
598 
599 	/* Set audit mask, termid and session id as specified */
600 	kctx->auk_info.ai_auid = STRUCT_FGET(info, ai_auid);
601 	kctx->auk_info.ai_mask = STRUCT_FGET(info, ai_mask);
602 #ifdef _LP64
603 	/* only convert to 64 bit if coming from a 32 bit binary */
604 	if (model == DATAMODEL_ILP32)
605 		kctx->auk_info.ai_termid.at_port =
606 		    DEVEXPL(STRUCT_FGET(info, ai_termid.at_port));
607 	else
608 		kctx->auk_info.ai_termid.at_port =
609 		    STRUCT_FGET(info, ai_termid.at_port);
610 #else
611 	kctx->auk_info.ai_termid.at_port = STRUCT_FGET(info, ai_termid.at_port);
612 #endif
613 	kctx->auk_info.ai_termid.at_type = STRUCT_FGET(info, ai_termid.at_type);
614 	bzero(&kctx->auk_info.ai_termid.at_addr[0],
615 	    sizeof (kctx->auk_info.ai_termid.at_addr));
616 	kctx->auk_info.ai_termid.at_addr[0] =
617 	    STRUCT_FGET(info, ai_termid.at_addr[0]);
618 	kctx->auk_info.ai_termid.at_addr[1] =
619 	    STRUCT_FGET(info, ai_termid.at_addr[1]);
620 	kctx->auk_info.ai_termid.at_addr[2] =
621 	    STRUCT_FGET(info, ai_termid.at_addr[2]);
622 	kctx->auk_info.ai_termid.at_addr[3] =
623 	    STRUCT_FGET(info, ai_termid.at_addr[3]);
624 	kctx->auk_info.ai_asid = STRUCT_FGET(info, ai_asid);
625 
626 	if (kctx->auk_info.ai_termid.at_type == AU_IPv6 &&
627 	    IN6_IS_ADDR_V4MAPPED(
628 	    ((in6_addr_t *)kctx->auk_info.ai_termid.at_addr))) {
629 		kctx->auk_info.ai_termid.at_type = AU_IPv4;
630 		kctx->auk_info.ai_termid.at_addr[0] =
631 		    kctx->auk_info.ai_termid.at_addr[3];
632 		kctx->auk_info.ai_termid.at_addr[1] = 0;
633 		kctx->auk_info.ai_termid.at_addr[2] = 0;
634 		kctx->auk_info.ai_termid.at_addr[3] = 0;
635 	}
636 	if (kctx->auk_info.ai_termid.at_type == AU_IPv6)
637 		kctx->auk_hostaddr_valid = IN6_IS_ADDR_UNSPECIFIED(
638 		    (in6_addr_t *)kctx->auk_info.ai_termid.at_addr) ? 0 : 1;
639 	else
640 		kctx->auk_hostaddr_valid =
641 		    (kctx->auk_info.ai_termid.at_addr[0] ==
642 		    htonl(INADDR_ANY)) ? 0 : 1;
643 
644 	return (0);
645 }
646 
647 static int
648 getqctrl(caddr_t data)
649 {
650 	au_kcontext_t	*kctx = GET_KCTX_PZ;
651 	STRUCT_DECL(au_qctrl, qctrl);
652 	STRUCT_INIT(qctrl, get_udatamodel());
653 
654 	mutex_enter(&(kctx->auk_queue.lock));
655 	STRUCT_FSET(qctrl, aq_hiwater, kctx->auk_queue.hiwater);
656 	STRUCT_FSET(qctrl, aq_lowater, kctx->auk_queue.lowater);
657 	STRUCT_FSET(qctrl, aq_bufsz, kctx->auk_queue.bufsz);
658 	STRUCT_FSET(qctrl, aq_delay, kctx->auk_queue.delay);
659 	mutex_exit(&(kctx->auk_queue.lock));
660 
661 	if (copyout(STRUCT_BUF(qctrl), data, STRUCT_SIZE(qctrl)))
662 		return (EFAULT);
663 
664 	return (0);
665 }
666 
667 static int
668 setqctrl(caddr_t data)
669 {
670 	au_kcontext_t	*kctx;
671 	struct au_qctrl qctrl_tmp;
672 	STRUCT_DECL(au_qctrl, qctrl);
673 	STRUCT_INIT(qctrl, get_udatamodel());
674 
675 	if (!(audit_policy & AUDIT_PERZONE) && !INGLOBALZONE(curproc))
676 		return (EINVAL);
677 	kctx = GET_KCTX_NGZ;
678 
679 	if (copyin(data, STRUCT_BUF(qctrl), STRUCT_SIZE(qctrl)))
680 		return (EFAULT);
681 
682 	qctrl_tmp.aq_hiwater = (size_t)STRUCT_FGET(qctrl, aq_hiwater);
683 	qctrl_tmp.aq_lowater = (size_t)STRUCT_FGET(qctrl, aq_lowater);
684 	qctrl_tmp.aq_bufsz = (size_t)STRUCT_FGET(qctrl, aq_bufsz);
685 	qctrl_tmp.aq_delay = (clock_t)STRUCT_FGET(qctrl, aq_delay);
686 
687 	/* enforce sane values */
688 
689 	if (qctrl_tmp.aq_hiwater <= qctrl_tmp.aq_lowater)
690 		return (EINVAL);
691 
692 	if (qctrl_tmp.aq_hiwater < AQ_LOWATER)
693 		return (EINVAL);
694 
695 	if (qctrl_tmp.aq_hiwater > AQ_MAXHIGH)
696 		return (EINVAL);
697 
698 	if (qctrl_tmp.aq_bufsz < AQ_BUFSZ)
699 		return (EINVAL);
700 
701 	if (qctrl_tmp.aq_bufsz > AQ_MAXBUFSZ)
702 		return (EINVAL);
703 
704 	if (qctrl_tmp.aq_delay == 0)
705 		return (EINVAL);
706 
707 	if (qctrl_tmp.aq_delay > AQ_MAXDELAY)
708 		return (EINVAL);
709 
710 	/* update everything at once so things are consistant */
711 	mutex_enter(&(kctx->auk_queue.lock));
712 	kctx->auk_queue.hiwater = qctrl_tmp.aq_hiwater;
713 	kctx->auk_queue.lowater = qctrl_tmp.aq_lowater;
714 	kctx->auk_queue.bufsz = qctrl_tmp.aq_bufsz;
715 	kctx->auk_queue.delay = qctrl_tmp.aq_delay;
716 
717 	if (kctx->auk_queue.rd_block &&
718 	    kctx->auk_queue.cnt > kctx->auk_queue.lowater)
719 		cv_broadcast(&(kctx->auk_queue.read_cv));
720 
721 	if (kctx->auk_queue.wt_block &&
722 	    kctx->auk_queue.cnt < kctx->auk_queue.hiwater)
723 		cv_broadcast(&(kctx->auk_queue.write_cv));
724 
725 	mutex_exit(&(kctx->auk_queue.lock));
726 
727 	return (0);
728 }
729 
730 static int
731 getcwd(caddr_t data, int length)
732 {
733 	struct p_audit_data	*pad;
734 	struct audit_path	*app;
735 	int	pathlen;
736 
737 	pad = P2A(curproc);
738 	ASSERT(pad != NULL);
739 
740 	mutex_enter(&(pad->pad_lock));
741 	app = pad->pad_cwd;
742 	au_pathhold(app);
743 	mutex_exit(&(pad->pad_lock));
744 
745 	pathlen = app->audp_sect[1] - app->audp_sect[0];
746 	if (pathlen > length) {
747 		au_pathrele(app);
748 		return (E2BIG);
749 	}
750 
751 	if (copyout(app->audp_sect[0], data, pathlen)) {
752 		au_pathrele(app);
753 		return (EFAULT);
754 	}
755 
756 	au_pathrele(app);
757 	return (0);
758 }
759 
760 static int
761 getcar(caddr_t data, int length)
762 {
763 	struct p_audit_data	*pad;
764 	struct audit_path	*app;
765 	int	pathlen;
766 
767 	pad = P2A(curproc);
768 	ASSERT(pad != NULL);
769 
770 	mutex_enter(&(pad->pad_lock));
771 	app = pad->pad_root;
772 	au_pathhold(app);
773 	mutex_exit(&(pad->pad_lock));
774 
775 	pathlen = app->audp_sect[1] - app->audp_sect[0];
776 	if (pathlen > length) {
777 		au_pathrele(app);
778 		return (E2BIG);
779 	}
780 
781 	if (copyout(app->audp_sect[0], data, pathlen)) {
782 		au_pathrele(app);
783 		return (EFAULT);
784 	}
785 
786 	au_pathrele(app);
787 	return (0);
788 }
789 
790 static int
791 getstat(caddr_t data)
792 {
793 	au_kcontext_t	*kctx = GET_KCTX_PZ;
794 
795 	membar_consumer();
796 
797 	if (copyout((caddr_t)&(kctx->auk_statistics), data, sizeof (au_stat_t)))
798 		return (EFAULT);
799 	return (0);
800 }
801 
802 static int
803 setstat(caddr_t data)
804 {
805 	au_kcontext_t *kctx = GET_KCTX_PZ;
806 	au_stat_t au_stat;
807 
808 	if (!(audit_policy & AUDIT_PERZONE) && !INGLOBALZONE(curproc))
809 		return (EINVAL);
810 
811 	if (copyin(data, &au_stat, sizeof (au_stat_t)))
812 		return (EFAULT);
813 
814 	if (au_stat.as_generated == CLEAR_VAL)
815 		kctx->auk_statistics.as_generated = 0;
816 	if (au_stat.as_nonattrib == CLEAR_VAL)
817 		kctx->auk_statistics.as_nonattrib = 0;
818 	if (au_stat.as_kernel == CLEAR_VAL)
819 		kctx->auk_statistics.as_kernel = 0;
820 	if (au_stat.as_audit == CLEAR_VAL)
821 		kctx->auk_statistics.as_audit = 0;
822 	if (au_stat.as_auditctl == CLEAR_VAL)
823 		kctx->auk_statistics.as_auditctl = 0;
824 	if (au_stat.as_enqueue == CLEAR_VAL)
825 		kctx->auk_statistics.as_enqueue = 0;
826 	if (au_stat.as_written == CLEAR_VAL)
827 		kctx->auk_statistics.as_written = 0;
828 	if (au_stat.as_wblocked == CLEAR_VAL)
829 		kctx->auk_statistics.as_wblocked = 0;
830 	if (au_stat.as_rblocked == CLEAR_VAL)
831 		kctx->auk_statistics.as_rblocked = 0;
832 	if (au_stat.as_dropped == CLEAR_VAL)
833 		kctx->auk_statistics.as_dropped = 0;
834 	if (au_stat.as_totalsize == CLEAR_VAL)
835 		kctx->auk_statistics.as_totalsize = 0;
836 
837 	membar_producer();
838 
839 	return (0);
840 
841 }
842 
843 static int
844 setumask(caddr_t data)
845 {
846 	STRUCT_DECL(auditinfo, user_info);
847 	struct proc *p;
848 	const auditinfo_addr_t	*ainfo;
849 	model_t	model;
850 
851 	/* setumask not applicable in non-global zones without perzone policy */
852 	if (!(audit_policy & AUDIT_PERZONE) && (!INGLOBALZONE(curproc)))
853 		return (EINVAL);
854 
855 	model = get_udatamodel();
856 	STRUCT_INIT(user_info, model);
857 
858 	if (copyin(data, STRUCT_BUF(user_info), STRUCT_SIZE(user_info)))
859 		return (EFAULT);
860 
861 	mutex_enter(&pidlock);	/* lock the process queue against updates */
862 	for (p = practive; p != NULL; p = p->p_next) {
863 		cred_t	*cr;
864 
865 		/* if in non-global zone only modify processes in same zone */
866 		if (!HASZONEACCESS(curproc, p->p_zone->zone_id))
867 			continue;
868 
869 		mutex_enter(&p->p_lock);	/* so process doesn't go away */
870 
871 		/* skip system processes and ones being created or going away */
872 		if (p->p_stat == SIDL || p->p_stat == SZOMB ||
873 		    (p->p_flag & (SSYS | SEXITING | SEXITLWPS))) {
874 			mutex_exit(&p->p_lock);
875 			continue;
876 		}
877 
878 		mutex_enter(&p->p_crlock);
879 		crhold(cr = p->p_cred);
880 		mutex_exit(&p->p_crlock);
881 		ainfo = crgetauinfo(cr);
882 		if (ainfo == NULL) {
883 			mutex_exit(&p->p_lock);
884 			crfree(cr);
885 			continue;
886 		}
887 
888 		if (ainfo->ai_auid == STRUCT_FGET(user_info, ai_auid)) {
889 			au_mask_t	mask;
890 			int		err;
891 
892 			/*
893 			 * Here's a process which matches the specified auid.
894 			 * If its mask doesn't already match the new mask,
895 			 * save the new mask in the pad, to be picked up
896 			 * next syscall.
897 			 */
898 			mask = STRUCT_FGET(user_info, ai_mask);
899 			err = bcmp(&mask, &ainfo->ai_mask, sizeof (au_mask_t));
900 			crfree(cr);
901 			if (err != 0) {
902 				struct p_audit_data *pad = P2A(p);
903 				ASSERT(pad != NULL);
904 
905 				mutex_enter(&(pad->pad_lock));
906 				pad->pad_flags |= PAD_SETMASK;
907 				pad->pad_newmask = mask;
908 				mutex_exit(&(pad->pad_lock));
909 
910 				/*
911 				 * No need to call set_proc_pre_sys(), since
912 				 * t_pre_sys is ALWAYS on when audit is
913 				 * enabled...due to syscall auditing.
914 				 */
915 			}
916 		} else {
917 			crfree(cr);
918 		}
919 		mutex_exit(&p->p_lock);
920 	}
921 	mutex_exit(&pidlock);
922 
923 	return (0);
924 }
925 
926 static int
927 setsmask(caddr_t data)
928 {
929 	STRUCT_DECL(auditinfo, user_info);
930 	struct proc *p;
931 	const auditinfo_addr_t	*ainfo;
932 	model_t	model;
933 
934 	/* setsmask not applicable in non-global zones without perzone policy */
935 	if (!(audit_policy & AUDIT_PERZONE) && (!INGLOBALZONE(curproc)))
936 		return (EINVAL);
937 
938 	model = get_udatamodel();
939 	STRUCT_INIT(user_info, model);
940 
941 	if (copyin(data, STRUCT_BUF(user_info), STRUCT_SIZE(user_info)))
942 		return (EFAULT);
943 
944 	mutex_enter(&pidlock);	/* lock the process queue against updates */
945 	for (p = practive; p != NULL; p = p->p_next) {
946 		cred_t	*cr;
947 
948 		/* if in non-global zone only modify processes in same zone */
949 		if (!HASZONEACCESS(curproc, p->p_zone->zone_id))
950 			continue;
951 
952 		mutex_enter(&p->p_lock);	/* so process doesn't go away */
953 
954 		/* skip system processes and ones being created or going away */
955 		if (p->p_stat == SIDL || p->p_stat == SZOMB ||
956 		    (p->p_flag & (SSYS | SEXITING | SEXITLWPS))) {
957 			mutex_exit(&p->p_lock);
958 			continue;
959 		}
960 
961 		mutex_enter(&p->p_crlock);
962 		crhold(cr = p->p_cred);
963 		mutex_exit(&p->p_crlock);
964 		ainfo = crgetauinfo(cr);
965 		if (ainfo == NULL) {
966 			mutex_exit(&p->p_lock);
967 			crfree(cr);
968 			continue;
969 		}
970 
971 		if (ainfo->ai_asid == STRUCT_FGET(user_info, ai_asid)) {
972 			au_mask_t	mask;
973 			int		err;
974 
975 			/*
976 			 * Here's a process which matches the specified asid.
977 			 * If its mask doesn't already match the new mask,
978 			 * save the new mask in the pad, to be picked up
979 			 * next syscall.
980 			 */
981 			mask = STRUCT_FGET(user_info, ai_mask);
982 			err = bcmp(&mask, &ainfo->ai_mask, sizeof (au_mask_t));
983 			crfree(cr);
984 			if (err != 0) {
985 				struct p_audit_data *pad = P2A(p);
986 				ASSERT(pad != NULL);
987 
988 				mutex_enter(&(pad->pad_lock));
989 				pad->pad_flags |= PAD_SETMASK;
990 				pad->pad_newmask = mask;
991 				mutex_exit(&(pad->pad_lock));
992 
993 				/*
994 				 * No need to call set_proc_pre_sys(), since
995 				 * t_pre_sys is ALWAYS on when audit is
996 				 * enabled...due to syscall auditing.
997 				 */
998 			}
999 		} else {
1000 			crfree(cr);
1001 		}
1002 		mutex_exit(&p->p_lock);
1003 	}
1004 	mutex_exit(&pidlock);
1005 
1006 	return (0);
1007 }
1008 
1009 /*
1010  * Get the current audit state of the system
1011  */
1012 static int
1013 getcond(caddr_t data)
1014 {
1015 	au_kcontext_t *kctx = GET_KCTX_PZ;
1016 
1017 	if (copyout(&(kctx->auk_auditstate), data, sizeof (int)))
1018 		return (EFAULT);
1019 
1020 	return (0);
1021 }
1022 
1023 /*
1024  * Set the current audit state of the system to on (AUC_AUDITING) or
1025  * off (AUC_NOAUDIT).
1026  */
1027 /* ARGSUSED */
1028 static int
1029 setcond(caddr_t data)
1030 {
1031 	int auditstate;
1032 	au_kcontext_t *kctx;
1033 
1034 	if (!(audit_policy & AUDIT_PERZONE) && (!INGLOBALZONE(curproc)))
1035 		return (EINVAL);
1036 
1037 	kctx = GET_KCTX_NGZ;
1038 
1039 	if (copyin(data, &auditstate, sizeof (int)))
1040 		return (EFAULT);
1041 
1042 	switch (auditstate) {
1043 	case AUC_AUDITING:		/* Turn auditing on */
1044 		if (audit_active == C2AUDIT_UNLOADED)
1045 			audit_init_module();
1046 		kctx->auk_auditstate = AUC_AUDITING;
1047 		if (!(audit_policy & AUDIT_PERZONE) && INGLOBALZONE(curproc))
1048 			set_all_zone_usr_proc_sys(ALL_ZONES);
1049 		else
1050 			set_all_zone_usr_proc_sys(curproc->p_zone->zone_id);
1051 		break;
1052 
1053 	case AUC_NOAUDIT:		/* Turn auditing off */
1054 		if (kctx->auk_auditstate == AUC_NOAUDIT)
1055 			break;
1056 		kctx->auk_auditstate = AUC_NOAUDIT;
1057 
1058 		/* clear out the audit queue */
1059 
1060 		mutex_enter(&(kctx->auk_queue.lock));
1061 		if (kctx->auk_queue.wt_block)
1062 			cv_broadcast(&(kctx->auk_queue.write_cv));
1063 
1064 		/* unblock au_output_thread */
1065 		cv_broadcast(&(kctx->auk_queue.read_cv));
1066 
1067 		mutex_exit(&(kctx->auk_queue.lock));
1068 		break;
1069 
1070 	default:
1071 		return (EINVAL);
1072 	}
1073 
1074 	return (0);
1075 }
1076 
1077 static int
1078 getclass(caddr_t data)
1079 {
1080 	au_evclass_map_t event;
1081 	au_kcontext_t	*kctx = GET_KCTX_PZ;
1082 
1083 	if (copyin(data, &event, sizeof (au_evclass_map_t)))
1084 		return (EFAULT);
1085 
1086 	if (event.ec_number > MAX_KEVENTS)
1087 		return (EINVAL);
1088 
1089 	event.ec_class = kctx->auk_ets[event.ec_number];
1090 
1091 	if (copyout(&event, data, sizeof (au_evclass_map_t)))
1092 		return (EFAULT);
1093 
1094 	return (0);
1095 }
1096 
1097 static int
1098 setclass(caddr_t data)
1099 {
1100 	au_evclass_map_t event;
1101 	au_kcontext_t	*kctx;
1102 
1103 	if (!(audit_policy & AUDIT_PERZONE) && !INGLOBALZONE(curproc))
1104 		return (EINVAL);
1105 
1106 	kctx = GET_KCTX_NGZ;
1107 
1108 	if (copyin(data, &event, sizeof (au_evclass_map_t)))
1109 		return (EFAULT);
1110 
1111 	if (event.ec_number > MAX_KEVENTS)
1112 		return (EINVAL);
1113 
1114 	kctx->auk_ets[event.ec_number] = event.ec_class;
1115 
1116 	return (0);
1117 }
1118 
1119 static int
1120 getpinfo(caddr_t data)
1121 {
1122 	STRUCT_DECL(auditpinfo, apinfo);
1123 	proc_t *proc;
1124 	const auditinfo_addr_t	*ainfo;
1125 	model_t	model;
1126 	cred_t	*cr, *newcred;
1127 
1128 	model = get_udatamodel();
1129 	STRUCT_INIT(apinfo, model);
1130 
1131 	if (copyin(data, STRUCT_BUF(apinfo), STRUCT_SIZE(apinfo)))
1132 		return (EFAULT);
1133 
1134 	newcred = cralloc();
1135 
1136 	mutex_enter(&pidlock);
1137 	if ((proc = prfind(STRUCT_FGET(apinfo, ap_pid))) == NULL) {
1138 		mutex_exit(&pidlock);
1139 		crfree(newcred);
1140 		return (ESRCH);		/* no such process */
1141 	}
1142 	mutex_enter(&proc->p_lock);	/* so process doesn't go away */
1143 	mutex_exit(&pidlock);
1144 
1145 	audit_update_context(proc, newcred);	/* make sure it's up-to-date */
1146 
1147 	mutex_enter(&proc->p_crlock);
1148 	crhold(cr = proc->p_cred);
1149 	mutex_exit(&proc->p_crlock);
1150 	mutex_exit(&proc->p_lock);
1151 
1152 	ainfo = crgetauinfo(cr);
1153 	if (ainfo == NULL) {
1154 		crfree(cr);
1155 		return (EINVAL);
1156 	}
1157 
1158 	/* designated process has an ipv6 address? */
1159 	if (ainfo->ai_termid.at_type == AU_IPv6) {
1160 		crfree(cr);
1161 		return (EOVERFLOW);
1162 	}
1163 
1164 	STRUCT_FSET(apinfo, ap_auid, ainfo->ai_auid);
1165 	STRUCT_FSET(apinfo, ap_asid, ainfo->ai_asid);
1166 #ifdef _LP64
1167 	if (model == DATAMODEL_ILP32) {
1168 		dev32_t dev;
1169 		/* convert internal 64 bit form to 32 bit version */
1170 		if (cmpldev(&dev, ainfo->ai_termid.at_port) == 0) {
1171 			crfree(cr);
1172 			return (EOVERFLOW);
1173 		}
1174 		STRUCT_FSET(apinfo, ap_termid.port, dev);
1175 	} else
1176 		STRUCT_FSET(apinfo, ap_termid.port, ainfo->ai_termid.at_port);
1177 #else
1178 	STRUCT_FSET(apinfo, ap_termid.port, ainfo->ai_termid.at_port);
1179 #endif
1180 	STRUCT_FSET(apinfo, ap_termid.machine, ainfo->ai_termid.at_addr[0]);
1181 	STRUCT_FSET(apinfo, ap_mask, ainfo->ai_mask);
1182 
1183 	crfree(cr);
1184 
1185 	if (copyout(STRUCT_BUF(apinfo), data, STRUCT_SIZE(apinfo)))
1186 		return (EFAULT);
1187 
1188 	return (0);
1189 }
1190 
1191 static int
1192 getpinfo_addr(caddr_t data, int len)
1193 {
1194 	STRUCT_DECL(auditpinfo_addr, apinfo);
1195 	proc_t *proc;
1196 	const auditinfo_addr_t	*ainfo;
1197 	model_t	model;
1198 	cred_t	*cr, *newcred;
1199 
1200 	model = get_udatamodel();
1201 	STRUCT_INIT(apinfo, model);
1202 
1203 	if (len < STRUCT_SIZE(apinfo))
1204 		return (EOVERFLOW);
1205 
1206 	if (copyin(data, STRUCT_BUF(apinfo), STRUCT_SIZE(apinfo)))
1207 		return (EFAULT);
1208 
1209 	newcred = cralloc();
1210 
1211 	mutex_enter(&pidlock);
1212 	if ((proc = prfind(STRUCT_FGET(apinfo, ap_pid))) == NULL) {
1213 		mutex_exit(&pidlock);
1214 		crfree(newcred);
1215 		return (ESRCH);
1216 	}
1217 	mutex_enter(&proc->p_lock);	/* so process doesn't go away */
1218 	mutex_exit(&pidlock);
1219 
1220 	audit_update_context(proc, newcred);	/* make sure it's up-to-date */
1221 
1222 	mutex_enter(&proc->p_crlock);
1223 	crhold(cr = proc->p_cred);
1224 	mutex_exit(&proc->p_crlock);
1225 	mutex_exit(&proc->p_lock);
1226 
1227 	ainfo = crgetauinfo(cr);
1228 	if (ainfo == NULL) {
1229 		crfree(cr);
1230 		return (EINVAL);
1231 	}
1232 
1233 	STRUCT_FSET(apinfo, ap_auid, ainfo->ai_auid);
1234 	STRUCT_FSET(apinfo, ap_asid, ainfo->ai_asid);
1235 #ifdef _LP64
1236 	if (model == DATAMODEL_ILP32) {
1237 		dev32_t dev;
1238 		/* convert internal 64 bit form to 32 bit version */
1239 		if (cmpldev(&dev, ainfo->ai_termid.at_port) == 0) {
1240 			crfree(cr);
1241 			return (EOVERFLOW);
1242 		}
1243 		STRUCT_FSET(apinfo, ap_termid.at_port, dev);
1244 	} else
1245 		STRUCT_FSET(apinfo, ap_termid.at_port,
1246 		    ainfo->ai_termid.at_port);
1247 #else
1248 	STRUCT_FSET(apinfo, ap_termid.at_port, ainfo->ai_termid.at_port);
1249 #endif
1250 	STRUCT_FSET(apinfo, ap_termid.at_type, ainfo->ai_termid.at_type);
1251 	STRUCT_FSET(apinfo, ap_termid.at_addr[0], ainfo->ai_termid.at_addr[0]);
1252 	STRUCT_FSET(apinfo, ap_termid.at_addr[1], ainfo->ai_termid.at_addr[1]);
1253 	STRUCT_FSET(apinfo, ap_termid.at_addr[2], ainfo->ai_termid.at_addr[2]);
1254 	STRUCT_FSET(apinfo, ap_termid.at_addr[3], ainfo->ai_termid.at_addr[3]);
1255 	STRUCT_FSET(apinfo, ap_mask, ainfo->ai_mask);
1256 
1257 	crfree(cr);
1258 
1259 	if (copyout(STRUCT_BUF(apinfo), data, STRUCT_SIZE(apinfo)))
1260 		return (EFAULT);
1261 
1262 	return (0);
1263 }
1264 
1265 static int
1266 setpmask(caddr_t data)
1267 {
1268 	STRUCT_DECL(auditpinfo, apinfo);
1269 	proc_t *proc;
1270 	cred_t	*newcred;
1271 	auditinfo_addr_t	*ainfo;
1272 	struct p_audit_data	*pad;
1273 
1274 	model_t	model;
1275 
1276 	model = get_udatamodel();
1277 	STRUCT_INIT(apinfo, model);
1278 
1279 	if (copyin(data, STRUCT_BUF(apinfo), STRUCT_SIZE(apinfo)))
1280 		return (EFAULT);
1281 
1282 	mutex_enter(&pidlock);
1283 	if ((proc = prfind(STRUCT_FGET(apinfo, ap_pid))) == NULL) {
1284 		mutex_exit(&pidlock);
1285 		return (ESRCH);
1286 	}
1287 	mutex_enter(&proc->p_lock);	/* so process doesn't go away */
1288 	mutex_exit(&pidlock);
1289 
1290 	newcred = cralloc();
1291 	if ((ainfo = crgetauinfo_modifiable(newcred)) == NULL) {
1292 		mutex_exit(&proc->p_lock);
1293 		crfree(newcred);
1294 		return (EINVAL);
1295 	}
1296 
1297 	mutex_enter(&proc->p_crlock);
1298 	crcopy_to(proc->p_cred, newcred);
1299 	proc->p_cred = newcred;
1300 
1301 	ainfo->ai_mask = STRUCT_FGET(apinfo, ap_mask);
1302 
1303 	/*
1304 	 * Unlock. No need to broadcast changes via set_proc_pre_sys(),
1305 	 * since t_pre_sys is ALWAYS on when audit is enabled... due to
1306 	 * syscall auditing.
1307 	 */
1308 	crfree(newcred);
1309 	mutex_exit(&proc->p_crlock);
1310 
1311 	/* Reset flag for any previous pending mask change; this supercedes */
1312 	pad = P2A(proc);
1313 	ASSERT(pad != NULL);
1314 	mutex_enter(&(pad->pad_lock));
1315 	pad->pad_flags &= ~PAD_SETMASK;
1316 	mutex_exit(&(pad->pad_lock));
1317 
1318 	mutex_exit(&proc->p_lock);
1319 
1320 	return (0);
1321 }
1322 
1323 /*
1324  * The out of control system call
1325  * This is audit kitchen sink aka auditadm, aka auditon
1326  */
1327 int
1328 auditctl(
1329 	int	cmd,
1330 	caddr_t data,
1331 	int	length)
1332 {
1333 	int result;
1334 
1335 	switch (cmd) {
1336 	case A_GETCOND:
1337 	case A_GETCAR:
1338 	case A_GETCLASS:
1339 	case A_GETCWD:
1340 	case A_GETKAUDIT:
1341 	case A_GETKMASK:
1342 	case A_GETPINFO:
1343 	case A_GETPINFO_ADDR:
1344 	case A_GETPOLICY:
1345 	case A_GETQCTRL:
1346 	case A_GETSTAT:
1347 		if (secpolicy_audit_getattr(CRED()) != 0)
1348 			return (EPERM);
1349 		break;
1350 	default:
1351 		if (secpolicy_audit_config(CRED()) != 0)
1352 			return (EPERM);
1353 		break;
1354 	}
1355 
1356 	switch (cmd) {
1357 	case A_GETPOLICY:
1358 		result = getpolicy(data);
1359 		break;
1360 	case A_SETPOLICY:
1361 		result = setpolicy(data);
1362 		break;
1363 	case A_GETKMASK:
1364 		result = getkmask(data);
1365 		break;
1366 	case A_SETKMASK:
1367 		result = setkmask(data);
1368 		break;
1369 	case A_GETKAUDIT:
1370 		result = getkaudit(data, length);
1371 		break;
1372 	case A_SETKAUDIT:
1373 		result = setkaudit(data, length);
1374 		break;
1375 	case A_GETQCTRL:
1376 		result = getqctrl(data);
1377 		break;
1378 	case A_SETQCTRL:
1379 		result = setqctrl(data);
1380 		break;
1381 	case A_GETCWD:
1382 		result = getcwd(data, length);
1383 		break;
1384 	case A_GETCAR:
1385 		result = getcar(data, length);
1386 		break;
1387 	case A_GETSTAT:
1388 		result = getstat(data);
1389 		break;
1390 	case A_SETSTAT:
1391 		result = setstat(data);
1392 		break;
1393 	case A_SETUMASK:
1394 		result = setumask(data);
1395 		break;
1396 	case A_SETSMASK:
1397 		result = setsmask(data);
1398 		break;
1399 	case A_GETCOND:
1400 		result = getcond(data);
1401 		break;
1402 	case A_SETCOND:
1403 		result = setcond(data);
1404 		break;
1405 	case A_GETCLASS:
1406 		result = getclass(data);
1407 		break;
1408 	case A_SETCLASS:
1409 		result = setclass(data);
1410 		break;
1411 	case A_GETPINFO:
1412 		result = getpinfo(data);
1413 		break;
1414 	case A_GETPINFO_ADDR:
1415 		result = getpinfo_addr(data, length);
1416 		break;
1417 	case A_SETPMASK:
1418 		result = setpmask(data);
1419 		break;
1420 	default:
1421 		result = EINVAL;
1422 		break;
1423 	}
1424 	return (result);
1425 }
1426