mirror of
https://github.com/MariaDB/server.git
synced 2025-01-17 12:32:27 +01:00
251 lines
6.3 KiB
C
251 lines
6.3 KiB
C
|
/* 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 <windows.h>
|
||
|
#include <process.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
#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_decl(A,B) unsigned __cdecl A(void *B)
|
||
|
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_decl(test_thread,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;
|
||
|
}
|