| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302 | /* StarPU --- Runtime system for heterogeneous multicore architectures. * * Copyright (C) 2009, 2010  Université de Bordeaux 1 * Copyright (C) 2010  Mehdi Juhoor <mjuhoor@gmail.com> * Copyright (C) 2010  Centre National de la Recherche Scientifique * * StarPU is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2.1 of the License, or (at * your option) any later version. * * StarPU 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 Lesser General Public License in COPYING.LGPL for more details. */#include "dw_mult.h"TYPE *A, *B, *C;starpu_data_handle A_handle, B_handle, C_handle;/* * This program computes C = A * B  *  *   A of size (z,y) *   B of size (x,z) *   C of size (x,y)              |---------------|            z |       B       |              |---------------|       z              x     |----|   |---------------|     |    |   |               |     |    |   |               |     | A  | y |       C       |     |    |   |               |     |    |   |               |     |----|   |---------------| */static void check_output(void){	/* check results */	/* compute C = C - AB */	CPU_GEMM("N", "N", ydim, xdim, zdim, (TYPE)-1.0f, A, ydim, B, zdim, (TYPE)1.0f, C, ydim);			/* make sure C = 0 */	TYPE err;	err = CPU_ASUM(xdim*ydim, C, 1);	if (err < xdim*ydim*0.001) {		fprintf(stderr, "Results are OK\n");	}	else {		int max;		max = CPU_IAMAX(xdim*ydim, C, 1);		fprintf(stderr, "There were errors ... err = %f\n", err);		fprintf(stderr, "Max error : %e\n", C[max]);	}}void callback_func(void *arg){	/* do some accounting */	int id = starpu_worker_get_id();	flop_per_worker[id] += BLAS3_FLOP(conf.m, conf.n, conf.k);	ls_per_worker[id] += BLAS3_LS(conf.m, conf.n, conf.k);}static void init_problem_data(void){	unsigned i,j;#ifdef STARPU_USE_CUDA	if (pin) {		starpu_data_malloc_pinned_if_possible((void **)&A, zdim*ydim*sizeof(TYPE));		starpu_data_malloc_pinned_if_possible((void **)&B, xdim*zdim*sizeof(TYPE));		starpu_data_malloc_pinned_if_possible((void **)&C, xdim*ydim*sizeof(TYPE));	} else#endif	{#ifdef STARPU_HAVE_POSIX_MEMALIGN		posix_memalign((void **)&A, 4096, zdim*ydim*sizeof(TYPE));		posix_memalign((void **)&B, 4096, xdim*zdim*sizeof(TYPE));		posix_memalign((void **)&C, 4096, xdim*ydim*sizeof(TYPE));#else		A = malloc(zdim*ydim*sizeof(TYPE));		B = malloc(xdim*zdim*sizeof(TYPE));		C = malloc(xdim*ydim*sizeof(TYPE));#endif	}	/* fill the A and B matrices */	if (norandom) {		for (j=0; j < ydim; j++) {			for (i=0; i < zdim; i++) {				A[j+i*ydim] = (TYPE)(i);			}		}			for (j=0; j < zdim; j++) {			for (i=0; i < xdim; i++) {				B[j+i*zdim] = (TYPE)(j);			}		}	} 	else {#ifdef NORANDOM		srand(2008);		STARPU_ABORT();#endif		for (j=0; j < ydim; j++) {			for (i=0; i < zdim; i++) {				A[j+i*ydim] = (TYPE)(starpu_drand48());			}		}			for (j=0; j < zdim; j++) {			for (i=0; i < xdim; i++) {				B[j+i*zdim] = (TYPE)(starpu_drand48());			}		}	}	for (j=0; j < ydim; j++) {		for (i=0; i < xdim; i++) {			C[j+i*ydim] = (TYPE)(0);		}	}	display_memory_consumption();}static void partition_mult_data(void){	starpu_matrix_data_register(&A_handle, 0, (uintptr_t)A, 		ydim, ydim, zdim, sizeof(TYPE));	starpu_matrix_data_register(&B_handle, 0, (uintptr_t)B, 		zdim, zdim, xdim, sizeof(TYPE));	starpu_matrix_data_register(&C_handle, 0, (uintptr_t)C, 		ydim, ydim, xdim, sizeof(TYPE));	starpu_data_set_wt_mask(C_handle, 1<<0);	conf.k = zdim;	conf.m = ydim/nslicesy;	conf.n = xdim/nslicesx;	struct starpu_data_filter f;	f.filter_func = starpu_vertical_block_filter_func;	f.nchildren = nslicesx;	f.get_nchildren = NULL;	f.get_child_ops = NULL;			struct starpu_data_filter f2;	f2.filter_func = starpu_block_filter_func;	f2.nchildren = nslicesy;	f2.get_nchildren = NULL;	f2.get_child_ops = NULL;			starpu_data_partition(B_handle, &f);	starpu_data_partition(A_handle, &f2);	starpu_data_map_filters(C_handle, 2, &f, &f2);}static void unpartition_mult_data(void){	fprintf(stderr, "unpartition !!\n");	starpu_data_unpartition(C_handle, 0);	starpu_data_unregister(C_handle);}static starpu_codelet cl = {	.where = STARPU_CPU|STARPU_CUDA#ifdef SPU_FUNC_SGEMM		|STARPU_GORDON#endif		,	.cpu_func = STARPU_GEMM(cpu_mult),#ifdef STARPU_USE_CUDA	.cuda_func = STARPU_GEMM(cublas_mult),#endif#ifdef STARPU_USE_GORDON#ifdef SPU_FUNC_SGEMM	.gordon_func = SPU_FUNC_SGEMM,#else#warning SPU_FUNC_SGEMM is not available#endif#endif	.nbuffers = 3};static struct starpu_task *construct_task(unsigned x, unsigned y, unsigned z, unsigned iter){	/* A B[task] = C[task] */	struct starpu_task *task = starpu_task_create();	task->cl = &cl;	/* we have a callback to do some accounting */	task->callback_func = callback_func;	task->callback_arg = NULL;	task->buffers[0].handle = starpu_data_get_sub_data(A_handle, 1, y);	task->buffers[0].mode = STARPU_R;	task->buffers[1].handle = starpu_data_get_sub_data(B_handle, 1, x);	task->buffers[1].mode = STARPU_R;	task->buffers[2].handle = starpu_data_get_sub_data(C_handle, 2, x, y);	task->buffers[2].mode = STARPU_RW;	task->cl_arg = &conf;	task->cl_arg_size = sizeof(struct block_conf);	return task;}static void submit_new_iter(unsigned x, unsigned y, unsigned iter){	unsigned z;	z = 0;	{		struct starpu_task *task;		task = construct_task(x, y, z, iter);		starpu_task_submit(task);	}}static void launch_codelets(void){#ifdef STARPU_USE_FXT	_starpu_fxt_register_thread(0);#endif	/* partition the work into slices */	unsigned taskx, tasky;	srand(time(NULL));	/* should we use a single performance model for all archs and use an 	 * acceleration factor ? */	if (use_common_model) {		cl.model = &STARPU_GEMM(model_common);	}	else {		cl.model = &STARPU_GEMM(model);	}	for (taskx = 0; taskx < nslicesx; taskx++) 	{		for (tasky = 0; tasky < nslicesy; tasky++)		{			submit_new_iter(taskx, tasky, 0);		}	}}int main(__attribute__ ((unused)) int argc, 	 __attribute__ ((unused)) char **argv){	parse_args(argc, argv);	/* start the runtime */	starpu_init(NULL);	starpu_helper_cublas_init();	init_problem_data();	gettimeofday(&start, NULL);	partition_mult_data();	launch_codelets();	starpu_task_wait_for_all();	gettimeofday(&end, NULL);	double timing = (double)((end.tv_sec - start.tv_sec)*1000000 + (end.tv_usec - start.tv_usec));	display_stats(timing);	unpartition_mult_data();		if (check)		check_output();		starpu_helper_cublas_shutdown();	starpu_shutdown();	return 0;}
 |