/* * 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 2008 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #include <stdlib.h> #include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <errno.h> #include <string.h> #include <stdarg.h> #include <syslog.h> #include <libdevice.h> #include <sys/fibre-channel/fcio.h> #include "common.h" static int parse_line(char *line, char *path, char *wwn, char *filename); static int create_ap_instance(char *ap_id, char *wwn_string, char *filename, char *line); static void log_error(char *msg_id, char *input_tmplt, ...); static char ctoi(char c); /* * Simple wrapper for syslog error messages. * Allows easy addition of syserr output if desired. */ static void log_error(char *msg_id, char *input_tmplt, ...) { va_list ap; char input_merged_msg[200]; char *msg_template = "ID[luxadm.create_fabric_device.%s] %s"; /* * First %s for msg_id in merged msg. * Second %s is for input merged_msg */ char *merged_msg; va_start(ap, input_tmplt); /* insert caller's args */ (void) vsprintf(input_merged_msg, input_tmplt, ap); va_end(ap); merged_msg = (char *)malloc(strlen(msg_template) + strlen(input_merged_msg) + strlen(msg_id) + 1); if (merged_msg == NULL) { syslog(LOG_ERR, "ID[luxadm.create_fabric_device.2317] " "malloc failure, %s", strerror(errno)); } else { sprintf(merged_msg, msg_template, msg_id, input_merged_msg); /* first insert msg_id */ syslog(LOG_ERR, merged_msg, ""); (void) puts(merged_msg); /* also print message */ free(merged_msg); } } /* * Routines for reading tapestry repository file */ #define COMMENT_CHAR '#' int read_repos_file(char *repos_filename) { int fd; char *line; char *tmp_ptr, *mmap_ptr; char path[MAXPATHLEN]; int ret; char wwn[FC_WWN_SIZE*2+1]; struct stat stbuf; unsigned int filesize; unsigned int bytes_read; if (repos_filename == NULL || *repos_filename == NULL) { log_error("2310", "filename missing for -f option of " "luxadm -e create_fabric_device"); return (-1); } fd = open(repos_filename, O_RDONLY); if (fd == -1) { log_error("2311", "fopen failed: cannot open repository file %s. %d", repos_filename, strerror(errno)); return (-1); } if (fstat(fd, &stbuf) == -1) { close(fd); log_error("2312", "stat failed on file %s. %s", repos_filename, strerror(errno)); return (-1); } filesize = stbuf.st_size; tmp_ptr = mmap_ptr = mmap((caddr_t)0, filesize, (PROT_READ | PROT_WRITE), MAP_PRIVATE, fd, 0); if (mmap_ptr == MAP_FAILED) { log_error("2315", "Failed to mmap file %s. %s", repos_filename, strerror(errno)); return (-1); } bytes_read = 0; while (bytes_read < filesize) { line = tmp_ptr; while (bytes_read < filesize && *tmp_ptr != '\n') { bytes_read++; tmp_ptr++; } if (*tmp_ptr == '\n') { *tmp_ptr = NULL; tmp_ptr++; bytes_read++; } /* If the line is a comment, read another line */ if (*line == COMMENT_CHAR) { continue; } ret = parse_line(line, path, wwn, repos_filename); if (ret == 0) { ret = create_ap_instance(path, wwn, repos_filename, line); } } ret = close(fd); ret = munmap(mmap_ptr, filesize); return (ret); } /* * Input is paramater 1 - a line from repository * Output is other parameters, the path to the attachment point, * and the port wwn are parsed from the repository * Format is * "/devices/pci..../fp@1,0:fc::wwn" * If controller name is missing, that's okay. Other fields * must be present * * Return 0 on success or -1 on failure; all failures logged to syslog. */ #define WWN_DELIM "::" static int parse_line(char *line, char *path, char *wwn, char *filename) { char *p_path, *p_wwn, *p_delim; char *line_copy; line_copy = strdup(line); if (line_copy == NULL) { log_error("2317", "malloc failure, %s", strerror(errno)); } p_path = line_copy; p_delim = strstr(p_path, WWN_DELIM); if (p_delim == NULL) { log_error("2313", "Invalid line (%s) in file %s.", line, filename); free(line_copy); return (-1); } *p_delim = NULL; /* NULL terminate path */ if (strlcpy(path, p_path, MAXPATHLEN) >= MAXPATHLEN) { log_error("2318", "Path too long (%s) in file %s.", p_path, filename); free(line_copy); return (-1); } p_wwn = p_delim + strlen(WWN_DELIM); /* * Now look for the blank delimiter before the controller * * This is just the case when there may be a controller # * after the attachment point and WWN. For example - * /devices/pci@b,2000/pci@2/SUNW,qlc@4/fp@0,0:fc::220000203707f4f1 c4 */ p_delim = strchr(p_wwn, ' '); if (p_delim != NULL) { /* now p_delim points to blank */ *p_delim = NULL; /* terminate wwn at delim */ } else { char *p_last_char; p_last_char = p_wwn+strlen(p_wwn)-1; if (*p_last_char == '\n') { *p_last_char = NULL; } } strcpy(wwn, p_wwn); free(line_copy); return (0); } static char ctoi(char c) { if ((c >= '0') && (c <= '9')) c -= '0'; else if ((c >= 'A') && (c <= 'F')) c = c - 'A' + 10; else if ((c >= 'a') && (c <= 'f')) c = c - 'a' + 10; else c = -1; return (c); } /* * "string" is Input and "port_wwn" has the output * * This function converts a string to WWN. * For example a string like * "220000203707F4F1" gets converted to 0x220000203707F4F1 ... * where * port_wwn[0] = 0x22, * port_wwn[1] = 0x00, * port_wwn[2] = 0x00, * port_wwn[3] = 0x20, * port_wwn[4] = 0x37, * port_wwn[5] = 0x07, * port_wwn[6] = 0xF4, and * port_wwn[7] = 0xF1 */ static int string_to_wwn(const uchar_t *string, uchar_t *port_wwn) { int i; char c, c1; uchar_t *wwnp; wwnp = port_wwn; for (i = 0; i < WWN_SIZE; i++, wwnp++) { c = ctoi(*string++); c1 = ctoi(*string++); if (c == -1 || c1 == -1) return (-1); *wwnp = ((c << 4) + c1); } return (0); } static int create_ap_instance(char *ap_id, char *wwn_string, char *filename, char *line) { devctl_hdl_t bus_handle, dev_handle; devctl_ddef_t ddef_handle; int ret; uchar_t wwn_array[FC_WWN_SIZE]; ddef_handle = devctl_ddef_alloc("dummy", 0); if (ddef_handle == NULL) { log_error("2314", "Internal error to process line (%s) " "in file: %s. %s", line, filename, strerror(errno)); return (-1); } /* * g_string_to_wwn() has not been used here because it * prepends 2 NULLs. */ if (string_to_wwn((uchar_t *)wwn_string, wwn_array) != 0) { log_error("2314", "Internal error to process line (%s) " "in file: %s. %s", line, filename, strerror(errno)); devctl_ddef_free(ddef_handle); return (-1); } (void) devctl_ddef_byte_array(ddef_handle, "port-wwn", FC_WWN_SIZE, wwn_array); if ((bus_handle = devctl_bus_acquire(ap_id, 0)) == NULL) { devctl_ddef_free(ddef_handle); log_error("2314", "Internal error to process line (%s) " "in file: %s. %s", line, filename, strerror(errno)); return (-1); } if (ret = devctl_bus_dev_create(bus_handle, ddef_handle, 0, &dev_handle)) { devctl_ddef_free(ddef_handle); devctl_release(bus_handle); log_error("2316", "configuration failed for line (%s) " "in file: %s. %s", line, filename, strerror(errno)); return (-1); } devctl_release(dev_handle); devctl_ddef_free(ddef_handle); devctl_release(bus_handle); return (ret); }