[nvptx, libgomp] Fix map_push

The map field of a struct ptx_stream is a FIFO.  The FIFO is implemented as a
single linked list, with pop-from-the-front semantics.

The function map_pop pops an element, either by:
- deallocating the element, if there is more than one element
- or marking the element inactive, if there's only one element

The responsibility of map_push is to push an element to the back, as well as
selecting the element to push, by:
- allocating an element, or
- reusing the element at the front if inactive and big enough, or
- dropping the element at the front if inactive and not big enough, and
  allocating one that's big enough

The current implemention gets at least the first and most basic scenario wrong:

> map = cuda_map_create (size);

We create an element, and assign it to map.

> for (t = s->map; t->next != NULL; t = t->next)
>   ;

We determine the last element in the fifo.

> t->next = map;

We append the new element.

> s->map = map;

But here, we throw away the rest of the FIFO, and declare the FIFO to be just
the new element.

This problem causes the test-case asyncwait-1.c to fail intermittently on some
systems.  The pr87835.c test-case added here is a a minimized and modified
version of asyncwait-1.c (avoiding the kernel construct) that is more likely to
fail.

Fix this by rewriting map_pop more robustly, by:
- seperating the function in two phases: select element, push element
- when reusing or dropping an element, making sure that the element is cleanly
  popped from the queue
- rewriting the push element part in such a way that it can handle all cases
  without needing if statements, such that each line is exercised for each of
  the three cases.

2019-01-23  Tom de Vries  <tdevries@suse.de>

	PR target/87835
	* plugin/plugin-nvptx.c (map_push): Fix adding of allocated element.
	* testsuite/libgomp.oacc-c-c++-common/pr87835.c: New test.

From-SVN: r268176
This commit is contained in:
Tom de Vries 2019-01-23 08:16:11 +00:00 committed by Tom de Vries
parent c262f6b77c
commit 2ee6cb22c1
3 changed files with 100 additions and 21 deletions

View File

@ -1,3 +1,9 @@
2019-01-23 Tom de Vries <tdevries@suse.de>
PR target/87835
* plugin/plugin-nvptx.c (map_push): Fix adding of allocated element.
* testsuite/libgomp.oacc-c-c++-common/pr87835.c: New test.
2019-01-15 Tom de Vries <tdevries@suse.de> 2019-01-15 Tom de Vries <tdevries@suse.de>
PR target/80547 PR target/80547

View File

@ -296,35 +296,46 @@ map_pop (struct ptx_stream *s)
static CUdeviceptr static CUdeviceptr
map_push (struct ptx_stream *s, size_t size) map_push (struct ptx_stream *s, size_t size)
{ {
struct cuda_map *map = NULL, *t = NULL; struct cuda_map *map = NULL;
struct cuda_map **t;
assert (s); assert (s);
assert (s->map); assert (s->map);
/* Each PTX stream requires a separate data region to store the /* Select an element to push. */
launch arguments for cuLaunchKernel. Allocate a new
cuda_map and push it to the end of the list. */
if (s->map->active) if (s->map->active)
{ map = cuda_map_create (size);
map = cuda_map_create (size);
for (t = s->map; t->next != NULL; t = t->next)
;
t->next = map;
}
else if (s->map->size < size)
{
cuda_map_destroy (s->map);
map = cuda_map_create (size);
}
else else
map = s->map; {
/* Pop the inactive front element. */
struct cuda_map *pop = s->map;
s->map = pop->next;
pop->next = NULL;
s->map = map; if (pop->size < size)
s->map->active = true; {
cuda_map_destroy (pop);
return s->map->d; map = cuda_map_create (size);
}
else
map = pop;
}
/* Check that the element is as expected. */
assert (map->next == NULL);
assert (!map->active);
/* Mark the element active. */
map->active = true;
/* Push the element to the back of the list. */
for (t = &s->map; (*t) != NULL; t = &(*t)->next)
;
assert (t != NULL && *t == NULL);
*t = map;
return map->d;
} }
/* Target data function launch information. */ /* Target data function launch information. */

View File

@ -0,0 +1,62 @@
/* { dg-do run { target openacc_nvidia_accel_selected } } */
/* { dg-additional-options "-lcuda" } */
#include <openacc.h>
#include <stdlib.h>
#include "cuda.h"
#include <stdio.h>
#define n 128
int
main (void)
{
CUresult r;
CUstream stream1;
int N = n;
int a[n];
int b[n];
int c[n];
acc_init (acc_device_nvidia);
r = cuStreamCreate (&stream1, CU_STREAM_NON_BLOCKING);
if (r != CUDA_SUCCESS)
{
fprintf (stderr, "cuStreamCreate failed: %d\n", r);
abort ();
}
acc_set_cuda_stream (1, stream1);
for (int i = 0; i < n; i++)
{
a[i] = 3;
c[i] = 0;
}
#pragma acc data copy (a, b, c) copyin (N)
{
#pragma acc parallel async (1)
;
#pragma acc parallel async (1) num_gangs (320)
#pragma loop gang
for (int ii = 0; ii < N; ii++)
c[ii] = (a[ii] + a[N - ii - 1]);
#pragma acc parallel async (1)
#pragma acc loop seq
for (int ii = 0; ii < n; ii++)
a[ii] = 6;
#pragma acc wait (1)
}
for (int i = 0; i < n; i++)
if (c[i] != 6)
abort ();
return 0;
}