llvm/offload/test/mapping/padding_not_mapped.c

// RUN: %libomptarget-compile-generic -fopenmp-version=51
// RUN: %libomptarget-run-fail-generic 2>&1 \
// RUN: | %fcheck-generic

// The host memory layout for the following program looks like this:
//
//   | 4 bytes | 4 bytes | 8 bytes |
//   |   s.x   |   s.y   |   s.z   |
//   `-----------------------------'
//
// s is always at least 8-byte aligned in host memory due to s.z, so
// libomptarget's device padding for map(s.y,s.z) always maps to host memory
// that includes s.x.  At one time, s.x appeared to be mapped as a result, but
// libomptarget has since been fixed not to consider device padding as mapped to
// host memory.

#include <omp.h>
#include <stdio.h>

int main() {
  struct S { int x; int y; double z; } s = {1, 2, 3};

  // CHECK: &s.x = 0x[[#%x,HOST_ADDR:]], size = [[#%u,SIZE:]]
  fprintf(stderr, "&s = %p\n", &s);
  fprintf(stderr, "&s.x = %p, size = %ld\n", &s.x, sizeof s.x);
  fprintf(stderr, "&s.y = %p\n", &s.y);
  fprintf(stderr, "&s.z = %p\n", &s.z);

  // CHECK: s.x is present: 0
  // CHECK: s.x = 1{{$}}
  #pragma omp target enter data map(alloc: s.y, s.z)
  int dev = omp_get_default_device();
  fprintf(stderr, "s.x is present: %d\n", omp_target_is_present(&s.x, dev));
  #pragma omp target update from(s.x) // should have no effect
  fprintf(stderr, "s.x = %d\n", s.x);

  // CHECK: omptarget message: device mapping required by 'present' map type modifier does not exist for host address 0x{{0*}}[[#HOST_ADDR]] ([[#SIZE]] bytes)
  // CHECK: omptarget error: Call to getTargetPointer returned null pointer ('present' map type modifier).
  // CHECK: omptarget fatal error 1: failure of target construct while offloading is mandatory
  #pragma omp target enter data map(present, alloc: s.x)

  return 0;
}