/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (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 2002-2003 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" #include static int extract_label_info(ctf_file_t *fp, const ctf_lblent_t **ctl, uint_t *num_labels) { const ctf_header_t *h; /* * Labels are only supported in V2 or later */ if (fp->ctf_version < CTF_VERSION_2) return (ctf_set_errno(fp, ECTF_NOTSUP)); h = (const ctf_header_t *)fp->ctf_data.cts_data; /* LINTED - pointer alignment */ *ctl = (const ctf_lblent_t *)(fp->ctf_buf + h->cth_lbloff); *num_labels = (h->cth_objtoff - h->cth_lbloff) / sizeof (ctf_lblent_t); return (0); } /* * Returns the topmost label, or NULL if any errors are encountered */ const char * ctf_label_topmost(ctf_file_t *fp) { const ctf_lblent_t *ctlp; const char *s; uint_t num_labels; if (extract_label_info(fp, &ctlp, &num_labels) == CTF_ERR) return (NULL); /* errno is set */ if (num_labels == 0) { (void) ctf_set_errno(fp, ECTF_NOLABELDATA); return (NULL); } if ((s = ctf_strraw(fp, (ctlp + num_labels - 1)->ctl_label)) == NULL) (void) ctf_set_errno(fp, ECTF_CORRUPT); return (s); } /* * Iterate over all labels. We pass the label string and the lblinfo_t struct * to the specified callback function. */ int ctf_label_iter(ctf_file_t *fp, ctf_label_f *func, void *arg) { const ctf_lblent_t *ctlp; uint_t i, num_labels; ctf_lblinfo_t linfo; const char *lname; int rc; if (extract_label_info(fp, &ctlp, &num_labels) == CTF_ERR) return (CTF_ERR); /* errno is set */ if (num_labels == 0) return (ctf_set_errno(fp, ECTF_NOLABELDATA)); for (i = 0; i < num_labels; i++, ctlp++) { if ((lname = ctf_strraw(fp, ctlp->ctl_label)) == NULL) { ctf_dprintf("failed to decode label %u with " "typeidx %u\n", ctlp->ctl_label, ctlp->ctl_typeidx); return (ctf_set_errno(fp, ECTF_CORRUPT)); } linfo.ctb_typeidx = ctlp->ctl_typeidx; if ((rc = func(lname, &linfo, arg)) != 0) return (rc); } return (0); } typedef struct linfo_cb_arg { const char *lca_name; /* Label we want to retrieve info for */ ctf_lblinfo_t *lca_info; /* Where to store the info about the label */ } linfo_cb_arg_t; static int label_info_cb(const char *lname, const ctf_lblinfo_t *linfo, void *arg) { /* * If lname matches the label we are looking for, copy the * lblinfo_t struct for the caller. */ if (strcmp(lname, ((linfo_cb_arg_t *)arg)->lca_name) == 0) { /* * Allow caller not to allocate storage to test if label exists */ if (((linfo_cb_arg_t *)arg)->lca_info != NULL) bcopy(linfo, ((linfo_cb_arg_t *)arg)->lca_info, sizeof (ctf_lblinfo_t)); return (1); /* Indicate we found a match */ } return (0); } /* * Retrieve information about the label with name "lname" */ int ctf_label_info(ctf_file_t *fp, const char *lname, ctf_lblinfo_t *linfo) { linfo_cb_arg_t cb_arg; int rc; cb_arg.lca_name = lname; cb_arg.lca_info = linfo; if ((rc = ctf_label_iter(fp, label_info_cb, &cb_arg)) == CTF_ERR) return (rc); if (rc != 1) return (ctf_set_errno(fp, ECTF_NOLABEL)); return (0); }