/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2009 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* * Logical Domains Device Agent */ #include <errno.h> #include <fcntl.h> #include <libdladm.h> #include <libdllink.h> #include <libds.h> #include <stdio.h> #include <stdlib.h> #include <strings.h> #include <unistd.h> #include <sys/param.h> #include <sys/types.h> #include <sys/stat.h> #include "ldma.h" #define LDMA_MODULE LDMA_NAME_DEVICE #define LDMA_NVERSIONS (sizeof (ldma_versions) / sizeof (ds_ver_t)) #define LDMA_NHANDLERS (sizeof (ldma_handlers) / sizeof (ldma_msg_handler_t)) static ldm_msg_func_t ldma_dev_validate_path; static ldm_msg_func_t ldma_dev_validate_nic; static ds_ver_t ldma_versions[] = { { 1, 0 } }; static ldma_msg_handler_t ldma_handlers[] = { { LDMA_MSGDEV_VALIDATE_PATH, ldma_dev_validate_path }, { LDMA_MSGDEV_VALIDATE_NIC, ldma_dev_validate_nic } }; ldma_agent_info_t ldma_device_info = { LDMA_NAME_DEVICE, ldma_versions, LDMA_NVERSIONS, ldma_handlers, LDMA_NHANDLERS }; /*ARGSUSED*/ static ldma_request_status_t ldma_dev_validate_path(ds_ver_t *ver, ldma_message_header_t *request, size_t request_dlen, ldma_message_header_t **replyp, size_t *reply_dlenp) { ldma_message_header_t *reply = NULL; ldma_request_status_t status; struct stat st; char *path = NULL; uint32_t *path_type, reply_dlen; uint32_t plen; int fd; plen = request->msg_info; if (plen == 0 || plen > MAXPATHLEN || plen > request_dlen) { status = LDMA_REQ_INVALID; goto done; } path = malloc(plen + 1); if (path == NULL) { status = LDMA_REQ_FAILED; goto done; } (void) strncpy(path, LDMA_HDR2DATA(request), plen); path[plen] = '\0'; LDMA_DBG("VALIDATE_PATH(%s)", path); reply_dlen = sizeof (uint32_t); reply = ldma_alloc_result_msg(request, reply_dlen); if (reply == NULL) { status = LDMA_REQ_FAILED; goto done; } /* LINTED E_BAD_PTR_CAST_ALIGN */ path_type = (uint32_t *)(LDMA_HDR2DATA(reply)); reply->msg_info = 0x0; /* check if path exists */ if (stat(path, &st) != 0) { LDMA_DBG("VALIDATE_PATH(%s): stat failed with error %d", path, errno); switch (errno) { case EACCES: case ELOOP: case ENOENT: case ENOLINK: case ENOTDIR: /* path is inaccessible, the request is completed */ status = LDMA_REQ_COMPLETED; break; case ENAMETOOLONG: status = LDMA_REQ_INVALID; break; default: /* request has failed */ status = LDMA_REQ_FAILED; break; } goto done; } status = LDMA_REQ_COMPLETED; reply->msg_info |= LDMA_DEVPATH_EXIST; LDMA_DBG("VALIDATE_PATH(%s): file mode = 0x%x", path, st.st_mode); switch (st.st_mode & S_IFMT) { case S_IFREG: *path_type = LDMA_DEVPATH_TYPE_FILE; break; case S_IFCHR: case S_IFBLK: *path_type = LDMA_DEVPATH_TYPE_DEVICE; break; default: /* we don't advertise other types (fifo, directory...) */ *path_type = 0; } /* check if path can be opened read/write */ if ((fd = open(path, O_RDWR)) != -1) { reply->msg_info |= LDMA_DEVPATH_OPENRW | LDMA_DEVPATH_OPENRO; (void) close(fd); } else { LDMA_DBG("VALIDATE_PATH(%s): open RDWR failed with error %d", path, errno); /* check if path can be opened read only */ if ((fd = open(path, O_RDONLY)) != -1) { reply->msg_info |= LDMA_DEVPATH_OPENRO; (void) close(fd); } else { LDMA_DBG("VALIDATE_PATH(%s): open RDONLY failed " "with error %d", path, errno); } } done: if (status != LDMA_REQ_COMPLETED) { /* * We don't provide a reply message if the request has not * been completed. The LDoms agent daemon will send an * appropriate reply based on the return code of this function. */ free(reply); reply = NULL; reply_dlen = 0; LDMA_DBG("VALIDATE_PATH(%s): return error %d", (path)? path : "<none>", status); } else { LDMA_DBG("VALIDATE_PATH(%s): return status=0x%x type=0x%x", path, reply->msg_info, *path_type); } free(path); *replyp = reply; *reply_dlenp = reply_dlen; return (status); } /* * We check that the device is a network interface (NIC) using libdladm. */ /*ARGSUSED*/ static ldma_request_status_t ldma_dev_validate_nic(ds_ver_t *ver, ldma_message_header_t *request, size_t request_dlen, ldma_message_header_t **replyp, size_t *reply_dlenp) { dladm_handle_t dlhandle; datalink_id_t linkid; uint32_t flag, media; datalink_class_t class; ldma_message_header_t *reply = NULL; ldma_request_status_t status; char *nic = NULL; uint32_t nlen, reply_dlen; nlen = request->msg_info; if (nlen == 0 || nlen > MAXPATHLEN || nlen > request_dlen) { status = LDMA_REQ_INVALID; goto done; } nic = malloc(nlen + 1); if (nic == NULL) { status = LDMA_REQ_FAILED; goto done; } (void) strncpy(nic, LDMA_HDR2DATA(request), nlen); nic[nlen] = '\0'; LDMA_DBG("VALIDATE_NIC(%s)", nic); reply_dlen = 0; reply = ldma_alloc_result_msg(request, reply_dlen); if (reply == NULL) { status = LDMA_REQ_FAILED; goto done; } reply->msg_info = 0x0; if (dladm_open(&dlhandle) != DLADM_STATUS_OK) { status = LDMA_REQ_FAILED; goto done; } if (dladm_name2info(dlhandle, nic, &linkid, &flag, &class, &media) != DLADM_STATUS_OK) { LDMA_DBG("VALIDATE_NIC(%s): name2info failed", nic); } else { LDMA_DBG("VALIDATE_NIC(%s): media=0x%x", nic, media); reply->msg_info = LDMA_DEVNIC_EXIST; } dladm_close(dlhandle); status = LDMA_REQ_COMPLETED; done: if (status != LDMA_REQ_COMPLETED) { /* * We don't provide a reply message if the request has not * been completed. The LDoms agent daemon will send an * appropriate reply based on the return code of this function. */ free(reply); reply = NULL; reply_dlen = 0; LDMA_DBG("VALIDATE_NIC(%s): return error %d", (nic)? nic : "<none>", status); } else { LDMA_DBG("VALIDATE_NIC(%s): return status=0x%x", nic, reply->msg_info); } free(nic); *replyp = reply; *reply_dlenp = reply_dlen; return (status); }