/* * 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 (c) 1998,2001 by Sun Microsystems, Inc. * All rights reserved. */ #pragma ident "%Z%%M% %I% %E% SMI" #ifdef MALLOC_DEBUG #include <stdlib.h> #include <stdio.h> #include <thread.h> #include <synch.h> #include <string.h> #include <stdio.h> #include <syslog.h> #include <netdb.h> #include <netdir.h> #include <rpc/nettype.h> /* * To use debugging facility, compile with * -DMALLOC_DEBUG. * You can do this by setting the environment variable * MALLOC_DEBUG to "-DMALLOC_DEBUG" * * To make automountd dump trace records (i.e. make it call check_leaks), * run: * make malloc_dump */ struct alloc_list { char type[20]; void *addr; int size; char file[80]; int line; struct alloc_list *next; }; static struct alloc_list *halist = NULL; static mutex_t alloc_list_lock = DEFAULTMUTEX; int add_alloc(char *type, void *addr, size_t size, const char *file, int line) { struct alloc_list *alist = NULL; /* allocate the list item */ alist = (struct alloc_list *)malloc(sizeof (*alist)); if (alist == NULL) { syslog(LOG_ERR, "add_alloc: out of memory\n"); return (-1); } strcpy(alist->type, type); alist->addr = addr; alist->size = size; strcpy(alist->file, file); alist->line = line; /* add it to the head of the list */ if (halist == NULL) alist->next = NULL; else alist->next = halist; halist = alist; return (0); } int drop_alloc(const char *type, void *addr, const char *file, int line) { struct alloc_list *alist, *alist_prev; alist = halist; while (alist != NULL) { if (addr == alist->addr) { if (alist == halist) halist = halist->next; else alist_prev->next = alist->next; free(alist); break; } alist_prev = alist; alist = alist->next; } if (alist == NULL) { syslog(LOG_ERR, "*** POSSIBLE CORRUPTION ****\n"); syslog(LOG_ERR, "\tduplicate free, type %s, at %p in %s/%d\n", type, addr, file, line); return (-1); } return (0); } void * my_malloc(size_t size, const char *file, int line) { void *addr; addr = (void *)malloc(size); if (addr == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("MALLOC", addr, size, file, line); mutex_unlock(&alloc_list_lock); return (addr); } void * my_realloc(void *addr, size_t size, const char *file, int line) { void *ptr; ptr = (void *)realloc(addr, size); if (ptr == NULL) return (NULL); mutex_lock(&alloc_list_lock); drop_alloc("MALLOC", addr, file, line); add_alloc("MALLOC", ptr, size, file, line); mutex_unlock(&alloc_list_lock); return (ptr); } void my_free(void *addr, const char *file, int line) { mutex_lock(&alloc_list_lock); drop_alloc("MALLOC", addr, file, line); mutex_unlock(&alloc_list_lock); free(addr); } char * my_strdup(const char *straddr, const char *file, int line) { void *addr; size_t size; addr = strdup(straddr); if (addr == NULL) return (NULL); size = strlen(straddr); mutex_lock(&alloc_list_lock); add_alloc("STRDUP", addr, size, file, line); mutex_unlock(&alloc_list_lock); return ((char *)addr); } int my_sethostent(int stay, const char *file, int line) { (void) sethostent(stay); mutex_lock(&alloc_list_lock); add_alloc("SETHOSTENT", NULL, 0, file, line); mutex_unlock(&alloc_list_lock); return (0); } int my_endhostent(const char *file, int line) { int ret; ret = endhostent(); if (ret != 0) return (ret); mutex_lock(&alloc_list_lock); drop_alloc("SETHOSTENT", NULL, file, line); mutex_unlock(&alloc_list_lock); return (ret); } void * my_setnetconfig(const char *file, int line) { void *nconf; nconf = setnetconfig(); if (nconf == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("SETNETCONFIG", nconf, 0, file, line); mutex_unlock(&alloc_list_lock); return (nconf); } int my_endnetconfig(void *nconf, const char *file, int line) { int res; res = endnetconfig(nconf); if (res != 0) return (res); mutex_lock(&alloc_list_lock); drop_alloc("SETNETCONFIG", nconf, file, line); mutex_unlock(&alloc_list_lock); return (0); } void * my_setnetpath(const char *file, int line) { void *npath; npath = setnetpath(); if (npath == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("SETNETPATH", npath, 0, file, line); mutex_unlock(&alloc_list_lock); return (npath); } int my_endnetpath(void *npath, const char *file, int line) { int res; res = endnetpath(npath); if (res != 0) return (res); mutex_lock(&alloc_list_lock); drop_alloc("SETNETPATH", npath, file, line); mutex_unlock(&alloc_list_lock); return (0); } int my_netdir_getbyname( struct netconfig *tp, struct nd_hostserv *serv, struct nd_addrlist **addrs, const char *file, int line) { int res; res = netdir_getbyname(tp, serv, addrs); if (res != 0) return (res); mutex_lock(&alloc_list_lock); add_alloc("NETDIR_GETBYNAME", *addrs, 0, file, line); mutex_unlock(&alloc_list_lock); return (0); } void my_netdir_free(void *ptr, int type, const char *file, int line) { netdir_free(ptr, type); mutex_lock(&alloc_list_lock); drop_alloc("NETDIR_GETBYNAME", ptr, file, line); mutex_unlock(&alloc_list_lock); } struct hostent * my_getipnodebyname( const char *name, int af, int flags, int *error_num, char *file, int line) { struct hostent *res; res = getipnodebyname(name, af, flags, error_num); if (res == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("GETIPNODEBYNAME", res, 0, file, line); mutex_unlock(&alloc_list_lock); return (res); } void my_freehostent(struct hostent *hent, char *file, int line) { freehostent(hent); mutex_lock(&alloc_list_lock); drop_alloc("GETIPNODEBYNAME", hent, file, line); mutex_unlock(&alloc_list_lock); } struct netconfig * my_getnetconfigent(char *netid, char *file, int line) { struct netconfig *res; res = getnetconfigent(netid); if (res == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("GETNETCONFIGENT", res, 0, file, line); mutex_unlock(&alloc_list_lock); return (res); } void my_freenetconfigent(struct netconfig *netp, char *file, int line) { freenetconfigent(netp); mutex_lock(&alloc_list_lock); drop_alloc("GETNETCONFIGENT", netp, file, line); mutex_unlock(&alloc_list_lock); } void * my__rpc_setconf(char *nettype, char *file, int line) { void *res; res = __rpc_setconf(nettype); if (res == NULL) return (NULL); mutex_lock(&alloc_list_lock); add_alloc("RPC_SETCONF", res, 0, file, line); mutex_unlock(&alloc_list_lock); return (res); } void my__rpc_endconf(void *vhandle, char *file, int line) { __rpc_endconf(vhandle); mutex_lock(&alloc_list_lock); drop_alloc("RPC_SETCONF", vhandle, file, line); mutex_unlock(&alloc_list_lock); } extern void flush_caches(); void _flush_caches() { } #pragma weak flush_caches = _flush_caches void check_leaks(char *filename) { struct alloc_list *alist; FILE *fp; fp = fopen(filename, "a"); if (fp == NULL) { syslog(LOG_ERR, "check_leaks, could not open file: %s", filename); return; } flush_caches(); fprintf(fp, "*** POSSIBLE LEAKS ****\n"); mutex_lock(&alloc_list_lock); alist = halist; while (alist != NULL) { fprintf(fp, "\t%s: %d bytes at %p in %s/%d\n", alist->type, alist->size, alist->addr, alist->file, alist->line); alist = alist->next; } mutex_unlock(&alloc_list_lock); (void) fclose(fp); } #else /* * To prevent a compiler warning. */ static char filler; #endif