/* * 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. */ #pragma ident "%Z%%M% %I% %E% SMI" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static int mod_enum(topo_mod_t *, tnode_t *, const char *, topo_instance_t, topo_instance_t, void *, void *); static void mod_release(topo_mod_t *, tnode_t *); static int mod_fmri_create_meth(topo_mod_t *, tnode_t *, topo_version_t, nvlist_t *, nvlist_t **); static int mod_fmri_nvl2str(topo_mod_t *, tnode_t *, topo_version_t, nvlist_t *, nvlist_t **); static const topo_method_t mod_methods[] = { { TOPO_METH_FMRI, TOPO_METH_FMRI_DESC, TOPO_METH_FMRI_VERSION, TOPO_STABILITY_INTERNAL, mod_fmri_create_meth }, { TOPO_METH_NVL2STR, TOPO_METH_NVL2STR_DESC, TOPO_METH_NVL2STR_VERSION, TOPO_STABILITY_INTERNAL, mod_fmri_nvl2str }, { NULL } }; static const topo_modops_t mod_modops = { mod_enum, mod_release }; static const topo_modinfo_t mod_info = { "mod", FM_FMRI_SCHEME_MOD, MOD_VERSION, &mod_modops }; int mod_init(topo_mod_t *mod, topo_version_t version) { if (getenv("TOPOMODDEBUG")) topo_mod_setdebug(mod); topo_mod_dprintf(mod, "initializing mod builtin\n"); if (version != MOD_VERSION) return (topo_mod_seterrno(mod, EMOD_VER_NEW)); if (topo_mod_register(mod, &mod_info, TOPO_VERSION) != 0) { topo_mod_dprintf(mod, "failed to register mod_info: " "%s\n", topo_mod_errmsg(mod)); return (-1); /* mod errno already set */ } return (0); } void mod_fini(topo_mod_t *mod) { topo_mod_unregister(mod); } /*ARGSUSED*/ static int mod_enum(topo_mod_t *mod, tnode_t *pnode, const char *name, topo_instance_t min, topo_instance_t max, void *notused1, void *notused2) { (void) topo_method_register(mod, pnode, mod_methods); return (0); } static void mod_release(topo_mod_t *mod, tnode_t *node) { topo_method_unregister_all(mod, node); } static int mod_binary_path_get(topo_mod_t *mp, const char *objpath) { Elf *elf = NULL; Elf_Scn *scn = NULL; GElf_Ehdr ehdr; GElf_Shdr shdr; int fd; if ((fd = open(objpath, O_RDONLY)) < 0) { topo_mod_dprintf(mp, "unable to open %s\n", objpath); return (-1); } if (elf_version(EV_CURRENT) == EV_NONE) { topo_mod_dprintf(mp, "Elf version out of whack\n"); goto mbpg_bail; } if ((elf = elf_begin(fd, ELF_C_READ, NULL)) == NULL) { topo_mod_dprintf(mp, "elf_begin failed\n"); goto mbpg_bail; } if ((gelf_getehdr(elf, &ehdr)) == NULL) { topo_mod_dprintf(mp, "gelf_getehdr failed\n"); goto mbpg_bail; } scn = elf_getscn(elf, 0); /* "seek" to start of sections */ while ((scn = elf_nextscn(elf, scn)) != NULL) { const char *sh_name; if (gelf_getshdr(scn, &shdr) == NULL) { topo_mod_dprintf(mp, "gelf_getshdr failed\n"); goto mbpg_bail; } if (shdr.sh_type != SHT_PROGBITS) continue; sh_name = elf_strptr(elf, ehdr.e_shstrndx, (size_t)shdr.sh_name); if (strcmp(sh_name, ".filename") != 0) continue; if (elf_getdata(scn, NULL) == NULL) { topo_mod_dprintf(mp, "no filename data"); break; } break; } elf_end(elf); (void) close(fd); return (0); mbpg_bail: if (elf != NULL) elf_end(elf); if (fd >= 0) (void) close(fd); (void) topo_mod_seterrno(mp, EMOD_METHOD_INVAL); return (-1); } static int mod_nvl_data(topo_mod_t *mp, nvlist_t *out, const char *path) { struct modinfo mi; struct stat64 s; int id, e; if (stat64(path, &s) < 0) { topo_mod_dprintf(mp, "No system object file for driver %s", path); return (topo_mod_seterrno(mp, EMOD_METHOD_INVAL)); } id = OBJFS_MODID(s.st_ino); mi.mi_id = mi.mi_nextid = id; mi.mi_info = MI_INFO_ONE | MI_INFO_NOBASE; if (modctl(MODINFO, id, &mi) < 0) { return (topo_mod_seterrno(mp, EMOD_METHOD_INVAL)); } mi.mi_name[MODMAXNAMELEN - 1] = '\0'; mi.mi_msinfo[0].msi_linkinfo[MODMAXNAMELEN - 1] = '\0'; e = nvlist_add_string(out, FM_FMRI_SCHEME, FM_FMRI_SCHEME_MOD); e |= nvlist_add_uint8(out, FM_VERSION, FM_MOD_SCHEME_VERSION); e |= nvlist_add_int32(out, FM_FMRI_MOD_ID, id); e |= nvlist_add_string(out, FM_FMRI_MOD_NAME, mi.mi_name); e |= nvlist_add_string(out, FM_FMRI_MOD_DESC, mi.mi_msinfo[0].msi_linkinfo); if (e != 0) return (topo_mod_seterrno(mp, EMOD_FMRI_NVL)); return (0); } static nvlist_t * mod_fmri_create(topo_mod_t *mp, const char *driver) { nvlist_t *out = NULL; char objpath[PATH_MAX]; if (topo_mod_nvalloc(mp, &out, NV_UNIQUE_NAME) != 0) { (void) topo_mod_seterrno(mp, EMOD_FMRI_NVL); goto mfc_bail; } (void) snprintf(objpath, PATH_MAX, "%s/%s/object", OBJFS_ROOT, driver); /* * Validate the module object ELF header if possible */ if (mod_binary_path_get(mp, objpath) < 0) goto mfc_bail; if (mod_nvl_data(mp, out, objpath) < 0) { topo_mod_dprintf(mp, "failed to get modinfo for %s", driver); goto mfc_bail; } return (out); mfc_bail: nvlist_free(out); return (NULL); } /*ARGSUSED*/ static int mod_fmri_create_meth(topo_mod_t *mp, tnode_t *node, topo_version_t version, nvlist_t *in, nvlist_t **out) { nvlist_t *args; nvlist_t *modnvl; char *driver; if (version > TOPO_METH_FMRI_VERSION) return (topo_mod_seterrno(mp, EMOD_VER_NEW)); if (nvlist_lookup_nvlist(in, TOPO_METH_FMRI_ARG_NVL, &args) != 0 || nvlist_lookup_string(args, "DRIVER", &driver) != 0) { topo_mod_dprintf(mp, "no DRIVER string in method argument\n"); return (topo_mod_seterrno(mp, EMOD_METHOD_INVAL)); } modnvl = mod_fmri_create(mp, driver); if (modnvl == NULL) { *out = NULL; topo_mod_dprintf(mp, "failed to create contained mod FMRI\n"); return (-1); } *out = modnvl; return (0); } #define MAXINTSTR 11 static ssize_t fmri_nvl2str(nvlist_t *nvl, char *buf, size_t buflen) { nvlist_t *anvl = NULL; uint8_t version; ssize_t size = 0; int32_t modid; char *achas = NULL; char *adom = NULL; char *aprod = NULL; char *asrvr = NULL; char *ahost = NULL; char *modname = NULL; char numbuf[MAXINTSTR]; int err; if (nvlist_lookup_uint8(nvl, FM_VERSION, &version) != 0 || version > FM_MOD_SCHEME_VERSION) return (-1); /* Get authority, if present */ err = nvlist_lookup_nvlist(nvl, FM_FMRI_AUTHORITY, &anvl); if (err != 0 && err != ENOENT) return (-1); /* * For brevity, we only include the module name and id * present in the FMRI in our output string. The FMRI * also has data on the package containing the module. */ /* There must be a module name */ err = nvlist_lookup_string(nvl, FM_FMRI_MOD_NAME, &modname); if (err != 0 || modname == NULL) return (-1); /* There must be a module id */ err = nvlist_lookup_int32(nvl, FM_FMRI_MOD_ID, &modid); if (err != 0) return (-1); if (anvl != NULL) { (void) nvlist_lookup_string(anvl, FM_FMRI_AUTH_PRODUCT, &aprod); (void) nvlist_lookup_string(anvl, FM_FMRI_AUTH_CHASSIS, &achas); (void) nvlist_lookup_string(anvl, FM_FMRI_AUTH_DOMAIN, &adom); (void) nvlist_lookup_string(anvl, FM_FMRI_AUTH_SERVER, &asrvr); (void) nvlist_lookup_string(anvl, FM_FMRI_AUTH_HOST, &ahost); } /* mod:// */ topo_fmristr_build(&size, buf, buflen, FM_FMRI_SCHEME_MOD, NULL, "://"); /* authority, if any */ if (aprod != NULL) topo_fmristr_build(&size, buf, buflen, aprod, ":" FM_FMRI_AUTH_PRODUCT "=", NULL); if (achas != NULL) topo_fmristr_build(&size, buf, buflen, achas, ":" FM_FMRI_AUTH_CHASSIS "=", NULL); if (adom != NULL) topo_fmristr_build(&size, buf, buflen, adom, ":" FM_FMRI_AUTH_DOMAIN "=", NULL); if (asrvr != NULL) topo_fmristr_build(&size, buf, buflen, asrvr, ":" FM_FMRI_AUTH_SERVER "=", NULL); if (ahost != NULL) topo_fmristr_build(&size, buf, buflen, ahost, ":" FM_FMRI_AUTH_HOST "=", NULL); /* module parts */ topo_fmristr_build(&size, buf, buflen, modname, "/" FM_FMRI_MOD_NAME "=", "/"); (void) snprintf(numbuf, MAXINTSTR, "%d", modid); topo_fmristr_build(&size, buf, buflen, numbuf, FM_FMRI_MOD_ID "=", NULL); return (size); } /*ARGSUSED*/ static int mod_fmri_nvl2str(topo_mod_t *mod, tnode_t *node, topo_version_t version, nvlist_t *nvl, nvlist_t **out) { ssize_t len; char *name = NULL; nvlist_t *fmristr; if (version > TOPO_METH_NVL2STR_VERSION) return (topo_mod_seterrno(mod, EMOD_VER_NEW)); if ((len = fmri_nvl2str(nvl, NULL, 0)) == 0 || (name = topo_mod_alloc(mod, len + 1)) == NULL || fmri_nvl2str(nvl, name, len + 1) == 0) { if (name != NULL) topo_mod_free(mod, name, len + 1); return (topo_mod_seterrno(mod, EMOD_FMRI_NVL)); } if (topo_mod_nvalloc(mod, &fmristr, NV_UNIQUE_NAME) != 0) return (topo_mod_seterrno(mod, EMOD_FMRI_NVL)); if (nvlist_add_string(fmristr, "fmri-string", name) != 0) { topo_mod_free(mod, name, len + 1); nvlist_free(fmristr); return (topo_mod_seterrno(mod, EMOD_FMRI_NVL)); } topo_mod_free(mod, name, len + 1); *out = fmristr; return (0); }