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.
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.
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]
23 * Copyright 2010 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
27 #include <arpa/inet.h>
36 #include <net/if_dl.h>
37 #include <netinet/in.h>
43 #include <sys/socket.h>
44 #include <sys/types.h>
47 #include "libnwam_impl.h"
48 #include <libnwam_priv.h>
52 * Utility functions for door access, common validation functions etc.
55 pthread_mutex_t door_mutex
= PTHREAD_MUTEX_INITIALIZER
;
56 int nwam_door_fd
= -1;
59 open_door(const char *door_name
, int *door_fdp
)
61 struct door_info dinfo
;
64 (void) pthread_mutex_lock(&door_mutex
);
66 if (*door_fdp
!= -1) {
67 /* Check door fd is not old (from previous nwamd). */
68 if (door_info(*door_fdp
, &dinfo
) != 0 ||
69 (dinfo
.di_attributes
& DOOR_REVOKED
) != 0) {
70 (void) close(*door_fdp
);
74 if (*door_fdp
== -1) {
75 *door_fdp
= open(door_name
, O_RDONLY
);
80 (void) pthread_mutex_unlock(&door_mutex
);
86 nwam_make_door_call(const char *door_name
, int *door_fdp
,
87 void *request
, size_t request_size
)
92 door_args
.data_ptr
= (void *)request
;
93 door_args
.data_size
= request_size
;
94 door_args
.desc_ptr
= NULL
;
95 door_args
.desc_num
= 0;
96 door_args
.rbuf
= (void *)request
;
97 door_args
.rsize
= request_size
;
99 if ((err
= open_door(door_name
, door_fdp
)) != 0)
102 if (door_call(*door_fdp
, &door_args
) == -1)
109 send_msg_to_nwam(nwamd_door_arg_t
*request
)
113 if ((err
= nwam_make_door_call(NWAM_DOOR
, &nwam_door_fd
,
114 request
, sizeof (nwamd_door_arg_t
))) != 0) {
116 return (NWAM_ERROR_BIND
);
117 return (nwam_errno_to_nwam_error(err
));
120 switch (request
->nwda_status
) {
121 case NWAM_REQUEST_STATUS_OK
:
122 return (NWAM_SUCCESS
);
123 case NWAM_REQUEST_STATUS_UNKNOWN
:
124 return (NWAM_INVALID_ARG
);
125 case NWAM_REQUEST_STATUS_ALREADY
:
126 return (NWAM_ENTITY_IN_USE
);
127 case NWAM_REQUEST_STATUS_FAILED
:
128 return (request
->nwda_error
);
130 return (NWAM_ERROR_INTERNAL
);
135 nwam_request_register_unregister(nwam_request_type_t type
,
136 const char *event_msg_file
)
138 nwamd_door_arg_t req
;
140 req
.nwda_type
= type
;
142 (void) strlcpy(req
.nwda_data
.nwdad_register_info
.nwdad_name
,
144 sizeof (req
.nwda_data
.nwdad_register_info
.nwdad_name
));
146 return (send_msg_to_nwam(&req
));
150 nwam_request_action(nwam_object_type_t object_type
,
151 const char *name
, const char *parent
, nwam_action_t action
)
153 nwamd_door_arg_t req
;
155 assert(name
!= NULL
);
157 req
.nwda_type
= NWAM_REQUEST_TYPE_ACTION
;
158 req
.nwda_data
.nwdad_object_action
.nwdad_object_type
= object_type
;
159 req
.nwda_data
.nwdad_object_action
.nwdad_action
= action
;
160 (void) strlcpy(req
.nwda_data
.nwdad_object_action
.nwdad_name
, name
,
161 sizeof (req
.nwda_data
.nwdad_object_action
.nwdad_name
));
162 if (parent
!= NULL
) {
163 (void) strlcpy(req
.nwda_data
.nwdad_object_action
.nwdad_parent
,
165 sizeof (req
.nwda_data
.nwdad_object_action
.nwdad_parent
));
167 req
.nwda_data
.nwdad_object_action
.nwdad_parent
[0] = '\0';
170 return (send_msg_to_nwam(&req
));
174 nwam_request_state(nwam_object_type_t object_type
, const char *name
,
175 const char *parent
, nwam_state_t
*statep
, nwam_aux_state_t
*auxp
)
177 nwamd_door_arg_t req
;
180 assert(name
!= NULL
&& statep
!= NULL
&& auxp
!= NULL
);
182 req
.nwda_type
= NWAM_REQUEST_TYPE_STATE
;
184 req
.nwda_data
.nwdad_object_state
.nwdad_object_type
= object_type
;
186 (void) strlcpy(req
.nwda_data
.nwdad_object_state
.nwdad_name
, name
,
187 sizeof (req
.nwda_data
.nwdad_object_state
.nwdad_name
));
188 if (parent
!= NULL
) {
189 (void) strlcpy(req
.nwda_data
.nwdad_object_state
.nwdad_parent
,
191 sizeof (req
.nwda_data
.nwdad_object_state
.nwdad_parent
));
194 err
= send_msg_to_nwam(&req
);
196 if (err
== NWAM_SUCCESS
) {
197 *statep
= req
.nwda_data
.nwdad_object_state
.nwdad_state
;
198 *auxp
= req
.nwda_data
.nwdad_object_state
.nwdad_aux_state
;
205 nwam_request_wlan(nwam_request_type_t type
, const char *name
,
206 const char *essid
, const char *bssid
, uint32_t security_mode
,
207 uint_t keyslot
, const char *key
, boolean_t add_to_known_wlans
)
209 nwamd_door_arg_t req
;
211 assert(name
!= NULL
);
213 req
.nwda_type
= type
;
215 (void) strlcpy(req
.nwda_data
.nwdad_wlan_info
.nwdad_name
, name
,
216 sizeof (req
.nwda_data
.nwdad_wlan_info
));
218 (void) strlcpy(req
.nwda_data
.nwdad_wlan_info
.nwdad_essid
, essid
,
219 sizeof (req
.nwda_data
.nwdad_wlan_info
.nwdad_essid
));
221 req
.nwda_data
.nwdad_wlan_info
.nwdad_essid
[0] = '\0';
224 (void) strlcpy(req
.nwda_data
.nwdad_wlan_info
.nwdad_bssid
, bssid
,
225 sizeof (req
.nwda_data
.nwdad_wlan_info
.nwdad_bssid
));
227 req
.nwda_data
.nwdad_wlan_info
.nwdad_bssid
[0] = '\0';
230 (void) strlcpy(req
.nwda_data
.nwdad_wlan_info
.nwdad_key
, key
,
231 sizeof (req
.nwda_data
.nwdad_wlan_info
.nwdad_key
));
232 req
.nwda_data
.nwdad_wlan_info
.nwdad_keyslot
= keyslot
;
234 req
.nwda_data
.nwdad_wlan_info
.nwdad_key
[0] = '\0';
237 req
.nwda_data
.nwdad_wlan_info
.nwdad_security_mode
= security_mode
;
238 req
.nwda_data
.nwdad_wlan_info
.nwdad_add_to_known_wlans
=
241 return (send_msg_to_nwam(&req
));
245 nwam_request_wlan_scan_results(const char *name
, uint_t
*num_wlansp
,
246 nwam_wlan_t
**wlansp
)
248 nwamd_door_arg_t req
;
251 assert(name
!= NULL
&& num_wlansp
!= NULL
&& wlansp
!= NULL
);
253 req
.nwda_type
= NWAM_REQUEST_TYPE_WLAN_SCAN_RESULTS
;
255 (void) strlcpy(req
.nwda_data
.nwdad_wlan_info
.nwdad_name
, name
,
256 sizeof (req
.nwda_data
.nwdad_wlan_info
.nwdad_name
));
258 if ((err
= send_msg_to_nwam(&req
)) != NWAM_SUCCESS
)
261 *num_wlansp
= req
.nwda_data
.nwdad_wlan_info
.nwdad_num_wlans
;
263 *wlansp
= calloc(*num_wlansp
, sizeof (nwam_wlan_t
));
265 return (NWAM_NO_MEMORY
);
267 (void) memcpy(*wlansp
, req
.nwda_data
.nwdad_wlan_info
.nwdad_wlans
,
268 *num_wlansp
* sizeof (nwam_wlan_t
));
270 return (NWAM_SUCCESS
);
274 nwam_request_active_priority_group(int64_t *priorityp
)
276 nwamd_door_arg_t req
;
279 assert(priorityp
!= NULL
);
281 req
.nwda_type
= NWAM_REQUEST_TYPE_PRIORITY_GROUP
;
282 err
= send_msg_to_nwam(&req
);
284 if (err
== NWAM_SUCCESS
)
286 req
.nwda_data
.nwdad_priority_group_info
.nwdad_priority
;
291 /* String conversion functions */
294 nwam_value_type_to_string(nwam_value_type_t type
)
297 case NWAM_VALUE_TYPE_BOOLEAN
:
299 case NWAM_VALUE_TYPE_INT64
:
301 case NWAM_VALUE_TYPE_UINT64
:
303 case NWAM_VALUE_TYPE_STRING
:
311 nwam_string_to_value_type(const char *typestr
)
313 if (strncmp(typestr
, nwam_value_type_to_string(NWAM_VALUE_TYPE_BOOLEAN
),
314 strlen(typestr
)) == 0)
315 return (NWAM_VALUE_TYPE_BOOLEAN
);
316 if (strncmp(typestr
, nwam_value_type_to_string(NWAM_VALUE_TYPE_INT64
),
317 strlen(typestr
)) == 0)
318 return (NWAM_VALUE_TYPE_INT64
);
319 if (strncmp(typestr
, nwam_value_type_to_string(NWAM_VALUE_TYPE_UINT64
),
320 strlen(typestr
)) == 0)
321 return (NWAM_VALUE_TYPE_UINT64
);
322 if (strncmp(typestr
, nwam_value_type_to_string(NWAM_VALUE_TYPE_STRING
),
323 strlen(typestr
)) == 0)
324 return (NWAM_VALUE_TYPE_STRING
);
325 return (NWAM_VALUE_TYPE_UNKNOWN
);
329 nwam_action_to_string(nwam_action_t action
)
332 case NWAM_ACTION_ADD
:
334 case NWAM_ACTION_REMOVE
:
336 case NWAM_ACTION_REFRESH
:
338 case NWAM_ACTION_ENABLE
:
340 case NWAM_ACTION_DISABLE
:
342 case NWAM_ACTION_DESTROY
:
350 nwam_event_type_to_string(int event_type
)
352 switch (event_type
) {
353 case NWAM_EVENT_TYPE_NOOP
:
355 case NWAM_EVENT_TYPE_INIT
:
357 case NWAM_EVENT_TYPE_SHUTDOWN
:
359 case NWAM_EVENT_TYPE_OBJECT_ACTION
:
360 return ("OBJECT_ACTION");
361 case NWAM_EVENT_TYPE_OBJECT_STATE
:
362 return ("OBJECT_STATE");
363 case NWAM_EVENT_TYPE_PRIORITY_GROUP
:
364 return ("PRIORITY_GROUP");
365 case NWAM_EVENT_TYPE_INFO
:
367 case NWAM_EVENT_TYPE_WLAN_SCAN_REPORT
:
368 return ("WLAN_SCAN_REPORT");
369 case NWAM_EVENT_TYPE_WLAN_NEED_CHOICE
:
370 return ("WLAN_NEED_CHOICE");
371 case NWAM_EVENT_TYPE_WLAN_NEED_KEY
:
372 return ("WLAN_NEED_KEY");
373 case NWAM_EVENT_TYPE_WLAN_CONNECTION_REPORT
:
374 return ("WLAN_CONNECTION_REPORT");
375 case NWAM_EVENT_TYPE_IF_ACTION
:
376 return ("IF_ACTION");
377 case NWAM_EVENT_TYPE_IF_STATE
:
379 case NWAM_EVENT_TYPE_LINK_ACTION
:
380 return ("LINK_ACTION");
381 case NWAM_EVENT_TYPE_LINK_STATE
:
382 return ("LINK_STATE");
389 nwam_state_to_string(nwam_state_t state
)
392 case NWAM_STATE_UNINITIALIZED
:
393 return ("uninitialized");
394 case NWAM_STATE_INITIALIZED
:
395 return ("initialized");
396 case NWAM_STATE_OFFLINE
:
398 case NWAM_STATE_OFFLINE_TO_ONLINE
:
400 case NWAM_STATE_ONLINE_TO_OFFLINE
:
402 case NWAM_STATE_ONLINE
:
404 case NWAM_STATE_MAINTENANCE
:
405 return ("maintenance");
406 case NWAM_STATE_DEGRADED
:
408 case NWAM_STATE_DISABLED
:
416 nwam_aux_state_to_string(nwam_aux_state_t aux_state
)
419 case NWAM_AUX_STATE_UNINITIALIZED
:
420 return ("uninitialized");
421 case NWAM_AUX_STATE_INITIALIZED
:
422 return ("(re)initialized but not configured");
423 case NWAM_AUX_STATE_CONDITIONS_NOT_MET
:
424 return ("conditions for activation are unmet");
425 case NWAM_AUX_STATE_MANUAL_DISABLE
:
426 return ("disabled by administrator");
427 case NWAM_AUX_STATE_METHOD_FAILED
:
428 return ("method/service failed");
429 case NWAM_AUX_STATE_METHOD_MISSING
:
430 return ("method or FMRI not specified");
431 case NWAM_AUX_STATE_INVALID_CONFIG
:
432 return ("invalid configuration values");
433 case NWAM_AUX_STATE_METHOD_RUNNING
:
434 return ("method/service executing");
435 case NWAM_AUX_STATE_ACTIVE
:
437 case NWAM_AUX_STATE_LINK_WIFI_SCANNING
:
438 return ("scanning for WiFi networks");
439 case NWAM_AUX_STATE_LINK_WIFI_NEED_SELECTION
:
440 return ("need WiFi network selection");
441 case NWAM_AUX_STATE_LINK_WIFI_NEED_KEY
:
442 return ("need WiFi security key");
443 case NWAM_AUX_STATE_LINK_WIFI_CONNECTING
:
444 return ("connecting to WiFi network");
445 case NWAM_AUX_STATE_IF_WAITING_FOR_ADDR
:
446 return ("waiting for IP address to be set");
447 case NWAM_AUX_STATE_IF_DHCP_TIMED_OUT
:
448 return ("DHCP wait timeout, still trying...");
449 case NWAM_AUX_STATE_IF_DUPLICATE_ADDR
:
450 return ("duplicate address detected");
451 case NWAM_AUX_STATE_UP
:
452 return ("interface/link is up");
453 case NWAM_AUX_STATE_DOWN
:
454 return ("interface/link is down");
455 case NWAM_AUX_STATE_NOT_FOUND
:
456 return ("interface/link not found");
463 nwam_object_type_to_string(nwam_object_type_t type
)
466 case NWAM_OBJECT_TYPE_NCP
:
468 case NWAM_OBJECT_TYPE_NCU
:
470 case NWAM_OBJECT_TYPE_LOC
:
472 case NWAM_OBJECT_TYPE_ENM
:
474 case NWAM_OBJECT_TYPE_KNOWN_WLAN
:
475 return ("known wlan");
482 nwam_string_to_object_type(const char *typestr
)
484 if (strcasecmp(typestr
,
485 nwam_object_type_to_string(NWAM_OBJECT_TYPE_NCP
)) == 0)
486 return (NWAM_OBJECT_TYPE_NCP
);
487 if (strcasecmp(typestr
,
488 nwam_object_type_to_string(NWAM_OBJECT_TYPE_NCU
)) == 0)
489 return (NWAM_OBJECT_TYPE_NCU
);
490 if (strcasecmp(typestr
,
491 nwam_object_type_to_string(NWAM_OBJECT_TYPE_LOC
)) == 0)
492 return (NWAM_OBJECT_TYPE_LOC
);
493 if (strcasecmp(typestr
,
494 nwam_object_type_to_string(NWAM_OBJECT_TYPE_ENM
)) == 0)
495 return (NWAM_OBJECT_TYPE_ENM
);
496 if (strcasecmp(typestr
,
497 nwam_object_type_to_string(NWAM_OBJECT_TYPE_KNOWN_WLAN
)) == 0)
498 return (NWAM_OBJECT_TYPE_KNOWN_WLAN
);
499 return (NWAM_OBJECT_TYPE_UNKNOWN
);
503 nwam_errno_to_nwam_error(int errnum
)
507 return (NWAM_SUCCESS
);
509 return (NWAM_ERROR_BIND
);
512 return (NWAM_PERMISSION_DENIED
);
514 return (NWAM_ENTITY_NOT_FOUND
);
516 return (NWAM_ENTITY_INVALID
);
518 return (NWAM_ENTITY_EXISTS
);
521 return (NWAM_ENTITY_IN_USE
);
524 return (NWAM_NO_MEMORY
);
527 return (NWAM_INVALID_ARG
);
529 return (NWAM_ERROR_INTERNAL
);
533 /* Common validation functions */
536 * Do the flags represent a subset of valid_flags?
539 nwam_valid_flags(uint64_t flags
, uint64_t valid_flags
)
542 if ((flags
| valid_flags
) != valid_flags
)
543 return (NWAM_INVALID_ARG
);
544 return (NWAM_SUCCESS
);
548 nwam_valid_condition(nwam_value_t value
)
552 nwam_condition_object_type_t object_type
;
553 nwam_condition_t condition
;
555 if (nwam_value_get_string_array(value
, &conditions
, &numvalues
)
557 return (NWAM_ENTITY_INVALID_VALUE
);
559 for (i
= 0; i
< numvalues
; i
++) {
560 char *object_name
= NULL
;
562 if (nwam_condition_string_to_condition(conditions
[i
],
563 &object_type
, &condition
, &object_name
) != NWAM_SUCCESS
)
564 return (NWAM_ENTITY_INVALID_VALUE
);
567 return (NWAM_SUCCESS
);
570 /* check if boolean values are correct, generalize for array of booleans */
572 nwam_valid_boolean(nwam_value_t value
)
577 if (nwam_value_get_boolean_array(value
, &val
, &numvalues
)
579 return (NWAM_ENTITY_INVALID_VALUE
);
581 for (i
= 0; i
< numvalues
; i
++) {
582 if (val
[i
] != B_TRUE
&& val
[i
] != B_FALSE
)
583 return (NWAM_ENTITY_INVALID_VALUE
);
585 return (NWAM_SUCCESS
);
588 /* check if uint64 values are correct, generalize for array of ints */
590 nwam_valid_uint64(nwam_value_t value
)
595 if (nwam_value_get_int64_array(value
, &val
, &numvalues
)
597 return (NWAM_ENTITY_INVALID_VALUE
);
599 for (i
= 0; i
< numvalues
; i
++) {
601 return (NWAM_ENTITY_INVALID_VALUE
);
603 return (NWAM_SUCCESS
);
606 /* check if domain names are correct, generalize for array of domains */
608 nwam_valid_domain(nwam_value_t value
)
610 char **domainvalues
, *domain
;
614 if (nwam_value_get_string_array(value
, &domainvalues
, &numvalues
)
616 return (NWAM_ENTITY_INVALID_VALUE
);
618 for (i
= 0; i
< numvalues
; i
++) {
620 * First and last character must be alphanumeric.
621 * Only '.' and '-' are allowed.
623 domain
= domainvalues
[i
];
624 len
= strlen(domain
);
625 if (!isalnum(domain
[0]) || !isalnum(domain
[len
-1]))
626 return (NWAM_ENTITY_INVALID_VALUE
);
627 for (j
= 0; j
< len
; j
++) {
628 if (!isalnum(domain
[j
]) &&
629 domain
[j
] != '.' && domain
[j
] != '-')
630 return (NWAM_ENTITY_INVALID_VALUE
);
633 return (NWAM_SUCCESS
);
636 /* check if address prefix is valid */
638 nwam_valid_prefix(char *addr
, int max_plen
)
643 if ((prefix
= strchr(addr
, '/')) != NULL
) {
645 prefixlen
= strtol(prefix
, &end
, 10);
646 if (prefix
== end
|| prefixlen
< 0 || prefixlen
> max_plen
)
647 return (NWAM_ENTITY_INVALID_VALUE
);
649 return (NWAM_SUCCESS
);
652 /* check if IPv4 addresses are correct, generalize for array of addresses */
654 nwam_valid_host_v4(nwam_value_t value
)
656 char **addrvalues
, *addr
;
658 struct sockaddr_in sa
;
660 if (nwam_value_get_string_array(value
, &addrvalues
, &numvalues
)
662 return (NWAM_ENTITY_INVALID_VALUE
);
664 for (i
= 0; i
< numvalues
; i
++) {
665 addr
= strdup(addrvalues
[i
]);
666 if (nwam_valid_prefix(addr
, IP_ABITS
) != NWAM_SUCCESS
) {
668 return (NWAM_ENTITY_INVALID_VALUE
);
670 /* replace '/' with '\0' */
671 addr
= strsep(&addr
, "/");
672 if (inet_pton(AF_INET
, addr
, &(sa
.sin_addr
)) != 1) {
674 return (NWAM_ENTITY_INVALID_VALUE
);
678 return (NWAM_SUCCESS
);
681 /* Check if IPv4 address for default route is valid */
683 nwam_valid_route_v4(nwam_value_t value
)
686 struct sockaddr_in sa
;
688 if (nwam_value_get_string(value
, &addrvalue
) != NWAM_SUCCESS
)
689 return (NWAM_ENTITY_INVALID_VALUE
);
691 if (inet_pton(AF_INET
, addrvalue
, &(sa
.sin_addr
)) != 1)
692 return (NWAM_ENTITY_INVALID_VALUE
);
694 return (NWAM_SUCCESS
);
697 /* check if IPv6 addresses are correct, generalize for array of addresses */
699 nwam_valid_host_v6(nwam_value_t value
)
701 char **addrvalues
, *addr
;
703 struct sockaddr_in6 sa
;
705 if (nwam_value_get_string_array(value
, &addrvalues
, &numvalues
)
707 return (NWAM_ENTITY_INVALID_VALUE
);
709 for (i
= 0; i
< numvalues
; i
++) {
710 addr
= strdup(addrvalues
[i
]);
711 if (nwam_valid_prefix(addr
, IPV6_ABITS
) != NWAM_SUCCESS
) {
713 return (NWAM_ENTITY_INVALID_VALUE
);
715 /* replace '/' with '\0' */
716 addr
= strsep(&addr
, "/");
717 if (inet_pton(AF_INET6
, addr
, &(sa
.sin6_addr
)) != 1) {
719 return (NWAM_ENTITY_INVALID_VALUE
);
723 return (NWAM_SUCCESS
);
726 /* Check if IPv4 address for default route is valid */
728 nwam_valid_route_v6(nwam_value_t value
)
731 struct sockaddr_in6 sa
;
733 if (nwam_value_get_string(value
, &addrvalue
) != NWAM_SUCCESS
)
734 return (NWAM_ENTITY_INVALID_VALUE
);
736 if (inet_pton(AF_INET6
, addrvalue
, &(sa
.sin6_addr
)) != 1)
737 return (NWAM_ENTITY_INVALID_VALUE
);
739 return (NWAM_SUCCESS
);
743 nwam_valid_host_any(nwam_value_t value
)
745 if (nwam_valid_host_v4(value
) != NWAM_SUCCESS
&&
746 nwam_valid_host_v6(value
) != NWAM_SUCCESS
)
747 return (NWAM_ENTITY_INVALID_VALUE
);
748 return (NWAM_SUCCESS
);
752 nwam_valid_host_or_domain(nwam_value_t value
)
754 if (nwam_valid_host_any(value
) != NWAM_SUCCESS
&&
755 nwam_valid_domain(value
) != NWAM_SUCCESS
)
756 return (NWAM_ENTITY_INVALID_VALUE
);
757 return (NWAM_SUCCESS
);
760 /* We do not validate file existence, merely that it is an absolute path. */
762 nwam_valid_file(nwam_value_t value
)
767 if (nwam_value_get_string_array(value
, &files
, &numvalues
)
769 return (NWAM_ENTITY_INVALID_VALUE
);
771 for (i
= 0; i
< numvalues
; i
++) {
773 while (isspace(files
[i
][j
]))
775 if (files
[i
][j
] != '/')
776 return (NWAM_ENTITY_INVALID_VALUE
);
778 return (NWAM_SUCCESS
);
782 * We do not validate existence of the object pointed to by the FMRI
783 * but merely ensure that it is a valid FMRI. We do this by
784 * using scf_handle_decode_fmri(), but ignore all errors bar
785 * SCF_ERROR_INVALID_ARGUMENT (which indicates the FMRI is invalid).
788 nwam_valid_fmri(nwam_value_t value
)
791 scf_handle_t
*h
= NULL
;
792 scf_service_t
*svc
= NULL
;
794 nwam_error_t err
= NWAM_SUCCESS
;
796 if ((err
= nwam_value_get_string_array(value
, &valstr
, &numvalues
))
800 h
= scf_handle_create(SCF_VERSION
);
802 return (NWAM_ERROR_INTERNAL
);
804 if (scf_handle_bind(h
) != 0) {
805 err
= NWAM_ERROR_INTERNAL
;
809 if ((svc
= scf_service_create(h
)) == NULL
) {
810 err
= NWAM_ERROR_INTERNAL
;
815 for (i
= 0; i
< numvalues
; i
++) {
816 if (scf_handle_decode_fmri(h
, valstr
[i
], NULL
, svc
,
817 NULL
, NULL
, NULL
, SCF_DECODE_FMRI_TRUNCATE
) == 0 ||
818 scf_error() != SCF_ERROR_INVALID_ARGUMENT
) {
822 err
= NWAM_ENTITY_INVALID_VALUE
;
826 scf_service_destroy(svc
);
827 scf_handle_destroy(h
);
831 /* verifies mac-address and bssids */
833 nwam_valid_mac_addr(nwam_value_t value
)
835 char **mac_addrs
, *addr
;
840 if (nwam_value_get_string_array(value
, &mac_addrs
, &numvalues
)
842 return (NWAM_ENTITY_INVALID_VALUE
);
844 for (i
= 0; i
< numvalues
; i
++) {
848 /* validate that a-fA-F0-9 and ':' only */
849 while (addr
[j
] != 0) {
850 if (!isxdigit(addr
[j
]) && addr
[j
] != ':')
851 return (NWAM_ENTITY_INVALID_VALUE
);
855 if ((hwaddr
= _link_aton(addr
, &hwaddrlen
)) == NULL
)
856 return (NWAM_ENTITY_INVALID_VALUE
);
860 return (NWAM_SUCCESS
);
864 nwam_uid_is_special(void)
866 uid_t uid
= getuid();
867 return (uid
== UID_NETADM
|| uid
== 0);
871 nwam_get_smf_string_property(const char *fmri
, const char *pgname
,
872 const char *propname
, char **valuep
)
874 scf_handle_t
*h
= NULL
;
875 scf_snapshot_t
*snap
= NULL
;
876 scf_instance_t
*inst
= NULL
;
877 scf_propertygroup_t
*pg
= NULL
;
878 scf_property_t
*prop
= NULL
;
879 scf_value_t
*val
= NULL
;
880 nwam_error_t err
= NWAM_SUCCESS
;
882 if ((*valuep
= malloc(NWAM_MAX_NAME_LEN
)) == NULL
)
883 return (NWAM_NO_MEMORY
);
885 if ((h
= scf_handle_create(SCF_VERSION
)) == NULL
||
886 scf_handle_bind(h
) != 0 ||
887 (inst
= scf_instance_create(h
)) == NULL
||
888 (snap
= scf_snapshot_create(h
)) == NULL
||
889 (pg
= scf_pg_create(h
)) == NULL
||
890 (prop
= scf_property_create(h
)) == NULL
||
891 (val
= scf_value_create(h
)) == NULL
) {
892 err
= NWAM_ERROR_INTERNAL
;
895 if (scf_handle_decode_fmri(h
, fmri
, NULL
, NULL
, inst
,
896 NULL
, NULL
, SCF_DECODE_FMRI_REQUIRE_INSTANCE
) != 0) {
897 err
= NWAM_ENTITY_NOT_FOUND
;
900 /* Retrieve value from running snapshot (if present) */
901 if (scf_instance_get_snapshot(inst
, "running", snap
) != 0) {
902 scf_snapshot_destroy(snap
);
905 if (scf_instance_get_pg_composed(inst
, snap
, pgname
, pg
) != 0 ||
906 scf_pg_get_property(pg
, propname
, prop
) != 0 ||
907 scf_property_get_value(prop
, val
) != 0 ||
908 scf_value_get_astring(val
, *valuep
, NWAM_MAX_NAME_LEN
) == -1) {
909 err
= NWAM_ENTITY_NOT_FOUND
;
912 if (err
!= NWAM_SUCCESS
)
915 scf_value_destroy(val
);
916 scf_property_destroy(prop
);
919 scf_snapshot_destroy(snap
);
920 scf_instance_destroy(inst
);
921 scf_handle_destroy(h
);
927 nwam_set_smf_string_property(const char *fmri
, const char *pgname
,
928 const char *propname
, const char *propval
)
930 scf_handle_t
*h
= NULL
;
931 scf_instance_t
*inst
= NULL
;
932 scf_propertygroup_t
*pg
= NULL
;
933 scf_property_t
*prop
= NULL
;
934 scf_value_t
*val
= NULL
;
935 scf_transaction_t
*tx
= NULL
;
936 scf_transaction_entry_t
*ent
= NULL
;
937 nwam_error_t err
= NWAM_SUCCESS
;
940 if ((h
= scf_handle_create(SCF_VERSION
)) == NULL
||
941 scf_handle_bind(h
) != 0 ||
942 (inst
= scf_instance_create(h
)) == NULL
||
943 (pg
= scf_pg_create(h
)) == NULL
||
944 (prop
= scf_property_create(h
)) == NULL
||
945 (val
= scf_value_create(h
)) == NULL
||
946 scf_value_set_astring(val
, propval
) != 0 ||
947 (tx
= scf_transaction_create(h
)) == NULL
||
948 (ent
= scf_entry_create(h
)) == NULL
) {
949 err
= NWAM_ERROR_INTERNAL
;
952 if (scf_handle_decode_fmri(h
, fmri
, NULL
, NULL
, inst
,
953 NULL
, NULL
, SCF_DECODE_FMRI_REQUIRE_INSTANCE
) != 0 ||
954 scf_instance_get_pg_composed(inst
, NULL
, pgname
, pg
) != 0) {
955 err
= NWAM_ENTITY_NOT_FOUND
;
960 if (scf_transaction_start(tx
, pg
) == -1 ||
961 scf_transaction_property_change(tx
, ent
, propname
, SCF_TYPE_ASTRING
)
962 == -1 || scf_entry_add_value(ent
, val
) != 0) {
963 err
= NWAM_ERROR_INTERNAL
;
967 result
= scf_transaction_commit(tx
);
970 (void) smf_refresh_instance(fmri
);
973 scf_transaction_reset(tx
);
974 if (scf_pg_update(pg
) == -1) {
975 err
= NWAM_ERROR_INTERNAL
;
980 err
= NWAM_ERROR_INTERNAL
;
984 scf_value_destroy(val
);
985 scf_property_destroy(prop
);
987 scf_instance_destroy(inst
);
988 scf_handle_destroy(h
);