4
0
mirror of git://sourceware.org/git/newlib-cygwin.git synced 2025-01-23 15:40:14 +08:00
Jeff Johnston 1787e9d033 AMD GCN Port contributed by Andrew Stubbs <ams@codesourcery.com>
Add support for the AMD GCN GPU architecture.  This is primarily intended for
use with OpenMP and OpenACC offloading.  It can also be used for stand-alone
programs, but this is intended mostly for testing the compiler and is not
expected to be useful in general.

The GPU architecture is highly parallel, and therefore Newlib must be
configured to use dynamic re-entrancy, and thread-safe malloc.

The only I/O available is a via a shared-memory interface provided by libgomp
and the gcn-run tool included with GCC.  At this time this is limited to
stdout, argc/argv, and the return code.
2019-01-15 10:48:08 -05:00

89 lines
2.5 KiB
C

/*
* Support file for amdgcn in newlib.
* Copyright (c) 2014, 2017 Mentor Graphics.
*
* The authors hereby grant permission to use, copy, modify, distribute,
* and license this software and its documentation for any purpose, provided
* that existing copyright notices are retained in all copies and that this
* notice is included verbatim in any distributions. No written agreement,
* license, or royalty fee is required for any of the authorized uses.
* Modifications to this software may be copyrighted by their authors
* and need not follow the licensing terms described here, provided that
* the new terms are clearly indicated on the first page of each file where
* they apply.
*/
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
/* This struct must match the one used by gcn-run and libgomp.
It holds all the data output from a kernel (besides mapping data).
The base address pointer can be found at kernargs+16.
The next_output counter must be atomically incremented for each
print output. Only when the print data is fully written can the
"written" flag be set. */
struct output {
int return_value;
int next_output;
struct printf_data {
int written;
char msg[128];
int type;
union {
int64_t ivalue;
double dvalue;
char text[128];
};
} queue[1000];
};
_READ_WRITE_RETURN_TYPE write (int fd, const void *buf, size_t count)
{
if (fd != 1 && fd != 2)
{
errno = EBADF;
return -1;
}
/* The output data is at ((void*)kernargs)[2]. */
register void **kernargs asm("s8");
struct output *data = (struct output *)kernargs[2];
/* Each output slot allows 256 bytes, so reserve as many as we need. */
int slot_count = ((count+1)/256)+1;
int index = __atomic_fetch_add (&data->next_output, slot_count,
__ATOMIC_ACQUIRE);
for (int c = count;
c >= 0 && index < 1000;
buf += 256, c -= 256, index++)
{
if (c < 128)
{
memcpy (data->queue[index].msg, buf, c);
data->queue[index].msg[c] = '\0';
data->queue[index].text[0] = '\0';
}
else if (c < 256)
{
memcpy (data->queue[index].msg, buf, 128);
memcpy (data->queue[index].text, buf+128, c-128);
data->queue[index].text[c-128] = '\0';
}
else
{
memcpy (data->queue[index].msg, buf, 128);
memcpy (data->queue[index].text, buf+128, 128);
}
data->queue[index].type = 3; /* Raw. */
__atomic_store_n (&data->queue[index].written, 1, __ATOMIC_RELEASE);
}
return count;
}