summaryrefslogtreecommitdiff
path: root/usr/src/lib/libfcoe/common/libfcoe.c
diff options
context:
space:
mode:
authorZhong Wang <Zhong.Wang@Sun.COM>2009-03-19 03:03:46 +0800
committerZhong Wang <Zhong.Wang@Sun.COM>2009-03-19 03:03:46 +0800
commit2a8164df8a5f42c8a00f10c67d7bc84f80ae9c41 (patch)
tree5ef9c65f09194743324b20261093e076d4eeea6b /usr/src/lib/libfcoe/common/libfcoe.c
parentbfe6f8f50e1ad7cfc72f4665989dc9e25e82e872 (diff)
downloadillumos-gate-2a8164df8a5f42c8a00f10c67d7bc84f80ae9c41.tar.gz
PSARC/2008/310 FCoE (Fibre Channel over Ethernet) Target
6701027 Need FCoE target for Solaris 6812750 Change in stmf/fct/stmf_sbd needed for support FCoE
Diffstat (limited to 'usr/src/lib/libfcoe/common/libfcoe.c')
-rw-r--r--usr/src/lib/libfcoe/common/libfcoe.c347
1 files changed, 347 insertions, 0 deletions
diff --git a/usr/src/lib/libfcoe/common/libfcoe.c b/usr/src/lib/libfcoe/common/libfcoe.c
new file mode 100644
index 0000000000..f06a620f26
--- /dev/null
+++ b/usr/src/lib/libfcoe/common/libfcoe.c
@@ -0,0 +1,347 @@
+/*
+ * 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.
+ */
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <wchar.h>
+#include <strings.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <libintl.h>
+#include <errno.h>
+#include <string.h>
+#include <assert.h>
+#include <syslog.h>
+#include <libfcoe.h>
+#include <fcoeio.h>
+
+#define FCOE_DEV_PATH "/devices/fcoe:admin"
+
+#define OPEN_FCOE 0
+#define OPEN_EXCL_FCOE O_EXCL
+
+/*
+ * Open for fcoe module
+ *
+ * flag - open flag (OPEN_FCOE, OPEN_EXCL_FCOE)
+ * fd - pointer to integer. On success, contains the fcoe file descriptor
+ */
+static int
+openFcoe(int flag, int *fd)
+{
+ int ret = FCOE_STATUS_ERROR;
+
+ if ((*fd = open(FCOE_DEV_PATH, O_NDELAY | O_RDONLY | flag)) != -1) {
+ ret = FCOE_STATUS_OK;
+ } else {
+ if (errno == EPERM || errno == EACCES) {
+ ret = FCOE_STATUS_ERROR_PERM;
+ } else {
+ ret = FCOE_STATUS_ERROR_OPEN_DEV;
+ }
+ syslog(LOG_DEBUG, "openFcoe:open failure:%s:errno(%d)",
+ FCOE_DEV_PATH, errno);
+ }
+
+ return (ret);
+}
+
+static int
+isWWNZero(FCOE_PORT_WWN portwwn)
+{
+ int i;
+ int size = sizeof (FCOE_PORT_WWN);
+
+ for (i = 0; i < size; i++) {
+ if (portwwn.wwn[i] != 0) {
+ return (0);
+ }
+ }
+ return (1);
+}
+
+FCOE_STATUS
+FCOE_CreatePort(
+ const FCOE_UINT8 *macLinkName,
+ FCOE_UINT8 portType,
+ FCOE_PORT_WWN pwwn,
+ FCOE_PORT_WWN nwwn,
+ FCOE_UINT8 promiscuous)
+{
+ FCOE_STATUS status = FCOE_STATUS_OK;
+ int fcoe_fd;
+ fcoeio_t fcoeio;
+ fcoeio_create_port_param_t param;
+
+ bzero(&param, sizeof (fcoeio_create_port_param_t));
+
+ if (macLinkName == NULL) {
+ return (FCOE_STATUS_ERROR_INVAL_ARG);
+ }
+
+ if (portType != FCOE_PORTTYPE_INITIATOR &&
+ portType != FCOE_PORTTYPE_TARGET) {
+ return (FCOE_STATUS_ERROR_INVAL_ARG);
+ }
+
+ if (!isWWNZero(pwwn)) {
+ param.fcp_pwwn_provided = 1;
+ bcopy(pwwn.wwn, param.fcp_pwwn, 8);
+ }
+
+ if (!isWWNZero(nwwn)) {
+ param.fcp_nwwn_provided = 1;
+ bcopy(nwwn.wwn, param.fcp_nwwn, 8);
+ }
+
+ if (param.fcp_pwwn_provided == 1 &&
+ param.fcp_nwwn_provided == 1 &&
+ bcmp(&pwwn, &nwwn, 8) == 0) {
+ return (FCOE_STATUS_ERROR_WWN_SAME);
+ }
+
+ if (strlen((char *)macLinkName) > FCOE_MAX_MAC_NAME_LEN-1) {
+ return (FCOE_STATUS_ERROR_MAC_LEN);
+ }
+
+ param.fcp_force_promisc = promiscuous;
+ (void) strcpy((char *)param.fcp_mac_name, (char *)macLinkName);
+ param.fcp_port_type = (fcoe_cli_type_t)portType;
+
+ if ((status = openFcoe(OPEN_FCOE, &fcoe_fd)) != FCOE_STATUS_OK) {
+ return (status);
+ }
+
+ (void) memset(&fcoeio, 0, sizeof (fcoeio));
+ fcoeio.fcoeio_cmd = FCOEIO_CREATE_FCOE_PORT;
+
+ fcoeio.fcoeio_ilen = sizeof (param);
+ fcoeio.fcoeio_xfer = FCOEIO_XFER_WRITE;
+ fcoeio.fcoeio_ibuf = (uintptr_t)&param;
+
+ if (ioctl(fcoe_fd, FCOEIO_CMD, &fcoeio) != 0) {
+ switch (fcoeio.fcoeio_status) {
+ case FCOEIOE_INVAL_ARG:
+ status = FCOE_STATUS_ERROR_INVAL_ARG;
+ break;
+
+ case FCOEIOE_BUSY:
+ status = FCOE_STATUS_ERROR_BUSY;
+ break;
+
+ case FCOEIOE_ALREADY:
+ status = FCOE_STATUS_ERROR_ALREADY;
+ break;
+
+ case FCOEIOE_PWWN_CONFLICTED:
+ status = FCOE_STATUS_ERROR_PWWN_CONFLICTED;
+ break;
+
+ case FCOEIOE_NWWN_CONFLICTED:
+ status = FCOE_STATUS_ERROR_NWWN_CONFLICTED;
+ break;
+
+ case FCOEIOE_CREATE_MAC:
+ status = FCOE_STATUS_ERROR_CREATE_MAC;
+ break;
+
+ case FCOEIOE_OPEN_MAC:
+ status = FCOE_STATUS_ERROR_OPEN_MAC;
+ break;
+
+ case FCOEIOE_CREATE_PORT:
+ status = FCOE_STATUS_ERROR_CREATE_PORT;
+ break;
+
+ case FCOEIOE_NEED_JUMBO_FRAME:
+ status = FCOE_STATUS_ERROR_NEED_JUMBO_FRAME;
+ break;
+
+ case FCOEIOE_VNIC_UNSUPPORT:
+ status = FCOE_STATUS_ERROR_VNIC_UNSUPPORT;
+ break;
+
+ default:
+ status = FCOE_STATUS_ERROR;
+ }
+ } else {
+ status = FCOE_STATUS_OK;
+ }
+ (void) close(fcoe_fd);
+ return (status);
+}
+
+FCOE_STATUS
+FCOE_DeletePort(const FCOE_UINT8 *macLinkName)
+{
+ FCOE_STATUS status = FCOE_STATUS_OK;
+ int fcoe_fd;
+ fcoeio_t fcoeio;
+
+ if (macLinkName == NULL) {
+ return (FCOE_STATUS_ERROR_INVAL_ARG);
+ }
+
+ if (strlen((char *)macLinkName) > FCOE_MAX_MAC_NAME_LEN-1) {
+ return (FCOE_STATUS_ERROR_MAC_LEN);
+ }
+
+ if ((status = openFcoe(OPEN_FCOE, &fcoe_fd)) != FCOE_STATUS_OK) {
+ return (status);
+ }
+
+ (void) memset(&fcoeio, 0, sizeof (fcoeio));
+ fcoeio.fcoeio_cmd = FCOEIO_DELETE_FCOE_PORT;
+
+ fcoeio.fcoeio_ilen = strlen((char *)macLinkName)+1;
+ fcoeio.fcoeio_xfer = FCOEIO_XFER_WRITE;
+ fcoeio.fcoeio_ibuf = (uintptr_t)macLinkName;
+
+ if (ioctl(fcoe_fd, FCOEIO_CMD, &fcoeio) != 0) {
+ switch (fcoeio.fcoeio_status) {
+ case FCOEIOE_INVAL_ARG:
+ status = FCOE_STATUS_ERROR_INVAL_ARG;
+ break;
+
+ case FCOEIOE_BUSY:
+ status = FCOE_STATUS_ERROR_BUSY;
+ break;
+
+ case FCOEIOE_ALREADY:
+ status = FCOE_STATUS_ERROR_ALREADY;
+ break;
+
+ case FCOEIOE_MAC_NOT_FOUND:
+ status = FCOE_STATUS_ERROR_MAC_NOT_FOUND;
+ break;
+
+ case FCOEIOE_OFFLINE_FAILURE:
+ status = FCOE_STATUS_ERROR_OFFLINE_DEV;
+ break;
+
+ default:
+ status = FCOE_STATUS_ERROR;
+ }
+ } else {
+ status = FCOE_STATUS_OK;
+ }
+ (void) close(fcoe_fd);
+ return (status);
+}
+
+FCOE_STATUS
+FCOE_GetPortList(
+ FCOE_UINT32 *port_num,
+ FCOE_PORT_ATTRIBUTE **portlist)
+{
+ FCOE_STATUS status = FCOE_STATUS_OK;
+ int fcoe_fd;
+ fcoeio_t fcoeio;
+ fcoe_port_list_t *inportlist = NULL;
+ FCOE_PORT_ATTRIBUTE *outportlist = NULL;
+ int i;
+ int size = 64; /* default first attempt */
+ int retry = 0;
+ int bufsize;
+
+ if (port_num == NULL || portlist == NULL) {
+ return (FCOE_STATUS_ERROR_INVAL_ARG);
+ }
+ *port_num = 0;
+
+ if ((status = openFcoe(OPEN_FCOE, &fcoe_fd)) != FCOE_STATUS_OK) {
+ return (status);
+ }
+
+ /* Get fcoe port list */
+ (void) memset(&fcoeio, 0, sizeof (fcoeio));
+ retry = 0;
+
+ do {
+ bufsize = sizeof (fcoe_port_instance_t) * (size - 1) +
+ sizeof (fcoe_port_list_t);
+ inportlist = (fcoe_port_list_t *)malloc(bufsize);
+ fcoeio.fcoeio_cmd = FCOEIO_GET_FCOE_PORT_LIST;
+ fcoeio.fcoeio_olen = bufsize;
+ fcoeio.fcoeio_xfer = FCOEIO_XFER_READ;
+ fcoeio.fcoeio_obuf = (uintptr_t)inportlist;
+
+ if (ioctl(fcoe_fd, FCOEIO_CMD, &fcoeio) != 0) {
+ if (fcoeio.fcoeio_status == FCOEIOE_MORE_DATA) {
+ size = inportlist->numPorts;
+ }
+ free(inportlist);
+ switch (fcoeio.fcoeio_status) {
+ case FCOEIOE_INVAL_ARG:
+ status = FCOE_STATUS_ERROR_INVAL_ARG;
+ (void) close(fcoe_fd);
+ return (status);
+
+ case FCOEIOE_BUSY:
+ status = FCOE_STATUS_ERROR_BUSY;
+ retry++;
+ break;
+
+ case FCOEIOE_MORE_DATA:
+ status = FCOE_STATUS_ERROR_MORE_DATA;
+ retry++;
+ default:
+ status = FCOE_STATUS_ERROR;
+ }
+ } else {
+ status = FCOE_STATUS_OK;
+ break;
+ }
+ } while (retry <= 3 && status != FCOE_STATUS_OK);
+
+ if (status == FCOE_STATUS_OK) {
+ outportlist = (PFCOE_PORT_ATTRIBUTE)
+ malloc(sizeof (FCOE_PORT_ATTRIBUTE) * inportlist->numPorts);
+
+ for (i = 0; i < inportlist->numPorts; i++) {
+ fcoe_port_instance_t *pi = &inportlist->ports[i];
+ FCOE_PORT_ATTRIBUTE *po = &outportlist[i];
+ bcopy(pi->fpi_pwwn, &po->port_wwn, 8);
+ bcopy(pi->fpi_mac_link_name, po->mac_link_name, 32);
+ bcopy(pi->fpi_mac_factory_addr,
+ po->mac_factory_addr, 6);
+ bcopy(pi->fpi_mac_current_addr,
+ po->mac_current_addr, 6);
+ po->port_type = (FCOE_UINT8)pi->fpi_port_type;
+ po->mtu_size = pi->fpi_mtu_size;
+ po->mac_promisc = pi->fpi_mac_promisc;
+ }
+ *port_num = inportlist->numPorts;
+ *portlist = outportlist;
+ free(inportlist);
+ } else {
+ *port_num = 0;
+ *portlist = NULL;
+ }
+ (void) close(fcoe_fd);
+ return (status);
+}