mirror of
				https://github.com/MariaDB/server.git
				synced 2025-11-03 20:36:16 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			172 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* Copyright (C) 2006-2008 MySQL AB, 2008-2009 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 */
 | 
						|
 | 
						|
#include <tap.h>
 | 
						|
 | 
						|
#include <my_global.h>
 | 
						|
#include <my_sys.h>
 | 
						|
#include <lf.h>
 | 
						|
#include <m_string.h>
 | 
						|
#include "../trnman.h"
 | 
						|
 | 
						|
pthread_mutex_t rt_mutex;
 | 
						|
pthread_attr_t attr;
 | 
						|
size_t stacksize= 0;
 | 
						|
#define STACK_SIZE (((int)stacksize-2048)*STACK_DIRECTION)
 | 
						|
 | 
						|
int rt_num_threads;
 | 
						|
int litmus;
 | 
						|
 | 
						|
/*
 | 
						|
  create and end (commit or rollback) transactions randomly
 | 
						|
*/
 | 
						|
#define MAX_ITER 100
 | 
						|
pthread_handler_t test_trnman(void *arg)
 | 
						|
{
 | 
						|
  uint   x, y, i, n;
 | 
						|
  TRN    *trn[MAX_ITER];
 | 
						|
  int    m= (*(int *)arg);
 | 
						|
 | 
						|
  if (my_thread_init())
 | 
						|
    BAIL_OUT("my_thread_init failed!");
 | 
						|
 | 
						|
  for (x= ((int)(intptr)(&m)); m > 0; )
 | 
						|
  {
 | 
						|
    y= x= (x*3628273133LL + 1500450271LL) % 9576890767LL; /* three prime numbers */
 | 
						|
    m-= n= x % MAX_ITER;
 | 
						|
    for (i= 0; i < n; i++)
 | 
						|
    {
 | 
						|
      trn[i]= trnman_new_trn(0);
 | 
						|
      if (!trn[i])
 | 
						|
      {
 | 
						|
        diag("trnman_new_trn() failed");
 | 
						|
        litmus++;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    for (i= 0; i < n; i++)
 | 
						|
    {
 | 
						|
      y= (y*19 + 7) % 31;
 | 
						|
      trnman_end_trn(trn[i], y & 1);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  pthread_mutex_lock(&rt_mutex);
 | 
						|
  rt_num_threads--;
 | 
						|
  pthread_mutex_unlock(&rt_mutex);
 | 
						|
 | 
						|
  my_thread_end();
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
#undef MAX_ITER
 | 
						|
 | 
						|
void run_test(const char *test, pthread_handler handler, int n, int m)
 | 
						|
{
 | 
						|
  pthread_t *threads;
 | 
						|
  ulonglong now= microsecond_interval_timer();
 | 
						|
  int i;
 | 
						|
 | 
						|
  litmus= 0;
 | 
						|
 | 
						|
  threads= (pthread_t *)my_malloc(PSI_NOT_INSTRUMENTED, sizeof(void *)*n, MYF(0));
 | 
						|
  if (!threads)
 | 
						|
  {
 | 
						|
    diag("Out of memory");
 | 
						|
    abort();
 | 
						|
  }
 | 
						|
 | 
						|
  diag("Testing %s with %d threads, %d iterations... ", test, n, m);
 | 
						|
  rt_num_threads= n;
 | 
						|
  for (i= 0; i < n ; i++)
 | 
						|
    if (pthread_create(threads+i, &attr, handler, &m))
 | 
						|
    {
 | 
						|
      diag("Could not create thread");
 | 
						|
      abort();
 | 
						|
    }
 | 
						|
  for (i= 0 ; i < n ; i++)
 | 
						|
    pthread_join(threads[i], 0);
 | 
						|
  now= microsecond_interval_timer() - now;
 | 
						|
  ok(litmus == 0, "Tested %s in %g secs (%d)", test, ((double)now)/1e6, litmus);
 | 
						|
  my_free(threads);
 | 
						|
}
 | 
						|
 | 
						|
#define ok_read_from(T1, T2, RES)                       \
 | 
						|
  i= trnman_can_read_from(trn[T1], trid[T2]);           \
 | 
						|
  ok(i == RES, "trn" #T1 " %s read from trn" #T2, i ? "can" : "cannot")
 | 
						|
#define start_transaction(T)                            \
 | 
						|
  trn[T]= trnman_new_trn(0);                            \
 | 
						|
  trid[T]= trn[T]->trid
 | 
						|
#define commit(T)               trnman_commit_trn(trn[T])
 | 
						|
#define abort(T)                trnman_abort_trn(trn[T])
 | 
						|
 | 
						|
#define Ntrns 4
 | 
						|
void test_trnman_read_from()
 | 
						|
{
 | 
						|
  TRN *trn[Ntrns];
 | 
						|
  TrID trid[Ntrns];
 | 
						|
  int i;
 | 
						|
 | 
						|
  start_transaction(0);                    /* start trn1 */
 | 
						|
  start_transaction(1);                    /* start trn2 */
 | 
						|
  ok_read_from(1, 0, 0);
 | 
						|
  commit(0);                               /* commit trn1 */
 | 
						|
  start_transaction(2);                    /* start trn4 */
 | 
						|
  abort(2);                                /* abort trn4 */
 | 
						|
  start_transaction(3);                    /* start trn5 */
 | 
						|
  ok_read_from(3, 0, 1);
 | 
						|
  ok_read_from(3, 1, 0);
 | 
						|
  ok_read_from(3, 2, 0);
 | 
						|
  ok_read_from(3, 3, 1);
 | 
						|
  commit(1);                               /* commit trn2 */
 | 
						|
  ok_read_from(3, 1, 0);
 | 
						|
  commit(3);                               /* commit trn5 */
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
int main(int argc __attribute__((unused)), char **argv)
 | 
						|
{
 | 
						|
  MY_INIT(argv[0]);
 | 
						|
 | 
						|
  plan(7);
 | 
						|
 | 
						|
  pthread_mutex_init(&rt_mutex, 0);
 | 
						|
  pthread_attr_init(&attr);
 | 
						|
#ifdef HAVE_PTHREAD_ATTR_GETSTACKSIZE
 | 
						|
  pthread_attr_getstacksize(&attr, &stacksize);
 | 
						|
  if (stacksize == 0)
 | 
						|
#endif
 | 
						|
    stacksize= PTHREAD_STACK_MIN;
 | 
						|
 | 
						|
#define CYCLES 10000
 | 
						|
#define THREADS 10
 | 
						|
 | 
						|
  trnman_init(0);
 | 
						|
 | 
						|
  test_trnman_read_from();
 | 
						|
  run_test("trnman", test_trnman, THREADS, CYCLES);
 | 
						|
 | 
						|
  diag("mallocs: %d", trnman_allocated_transactions);
 | 
						|
  {
 | 
						|
    ulonglong now= microsecond_interval_timer();
 | 
						|
    trnman_destroy();
 | 
						|
    now= microsecond_interval_timer() - now;
 | 
						|
    diag("trnman_destroy: %g", ((double)now)/1e6);
 | 
						|
  }
 | 
						|
 | 
						|
  pthread_mutex_destroy(&rt_mutex);
 | 
						|
  my_end(0);
 | 
						|
  return exit_status();
 | 
						|
}
 | 
						|
 | 
						|
#include "../ma_check_standalone.h"
 |