Backport of svn revision 514:
libupnp and multi-flows scenario patch Submited by Carlo Parata from STMicroelectronics. Hi Roberto and Nektarios, after an analysis of the problem of libupnp with a multi-flows scenario, I noticed that the only cause of the freezed system is the ThreadPool management. There are not mutex problems. In practise, if all threads in the thread pool are busy executing jobs, a new worker thread should be created if a job is scheduled (I inspired to tombupnp library). So I solved the problem with a little patch in threadutil library that you can find attached in this e-mail. I hope to have helped you. git-svn-id: https://pupnp.svn.sourceforge.net/svnroot/pupnp/branches/branch-1.6.x@515 119443c7-1b9e-41f8-b6fc-b9c35fce742c
This commit is contained in:
		
							
								
								
									
										12
									
								
								ChangeLog
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								ChangeLog
									
									
									
									
									
								
							| @@ -2,6 +2,18 @@ | |||||||
| Version 1.6.7 | Version 1.6.7 | ||||||
| ******************************************************************************* | ******************************************************************************* | ||||||
|  |  | ||||||
|  | 2010-03-21 Marcelo Jimenez <mroberto(at)users.sourceforge.net> | ||||||
|  | 	* libupnp and multi-flows scenario patch | ||||||
|  | 	Submited by Carlo Parata from STMicroelectronics. | ||||||
|  | Hi Roberto and Nektarios, | ||||||
|  | after an analysis of the problem of libupnp with a multi-flows scenario, I | ||||||
|  | noticed that the only cause of the freezed system is the ThreadPool | ||||||
|  | management. There are not mutex problems. In practise, if all threads in the | ||||||
|  | thread pool are busy executing jobs, a new worker thread should be created if | ||||||
|  | a job is scheduled (I inspired to tombupnp library). So I solved the problem | ||||||
|  | with a little patch in threadutil library that you can find attached in this | ||||||
|  | e-mail. I hope to have helped you. | ||||||
|  |  | ||||||
| 2010-03-21 Marcelo Jimenez <mroberto(at)users.sourceforge.net> | 2010-03-21 Marcelo Jimenez <mroberto(at)users.sourceforge.net> | ||||||
| 	* SF Patch Tracker [ 2964973 ] install: will not overwrite just-created | 	* SF Patch Tracker [ 2964973 ] install: will not overwrite just-created | ||||||
| 	...blah... with... | 	...blah... with... | ||||||
|   | |||||||
| @@ -267,18 +267,17 @@ typedef struct THREADPOOL | |||||||
| { | { | ||||||
| 	ithread_mutex_t mutex;		/* mutex to protect job qs */ | 	ithread_mutex_t mutex;		/* mutex to protect job qs */ | ||||||
| 	ithread_cond_t condition;	/* condition variable to signal Q */ | 	ithread_cond_t condition;	/* condition variable to signal Q */ | ||||||
| 	ithread_cond_t start_and_shutdown; /* condition variable for start  | 	ithread_cond_t start_and_shutdown; /* condition variable for start and stop */ | ||||||
| 					and stop */ |  | ||||||
| 	int lastJobId;			/* ids for jobs */ | 	int lastJobId;			/* ids for jobs */ | ||||||
| 	int shutdown;			/* whether or not we are shutting down */ | 	int shutdown;			/* whether or not we are shutting down */ | ||||||
| 	int totalThreads;		/* total number of threads */ | 	int totalThreads;		/* total number of threads */ | ||||||
|  | 	int busyThreads;		/* number of threads that are currently executing jobs */ | ||||||
| 	int persistentThreads;		/* number of persistent threads */ | 	int persistentThreads;		/* number of persistent threads */ | ||||||
| 	FreeList jobFreeList;		/* free list of jobs */ | 	FreeList jobFreeList;		/* free list of jobs */ | ||||||
| 	LinkedList lowJobQ;		/* low priority job Q */ | 	LinkedList lowJobQ;		/* low priority job Q */ | ||||||
| 	LinkedList medJobQ;		/* med priority job Q */ | 	LinkedList medJobQ;		/* med priority job Q */ | ||||||
| 	LinkedList highJobQ;		/* high priority job Q */ | 	LinkedList highJobQ;		/* high priority job Q */ | ||||||
| 	ThreadPoolJob *persistentJob;	/* persistent job */ | 	ThreadPoolJob *persistentJob;	/* persistent job */ | ||||||
|  |  | ||||||
| 	ThreadPoolAttr attr;		/* thread pool attributes */ | 	ThreadPoolAttr attr;		/* thread pool attributes */ | ||||||
|  |  | ||||||
| 	/* statistics */ | 	/* statistics */ | ||||||
|   | |||||||
| @@ -472,6 +472,7 @@ static void *WorkerThread( void *arg ) | |||||||
| 	while( 1 ) { | 	while( 1 ) { | ||||||
| 		ithread_mutex_lock( &tp->mutex ); | 		ithread_mutex_lock( &tp->mutex ); | ||||||
| 		if( job ) { | 		if( job ) { | ||||||
|  | 			tp->busyThreads--; | ||||||
| 			FreeThreadPoolJob( tp, job ); | 			FreeThreadPoolJob( tp, job ); | ||||||
| 			job = NULL; | 			job = NULL; | ||||||
| 		} | 		} | ||||||
| @@ -584,6 +585,7 @@ static void *WorkerThread( void *arg ) | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		tp->busyThreads++; | ||||||
| 		ithread_mutex_unlock( &tp->mutex ); | 		ithread_mutex_unlock( &tp->mutex ); | ||||||
|  |  | ||||||
| 		if( SetPriority( job->priority ) != 0 ) { | 		if( SetPriority( job->priority ) != 0 ) { | ||||||
| @@ -688,7 +690,7 @@ static int CreateWorker( ThreadPool *tp ) | |||||||
|  *      ThreadPool* tp |  *      ThreadPool* tp | ||||||
|  * |  * | ||||||
|  *****************************************************************************/ |  *****************************************************************************/ | ||||||
| static void AddWorker( ThreadPool *tp ) | static void AddWorker(ThreadPool *tp) | ||||||
| { | { | ||||||
| 	int jobs = 0; | 	int jobs = 0; | ||||||
| 	int threads = 0; | 	int threads = 0; | ||||||
| @@ -697,8 +699,10 @@ static void AddWorker( ThreadPool *tp ) | |||||||
|  |  | ||||||
| 	jobs = tp->highJobQ.size + tp->lowJobQ.size + tp->medJobQ.size; | 	jobs = tp->highJobQ.size + tp->lowJobQ.size + tp->medJobQ.size; | ||||||
| 	threads = tp->totalThreads - tp->persistentThreads; | 	threads = tp->totalThreads - tp->persistentThreads; | ||||||
| 	while( threads == 0 || (jobs / threads) >= tp->attr.jobsPerThread ) { |  	while (threads == 0 || | ||||||
| 		if( CreateWorker( tp ) != 0 ) { | 	    (jobs / threads) >= tp->attr.jobsPerThread || | ||||||
|  | 	    (tp->totalThreads == tp->busyThreads) ) { | ||||||
|  | 		if (CreateWorker(tp) != 0) { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
| 		threads++; | 		threads++; | ||||||
| @@ -804,6 +808,7 @@ int ThreadPoolInit( ThreadPool *tp, ThreadPoolAttr *attr ) | |||||||
| 		tp->lastJobId = 0; | 		tp->lastJobId = 0; | ||||||
| 		tp->shutdown = 0; | 		tp->shutdown = 0; | ||||||
| 		tp->totalThreads = 0; | 		tp->totalThreads = 0; | ||||||
|  | 		tp->busyThreads = 0; | ||||||
| 		tp->persistentThreads = 0; | 		tp->persistentThreads = 0; | ||||||
| 		for( i = 0; i < tp->attr.minThreads; ++i ) { | 		for( i = 0; i < tp->attr.minThreads; ++i ) { | ||||||
| 			if( ( retCode = CreateWorker( tp ) ) != 0 ) { | 			if( ( retCode = CreateWorker( tp ) ) != 0 ) { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Marcelo Roberto Jimenez
					Marcelo Roberto Jimenez