mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-26 16:38:11 +01:00 
			
		
		
		
	 17b4f99928
			
		
	
	
	17b4f99928
	
	
	
		
			
			This commit is based on the work of Michal Schorm, rebased on the
earliest MariaDB version.
Th command line used to generate this diff was:
find ./ -type f \
  -exec sed -i -e 's/Foundation, Inc., 59 Temple Place, Suite 330, Boston, /Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, /g' {} \; \
  -exec sed -i -e 's/Foundation, Inc. 59 Temple Place.* Suite 330, Boston, /Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, /g' {} \; \
  -exec sed -i -e 's/MA.*.....-1307.*USA/MA 02110-1335 USA/g' {} \; \
  -exec sed -i -e 's/Foundation, Inc., 59 Temple/Foundation, Inc., 51 Franklin/g' {} \; \
  -exec sed -i -e 's/Place, Suite 330, Boston, MA.*02111-1307.*USA/Street, Fifth Floor, Boston, MA 02110-1335 USA/g' {} \; \
  -exec sed -i -e 's/MA.*.....-1307/MA 02110-1335/g' {} \;
		
	
			
		
			
				
	
	
		
			162 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Copyright (C) 2008 Sun Microsystems, Inc
 | |
| 
 | |
|    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 Street, Fifth Floor, Boston, MA 02110-1335 USA */
 | |
| 
 | |
| /* Testing of deadlock detector */
 | |
| 
 | |
| #include <my_global.h>
 | |
| #include <mysys_priv.h>
 | |
| 
 | |
| 
 | |
| int main(int argc __attribute__((unused)), char** argv)
 | |
| {
 | |
|   pthread_mutex_t LOCK_A, LOCK_B, LOCK_C, LOCK_D, LOCK_E, LOCK_F, LOCK_G;
 | |
|   pthread_mutex_t LOCK_H, LOCK_I;
 | |
|   MY_INIT(argv[0]);
 | |
|   DBUG_ENTER("main");
 | |
| 
 | |
|   DBUG_PUSH("d:t:O,/tmp/trace");
 | |
|   printf("This program is testing the mutex deadlock detection.\n"
 | |
|          "It should print out different failures of wrong mutex usage"
 | |
|          "on stderr\n\n");
 | |
| 
 | |
|   safe_mutex_deadlock_detector= 1;
 | |
|   pthread_mutex_init(&LOCK_A, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_B, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_C, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_D, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_E, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_F, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_G, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_H, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_I, MY_MUTEX_INIT_FAST);
 | |
| 
 | |
|   printf("Testing A->B and B->A\n");
 | |
|   fflush(stdout);
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   /* Test different (wrong) lock order */
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_lock(&LOCK_A);                  /* Should give warning */
 | |
| 
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   /* Check that we don't get another warning for same lock */
 | |
|   printf("Testing A->B and B->A again (should not give a warning)\n");
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   /*
 | |
|     Test of ring with many mutex
 | |
|     We also unlock mutex in different orders to get the unlock code properly
 | |
|     tested.
 | |
|   */
 | |
|   printf("Testing A->C and C->D and D->A\n");
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   pthread_mutex_lock(&LOCK_C);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_C);
 | |
|   pthread_mutex_lock(&LOCK_C);
 | |
|   pthread_mutex_lock(&LOCK_D);
 | |
|   pthread_mutex_unlock(&LOCK_D);
 | |
|   pthread_mutex_unlock(&LOCK_C);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_D);
 | |
|   pthread_mutex_lock(&LOCK_A);                  /* Should give warning */
 | |
| 
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_D);
 | |
| 
 | |
|   printf("Testing E -> F ; H -> I ; F -> H ; H -> I -> E\n");
 | |
|   fflush(stdout);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_E);
 | |
|   pthread_mutex_lock(&LOCK_F);
 | |
|   pthread_mutex_unlock(&LOCK_E);
 | |
|   pthread_mutex_unlock(&LOCK_F);
 | |
|   pthread_mutex_lock(&LOCK_H);
 | |
|   pthread_mutex_lock(&LOCK_I);
 | |
|   pthread_mutex_unlock(&LOCK_I);
 | |
|   pthread_mutex_unlock(&LOCK_H);
 | |
|   pthread_mutex_lock(&LOCK_F);
 | |
|   pthread_mutex_lock(&LOCK_H);
 | |
|   pthread_mutex_unlock(&LOCK_H);
 | |
|   pthread_mutex_unlock(&LOCK_F);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_H);
 | |
|   pthread_mutex_lock(&LOCK_I);
 | |
|   pthread_mutex_lock(&LOCK_E);                  /* Should give warning */
 | |
| 
 | |
|   pthread_mutex_unlock(&LOCK_E);
 | |
|   pthread_mutex_unlock(&LOCK_I);
 | |
|   pthread_mutex_unlock(&LOCK_H);
 | |
| 
 | |
|   printf("\nFollowing shouldn't give any warnings\n");
 | |
|   printf("Testing A->B and B->A without deadlock detection\n");
 | |
|   fflush(stdout);
 | |
| 
 | |
|   /* Reinitialize mutex to get rid of old wrong usage markers */
 | |
|   pthread_mutex_destroy(&LOCK_A);
 | |
|   pthread_mutex_destroy(&LOCK_B);
 | |
|   pthread_mutex_init(&LOCK_A, MY_MUTEX_INIT_FAST);
 | |
|   pthread_mutex_init(&LOCK_B, MY_MUTEX_INIT_FAST);
 | |
| 
 | |
|   /* Start testing */
 | |
|   my_pthread_mutex_lock(&LOCK_A, MYF(MYF_NO_DEADLOCK_DETECTION));
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   my_pthread_mutex_lock(&LOCK_B, MYF(MYF_NO_DEADLOCK_DETECTION));
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   printf("Testing A -> C ; B -> C ; A->B\n");
 | |
|   fflush(stdout);
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   pthread_mutex_lock(&LOCK_C);
 | |
|   pthread_mutex_unlock(&LOCK_C);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_lock(&LOCK_C);
 | |
|   pthread_mutex_unlock(&LOCK_C);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
| 
 | |
|   pthread_mutex_lock(&LOCK_A);
 | |
|   pthread_mutex_lock(&LOCK_B);
 | |
|   pthread_mutex_unlock(&LOCK_B);
 | |
|   pthread_mutex_unlock(&LOCK_A);
 | |
| 
 | |
|   /* Cleanup */
 | |
|   pthread_mutex_destroy(&LOCK_A);
 | |
|   pthread_mutex_destroy(&LOCK_B);
 | |
|   pthread_mutex_destroy(&LOCK_C);
 | |
|   pthread_mutex_destroy(&LOCK_D);
 | |
|   pthread_mutex_destroy(&LOCK_E);
 | |
|   pthread_mutex_destroy(&LOCK_F);
 | |
|   pthread_mutex_destroy(&LOCK_G);
 | |
|   pthread_mutex_destroy(&LOCK_H);
 | |
|   pthread_mutex_destroy(&LOCK_I);
 | |
| 
 | |
|   my_end(MY_DONT_FREE_DBUG);
 | |
|   exit(0);
 | |
| }
 |