/* * 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 2004 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* Copyright (c) 1988 AT&T */ /* All Rights Reserved */ #pragma ident "%Z%%M% %I% %E% SMI" /* * C++ Demangler Source Code * @(#)master 1.5 * 7/27/88 13:54:37 */ #include #include #include #include #include #include "elf_dem.h" #include "String.h" /* * This code emulates the C++ String package * in a crude way. */ jmp_buf jbuf; /* * This function will expand the space * available to a String so that more data * can be appended to it */ static String * grow(s) String *s; { String *ns; int sz = s->sg.max * 2; assert(sz > 0); #ifdef ELF if ((ns = (String *)malloc(sz + sizeof (StringGuts) + 1)) == NULL) longjmp(jbuf, 1); (void) memcpy(ns, s, s->sg.max + sizeof (StringGuts) + 1); free(s); #else if ((ns = (String *)realloc(s, sz + sizeof (StringGuts) + 1)) == NULL) longjmp(jbuf, 1); #endif ns->sg.max = sz; return (ns); } /* * This function will expand the space * available to a String so that more data * can be prepended to it. */ static String * ror(s, n) String *s; int n; { assert(s != 0); while (s->sg.end + n > s->sg.max) s = grow(s); #ifdef __STDC__ assert(n >= 0); assert(s->sg.end >= s->sg.start); (void) memmove(s->data + n, s->data, s->sg.end - s->sg.start); #else { int i; for (i = s->sg.end - 1; i >= s->sg.start; i--) s->data[i+n] = s->data[i]; } #endif s->sg.end += n; s->sg.start += n; s->data[s->sg.end] = 0; return (s); } /* * This function will prepend c * to s */ String * prep_String(c, s) char *c; String *s; { return (nprep_String(c, s, ID_NAME_MAX)); } /* * This function will prepend the * first n characters of c to s */ String * nprep_String(c, s, n) const char *c; String *s; int n; { int len = strlen(c); assert(s != 0); if (len > n) len = n; if (len > s->sg.start) s = ror(s, len - s->sg.start); s->sg.start -= len; (void) memcpy(s->data + s->sg.start, c, len); return (s); } /* * This function will append * c to s. */ String * app_String(s, c) String *s; const char *c; { return (napp_String(s, c, ID_NAME_MAX)); } /* * This function will append the * first n characters of c to s */ String * napp_String(String *s, const char *c, int n) { int len = strlen(c); int catlen; assert(s != 0); if (n < len) len = n; catlen = s->sg.end + len; while (catlen > s->sg.max) s = grow(s); (void) memcpy(s->data + s->sg.end, c, len); s->sg.end += len; s->data[s->sg.end] = '\0'; return (s); } /* * This function initializes a * String. It returns its argument if * its argument is non-zero. * This prevents the same string * from being re-initialized. */ String * mk_String(s) String *s; { if (s) return (s); s = (String *)malloc(STRING_START + sizeof (StringGuts) + 1); if (s == NULL) longjmp(jbuf, 1); s->sg.start = s->sg.end = STRING_START/2; s->sg.max = STRING_START; s->data[s->sg.end] = '\0'; return (s); } void free_String(s) String *s; { if (s) free(s); } /* * This function copies * c into s. * Used for initialization. */ String * set_String(s, c) String *s; char *c; { int len = strlen(c)*2; while (len > s->sg.max) s = grow(s); s->sg.start = s->sg.end = s->sg.max / 2; s = app_String(s, c); return (s); } /* * Chop n characters off the end of a string. * Return the truncated string. */ String * trunc_String(String *s, int n) { assert(n <= s->sg.end - s->sg.start); s->sg.end -= n; s->data[s->sg.end] = '\0'; return (s); }