code example
shmem SGI shmem
one-sided communication
POSIX threads
Cell Cell Broadband Engine
AltiVec PowerPC SIMD instructions
file sub1.h: common interface

class data1 {
public:
   data1() {}
   int n0, n1;
   float *x, *y;
   float e;
   int pad[27];// pad to 128 byte length
};

file spe1.cc: SPU code

#include <spu_mfcio.h>
#include "sub1.h"
#define BLOCK (2048)
int main(unsigned long long id,
     unsigned long long argp, unsigned long long envp) {
  data1 vec __attribute__ (aligned(128));
  // read parameter
  mfc_get(&vec, (unsigned int)argp, sizeof(data1), 1, 0, 0);
  mfc_write_tag_mask(0xffffffff);
  mfc_read_tag_status_all();// wait for data transfer
  float e = 0;
  // allocate double buffer in local SPU memory
  float *x = (float*)malloc(BLOCK*sizeof(float));
  float *x0 = (float*)malloc(BLOCK*sizeof(float));
  float *y = (float*)malloc((BLOCK+32)*sizeof(float)) + 1;
  float *y0 = (float*)malloc((BLOCK+32)*sizeof(float)) + 1;
  // get first buffer
  mfc_get(x, &vec.x[vec.n0], BLOCK*sizeof(float), 2, 0, 0);
  mfc_get(y-1, &vec.y[vec.n0], (BLOCK+32)*sizeof(float), 3, 0, 0);
  mfc_read_tag_status_all();
  for (int ib=vec.n0; ib<vec.n1; ib += BLOCK) {
   if (ib+BLOCK<vec.n1) {
    // get next buffer, fence after put x
    mfc_getf(x0, &vec.x[ib+BLOCK], BLOCK*sizeof(float), 2, 0, 0);
    mfc_get(y0-1, &vec.y[ib+BLOCK], (BLOCK+32)*sizeof(float), 3, 0, 0);
   }
  // do computation
  float ve[4] = {0, 0, 0, 0};
  for (int i=0; iBLOCK; i+=4) {
   float* yp = &y[i+1], y0 = &y[i], ym = &y[i-1];
   vec_st(vec_madd(
    vec_splats(.5),
    vec_add(
     vec_perm(vec_ld(0,ym), vec_ld(16,ym),
      vec_lvsl(0,ym)),
     vec_perm(vec_ld(0,yp), vec_ld(16,yp),
      vec_lvsl(0,yp))),
    vec_splats(0.)),
    0, &x[i]);
   vec_st(vec_add(
    vec_ld(0,&ve[0]),vec_madd(
     vec_ld(0,y0),
     vec_ld(0,y0),
     vec_splats(0.))),
    0, &ve[0]);
  }
  e += ve[0] + ve[1] + ve[2] + ve[3];
   mfc_read_tag_status_all(); // wait for data transfer
   // put current buffer
   mfc_put(x, &vec.x[ib], BLOCK*sizeof(float), 2, 0, 0);
   float *t = x; x = x0; x0 = t;
   t = y; y = y0; y0 = t; // swap buffers
  }
  // put reduction value
  vec.e = e;
  mfc_put(&vec, argp, sizeof(data1), 1, 0, 0);
  mfc_read_tag_status_all(); // wait for data transfer
  return 0;
}

main file
#include <mpp/shmem.h>
#include <pthread.h>
#include <libspe2.h>
#include "sub1.h"
#include <altivec.h>

float *x, *y;
#define PROC (8)
float e_vec[PROC];
int n_thread0, n_thread1;

extern spe_program_handle_t spe1; // defined in SPU code
void *thread1(void *arg) {
  int p = (int)arg;
  int n0 = n_thread0 + (p * (n_thread1-n_thread0)) / PROC;
  int n1 = n_thread0 + ((p+1) * (n_thread1-n_thread0)) / PROC;
  data1 block __attribute__ (aligned(128));
  spe_context_ptr_t ctxs;
  spe_stop_info_t st;
  ctxs = spe_context_create(0, NULL);
  spe_program_load (ctxs, &spe1);
  block.n0 = n0;
  block.n1 = n1;
  block.x = &x[n0];
  block.y = &y[n0-1];
  unsigned int entry = SPE_DEFAULT_ENTRY;
  float e = 0;
  // execute code on a single SPU and wait for termination
  spe_context_run(ctxs[p], &entry, 0, &block, NULL, st);
  spe_context_destroy(ctxs);
  e += block.e;
  e_vec[p] = e;
  return (void*) 0;
}

int main(int argc, char *argv[]) {
  int n = ...;
  start_pes(0);
  int nn = (n-1) / _num_pes();
  int n_local0 = 1 + _my_pe() * nn;
  int n_local1 = 1 + (_my_pe()+1) * nn;
  // allocate only local part + ghost zone of the arrays x,y
  float *x, *y;
  x = (float*)shmalloc((n_local1 - n_local0 + 2)*sizeof(float));
  y = (float*)shmalloc((n_local1 - n_local0 + 2)*sizeof(float));
  x -= (n_local0 - 1);
  y -= (n_local0 - 1);
  shmem_barrier_all();

  ... // fill x, y

  // fill ghost zone
  if (_my_pe() > 0)
   shmem_float_put(&y[n_local1], &y[n_local0], 1, _my_pe()-1);
  if (_my_pe() < _num_pes()-1)
   shmem_float_put(&y[n_local0-1], &y[n_local1-1], 1, _my_pe()+1);
  shmem_barrier_all();

  pthread_t threads[PROC];
  pthread_attr_t attr;
  pthread_attr_init(&attr);
  n_thread0 = n_local0;
  n_thread1 = n_local1;
  float e = 0;
  // start threads and wait for termination
  for (int p=0; p<PROC; ++p)
   pthread_create(&threads[p], &attr, thread1, (void *)p);
  for (int p=0; p<PROC; ++p) {
   pthread_join(threads[p], NULL);
   e += e_vec[p];
  }

  static float work[_SHMEM_REDUCE_MIN_WRKDATA_SIZE];
  static long sync[_SHMEM_REDUCE_MIN_WRKDATA_SIZE];
  static float el, es;
  el = e;
  shmem_float_sum_to_all(&es, &el, 1,
   0, 0, _num_pes(), work, sync);
  e = es;

  ... // output x, e

  x += (n_local0 - 1);
  y += (n_local0 - 1);
  shfree(x);
  shfree(y);
  return 0;
}

[start] [references] [download] [install]