/* Copyright (C) 2003-2005 MySQL AB This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ /* Testing of thread creation to find memory allocation bug ** This is coded to use as few extern functions as possible! ** ** The program must be compiled to be multithreaded ! ** ** The problem is that when this program is run it will allocate more and more ** memory, so there is a memory leak in the thread handling. The problem is how ** to avoid is ! ** ** It looks like the bug is that the std library doesn't free thread ** specific variables if one uses a thread variable. ** If one compiles this program with -DREMOVE_BUG ** there is no memory leaks anymore! ** ** This program is tested with Microsofts VC++ 5.0, but BC5.2 is also ** reported to have this bug. */ #include #include #include #define TEST_COUNT 100000 /***************************************************************************** ** The following is to emulate the posix thread interface *****************************************************************************/ typedef HANDLE pthread_t; typedef struct thread_attr { DWORD dwStackSize ; DWORD dwCreatingFlag ; int priority ; } pthread_attr_t ; typedef struct { int dummy; } pthread_condattr_t; typedef struct { unsigned int msg; pthread_t thread; DWORD thread_id; } pthread_cond_t; typedef CRITICAL_SECTION pthread_mutex_t; #define pthread_mutex_init(A,B) InitializeCriticalSection(A) #define pthread_mutex_lock(A) (EnterCriticalSection(A),0) #define pthread_mutex_unlock(A) LeaveCriticalSection(A) #define pthread_mutex_destroy(A) DeleteCriticalSection(A) #define pthread_handler_t unsigned __cdecl typedef unsigned (__cdecl *pthread_handler)(void *); #define pthread_self() GetCurrentThread() static unsigned int thread_count; static pthread_cond_t COND_thread_count; static pthread_mutex_t LOCK_thread_count; pthread_mutex_t THR_LOCK_malloc,THR_LOCK_open,THR_LOCK_keycache, THR_LOCK_lock,THR_LOCK_isam; /* ** We have tried to use '_beginthreadex' instead of '_beginthread' here ** but in this case the program leaks about 512 characters for each ** created thread ! */ int pthread_create(pthread_t *thread_id, pthread_attr_t *attr, pthread_handler func, void *param) { HANDLE hThread; hThread=(HANDLE)_beginthread(func, attr->dwStackSize ? attr->dwStackSize : 65535,param); if ((long) hThread == -1L) { return(errno ? errno : -1); } *thread_id=hThread; return(0); } void pthread_exit(unsigned A) { _endthread(); } /* ** The following simple implementation of conds works as long as ** only one thread uses pthread_cond_wait at a time. ** This is coded very carefully to work with thr_lock. */ static unsigned int WIN32_WAIT_SIGNAL= 30000; /* Start message to use */ int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr) { cond->msg=WIN32_WAIT_SIGNAL++; cond->thread=(pthread_t) pthread_self(); /* For global conds */ //IRENA cond->thread_id=GetCurrentThreadId(); return 0; } int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex) { MSG msg ; unsigned int msgCode=cond->msg; cond->thread=(pthread_t) pthread_self(); //IRENA //??? cond->thread_id=GetCurrentThreadId(); //VOID(ReleaseMutex(*mutex)); LeaveCriticalSection(mutex); do { WaitMessage() ; if (!PeekMessage(&msg, NULL, 1, 65534,PM_REMOVE)) { return errno=GetLastError() ; } } while (msg.message != msgCode) ; EnterCriticalSection(mutex); return 0 ; } int pthread_cond_signal(pthread_cond_t *cond) { if (!PostThreadMessage(cond->thread_id, cond->msg, 0,0)) { return errno=GetLastError() ; } return 0 ; } int pthread_attr_init(pthread_attr_t *connect_att) { connect_att->dwStackSize = 0; connect_att->dwCreatingFlag = 0; connect_att->priority = 0; return 0; } int pthread_attr_setstacksize(pthread_attr_t *connect_att,DWORD stack) { connect_att->dwStackSize=stack; return 0; } int pthread_attr_setprio(pthread_attr_t *connect_att,int priority) { connect_att->priority=priority; return 0; } int pthread_attr_destroy(pthread_attr_t *connect_att) { return 0; } /* from my_pthread.c */ #ifndef REMOVE_BUG __declspec(thread) int THR_KEY_my_errno; int _my_errno(void) { return THR_KEY_my_errno; } #endif /***************************************************************************** ** The test program *****************************************************************************/ pthread_handler_t test_thread(void *arg) { pthread_mutex_lock(&LOCK_thread_count); thread_count--; pthread_cond_signal(&COND_thread_count); /* Tell main we are ready */ pthread_mutex_unlock(&LOCK_thread_count); pthread_exit(0); return 0; } int main(int argc,char **argv) { pthread_t tid; pthread_attr_t thr_attr; int i,error; if ((error=pthread_cond_init(&COND_thread_count,NULL))) { fprintf(stderr,"Got error: %d from pthread_cond_init (errno: %d)", error,errno); exit(1); } pthread_mutex_init(&LOCK_thread_count,NULL); if ((error=pthread_attr_init(&thr_attr))) { fprintf(stderr,"Got error: %d from pthread_attr_init (errno: %d)", error,errno); exit(1); } if ((error=pthread_attr_setstacksize(&thr_attr,65536L))) { fprintf(stderr,"Got error: %d from pthread_attr_setstacksize (errno: %d)", error,errno); exit(1); } printf("Init ok. Creating %d threads\n",TEST_COUNT); for (i=1 ; i <= TEST_COUNT ; i++) { int *param= &i; if ((i % 100) == 0) { printf("%8d",i); fflush(stdout); } if ((error=pthread_mutex_lock(&LOCK_thread_count))) { fprintf(stderr,"\nGot error: %d from pthread_mutex_lock (errno: %d)", error,errno); exit(1); } if ((error=pthread_create(&tid,&thr_attr,test_thread,(void*) param))) { fprintf(stderr,"\nGot error: %d from pthread_create (errno: %d)\n", error,errno); pthread_mutex_unlock(&LOCK_thread_count); exit(1); } thread_count++; pthread_mutex_unlock(&LOCK_thread_count); if ((error=pthread_mutex_lock(&LOCK_thread_count))) fprintf(stderr,"\nGot error: %d from pthread_mutex_lock\n",error); while (thread_count) { if ((error=pthread_cond_wait(&COND_thread_count,&LOCK_thread_count))) fprintf(stderr,"\nGot error: %d from pthread_cond_wait\n",error); } pthread_mutex_unlock(&LOCK_thread_count); } pthread_attr_destroy(&thr_attr); printf("\nend\n"); return 0; }