44799f87c3
liboffloadmic/ * plugin/libgomp-plugin-intelmic.cpp (OFFLOAD_ACTIVE_WAIT_ENV): New define. (init): Set OFFLOAD_ACTIVE_WAIT env var to 0, if it is not set. * runtime/emulator/coi_common.h (PIPE_HOST_PATH): Replace with ... (PIPE_HOST2TGT_NAME): ... this. (PIPE_TARGET_PATH): Replace with ... (PIPE_TGT2HOST_NAME): ... this. (MALLOCN): New define. (READN): Likewise. (WRITEN): Likewise. (enum cmd_t): Replace CMD_RUN_FUNCTION with CMD_PIPELINE_RUN_FUNCTION. Add CMD_PIPELINE_CREATE, CMD_PIPELINE_DESTROY. * runtime/emulator/coi_device.cpp (engine_dir): New static variable. (pipeline_thread_routine): New static function. (COIProcessWaitForShutdown): Use global engine_dir instead of mic_dir. Rename pipe_host and pipe_target to pipe_host2tgt and pipe_tgt2host. If cmd is CMD_PIPELINE_CREATE, create a new thread for the pipeline. Remove cmd == CMD_RUN_FUNCTION case. * runtime/emulator/coi_device.h (COIERRORN): New define. * runtime/emulator/coi_host.cpp: Include set, map, queue. Replace typedefs with enums and structs. (struct Function): Remove name, add num_buffers, bufs_size, bufs_data_target, misc_data_len, misc_data, return_value_len, return_value, completion_event. (struct Callback): New. (struct Process): Remove pipeline. Add pipe_host2tgt and pipe_tgt2host. (struct Pipeline): Remove pipe_host and pipe_target. Add thread, destroy, is_destroyed, pipe_host2tgt_path, pipe_tgt2host_path, pipe_host2tgt, pipe_tgt2host, queue, process. (max_pipeline_num): New static variable. (pipelines): Likewise. (max_event_num): Likewise. (non_signalled_events): Likewise. (errored_events): Likewise. (callbacks): Likewise. (cleanup): Do not check tmp_dirs before free. (start_critical_section): New static function. (finish_critical_section): Likewise. (pipeline_is_destroyed): Likewise. (maybe_invoke_callback): Likewise. (signal_event): Likewise. (get_event_result): Likewise. (COIBufferCopy): Rename arguments according to headers. Add asserts. Use process' main pipes, instead of pipeline's pipes. Signal completion event. (COIBufferCreate): Rename arguments according to headers. Add asserts. Use process' main pipes, instead of pipeline's pipes. (COIBufferCreateFromMemory): Rename arguments according to headers. Add asserts. (COIBufferDestroy): Rename arguments according to headers. Add asserts. Use process' main pipes, instead of pipeline's pipes. (COIBufferGetSinkAddress): Rename arguments according to headers. Add asserts. (COIBufferMap): Rename arguments according to headers. Add asserts. Signal completion event. (COIBufferRead): Likewise. (COIBufferSetState): Likewise. (COIBufferUnmap): Likewise. (COIBufferWrite): Likewise. (COIEngineGetCount): Add assert. (COIEngineGetHandle): Rename arguments according to headers. Add assert. (COIEventWait): Rename arguments according to headers. Add asserts. Implement waiting for events with zero or infinite timeout. (COIEventRegisterCallback): New function. (pipeline_thread_routine): New static function. (COIPipelineCreate): Create a new thread for the pipeline. (COIPipelineDestroy): Exit pipeline thread. (COIPipelineRunFunction): Add the function into pipeline's queue, instead running it here. Wait for it's completion in case of synchronous execution. (COIProcessCreateFromMemory): Rename arguments according to headers. Add asserts. Create process' main pipes, instead of pipeline's pipes. (COIProcessDestroy): Rename arguments according to headers. Add asserts. Destroy all undestroyed pipelines. (COIProcessGetFunctionHandles): Rename arguments according to headers. Add asserts. Use process' main pipes, instead of pipeline's pipes. Remove useless function names. (COIProcessLoadLibraryFromMemory): Add asserts. Use process' main pipes, instead of pipeline's pipes. (COIProcessUnloadLibrary): Likewise. (COIEngineGetInfo): Add assert. * runtime/emulator/coi_host.h (COIERRORN): New define. From-SVN: r228248
1635 lines
44 KiB
C++
1635 lines
44 KiB
C++
/*
|
|
Copyright (c) 2014-2015 Intel Corporation. All Rights Reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in the
|
|
documentation and/or other materials provided with the distribution.
|
|
* Neither the name of Intel Corporation nor the names of its
|
|
contributors may be used to endorse or promote products derived
|
|
from this software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <set>
|
|
#include <map>
|
|
#include <queue>
|
|
|
|
#include "coi_host.h"
|
|
|
|
#include "coi_version_asm.h"
|
|
|
|
#define CYCLE_FREQUENCY 1000000000
|
|
|
|
enum buffer_t
|
|
{
|
|
BUFFER_NORMAL,
|
|
BUFFER_MEMORY
|
|
};
|
|
|
|
struct Engine
|
|
{
|
|
COI_ISA_TYPE type;
|
|
uint32_t index;
|
|
char *dir;
|
|
};
|
|
|
|
struct Function
|
|
{
|
|
void *ptr;
|
|
uint32_t num_buffers;
|
|
uint64_t *bufs_size;
|
|
void * *bufs_data_target;
|
|
uint16_t misc_data_len;
|
|
void *misc_data;
|
|
uint16_t return_value_len;
|
|
void *return_value;
|
|
COIEVENT completion_event;
|
|
};
|
|
|
|
struct Callback
|
|
{
|
|
COI_EVENT_CALLBACK ptr;
|
|
const void *data;
|
|
};
|
|
|
|
struct Process
|
|
{
|
|
pid_t pid;
|
|
int pipe_host2tgt;
|
|
int pipe_tgt2host;
|
|
Engine *engine;
|
|
void **functions;
|
|
};
|
|
|
|
struct Pipeline
|
|
{
|
|
pthread_t thread;
|
|
bool destroy;
|
|
bool is_destroyed;
|
|
char *pipe_host2tgt_path;
|
|
char *pipe_tgt2host_path;
|
|
int pipe_host2tgt;
|
|
int pipe_tgt2host;
|
|
std::queue<Function> queue;
|
|
Process *process;
|
|
};
|
|
|
|
struct Buffer
|
|
{
|
|
buffer_t type;
|
|
char *name;
|
|
int fd;
|
|
int fd_target;
|
|
uint64_t size;
|
|
void *data;
|
|
void *data_target;
|
|
Process *process;
|
|
};
|
|
|
|
|
|
/* Environment variables. */
|
|
extern char **environ;
|
|
|
|
/* List of directories for removing on exit. */
|
|
static char **tmp_dirs;
|
|
static unsigned tmp_dirs_num;
|
|
|
|
/* Number of emulated MIC engines. */
|
|
static long num_engines;
|
|
|
|
/* Number of the last COI pipeline. */
|
|
static uint32_t max_pipeline_num;
|
|
|
|
/* Set of undestroyed pipelines. */
|
|
static std::set<Pipeline *> pipelines;
|
|
|
|
/* Number of the last COI event, the event #0 is always signalled. */
|
|
static uint64_t max_event_num = 1;
|
|
|
|
/* Set of created COI events, which are not signalled. */
|
|
static std::set<uint64_t> non_signalled_events;
|
|
|
|
/* Set of COI events, which encountered errors. */
|
|
static std::map<uint64_t, COIRESULT> errored_events;
|
|
|
|
/* Set of registered callbacks, indexed by event number. */
|
|
static std::map<uint64_t, Callback> callbacks;
|
|
|
|
/* Mutex to sync parallel execution. */
|
|
static pthread_mutex_t mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
|
|
|
|
|
|
static COIRESULT
|
|
read_long_env (const char *env_name, long *var, long var_default)
|
|
{
|
|
char *str = getenv (env_name);
|
|
char *s;
|
|
|
|
if (!str || *str == '\0')
|
|
*var = var_default;
|
|
else
|
|
{
|
|
errno = 0;
|
|
*var = strtol (str, &s, 0);
|
|
if (errno != 0 || s == str || *s != '\0')
|
|
COIERROR ("Variable %s has invalid value.", env_name);
|
|
}
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
__attribute__((constructor))
|
|
static void
|
|
init ()
|
|
{
|
|
if (read_long_env (OFFLOAD_EMUL_NUM_ENV, &num_engines, 1) == COI_ERROR)
|
|
exit (0);
|
|
}
|
|
|
|
|
|
/* Helper function for directory removing. */
|
|
static COIRESULT remove_directory (char *path)
|
|
{
|
|
char *file;
|
|
struct dirent *entry;
|
|
struct stat statfile;
|
|
DIR *dir = opendir (path);
|
|
if (dir == NULL)
|
|
COIERROR ("Cannot open directory %s.", dir);
|
|
|
|
while (entry = readdir (dir))
|
|
{
|
|
if (!strcmp (entry->d_name, ".") || !strcmp (entry->d_name, ".."))
|
|
continue;
|
|
|
|
MALLOC (char *, file, strlen (path) + strlen (entry->d_name) + 2);
|
|
sprintf (file, "%s/%s", path, entry->d_name);
|
|
|
|
if (stat (file, &statfile) < 0)
|
|
COIERROR ("Cannot retrieve information about file %s.", file);
|
|
|
|
if (S_ISDIR (statfile.st_mode))
|
|
{
|
|
if (remove_directory (file) == COI_ERROR)
|
|
return COI_ERROR;
|
|
}
|
|
else
|
|
{
|
|
if (unlink (file) < 0)
|
|
COIERROR ("Cannot unlink file %s.", file);
|
|
}
|
|
|
|
free (file);
|
|
}
|
|
|
|
if (closedir (dir) < 0)
|
|
COIERROR ("Cannot close directory %s.", path);
|
|
if (rmdir (path) < 0)
|
|
COIERROR ("Cannot remove directory %s.", path);
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
__attribute__((destructor))
|
|
static void
|
|
cleanup ()
|
|
{
|
|
for (unsigned i = 0; i < tmp_dirs_num; i++)
|
|
{
|
|
remove_directory (tmp_dirs[i]);
|
|
free (tmp_dirs[i]);
|
|
}
|
|
free (tmp_dirs);
|
|
}
|
|
|
|
static COIRESULT
|
|
start_critical_section ()
|
|
{
|
|
if (pthread_mutex_lock (&mutex) != 0)
|
|
COIERROR ("Cannot lock mutex.");
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
static COIRESULT
|
|
finish_critical_section ()
|
|
{
|
|
if (pthread_mutex_unlock (&mutex) != 0)
|
|
COIERROR ("Cannot unlock mutex.");
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
static bool
|
|
pipeline_is_destroyed (const Pipeline *pipeline)
|
|
{
|
|
start_critical_section ();
|
|
bool res = pipeline->is_destroyed;
|
|
finish_critical_section ();
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
maybe_invoke_callback (const COIEVENT event, const COIRESULT result)
|
|
{
|
|
std::map<uint64_t, Callback>::iterator cb = callbacks.find (event.opaque[0]);
|
|
|
|
if (cb != callbacks.end ())
|
|
{
|
|
Callback callback = cb->second;
|
|
callback.ptr (event, result, callback.data);
|
|
callbacks.erase (cb);
|
|
}
|
|
}
|
|
|
|
static void
|
|
signal_event (const COIEVENT event, const COIRESULT result)
|
|
{
|
|
if (result != COI_SUCCESS)
|
|
errored_events.insert (std::pair <uint64_t, COIRESULT> (event.opaque[0],
|
|
result));
|
|
non_signalled_events.erase (event.opaque[0]);
|
|
|
|
maybe_invoke_callback (event, result);
|
|
}
|
|
|
|
static COIRESULT
|
|
get_event_result (const COIEVENT event)
|
|
{
|
|
COIRESULT res = COI_SUCCESS;
|
|
|
|
std::map<uint64_t, COIRESULT>::iterator ee
|
|
= errored_events.find (event.opaque[0]);
|
|
|
|
if (ee != errored_events.end ())
|
|
res = ee->second;
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
extern "C"
|
|
{
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferCopy, 1) (COIBUFFER in_DestBuffer,
|
|
COIBUFFER in_SourceBuffer,
|
|
uint64_t in_DestOffset,
|
|
uint64_t in_SourceOffset,
|
|
uint64_t in_Length,
|
|
COI_COPY_TYPE in_Type,
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIBufferCopy");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_DestBuffer != NULL);
|
|
assert (in_SourceBuffer != NULL);
|
|
assert (in_Type == COI_COPY_UNSPECIFIED);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *dest = (Buffer *) in_DestBuffer;
|
|
Buffer *source = (Buffer *) in_SourceBuffer;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Map buffers if needed. */
|
|
if (dest->data == 0 && dest->type == BUFFER_NORMAL)
|
|
if (COIBufferMap (in_DestBuffer, 0, dest->size, (COI_MAP_TYPE) 0,
|
|
0, 0, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
if (source->data == 0 && source->type == BUFFER_NORMAL)
|
|
if (COIBufferMap (in_SourceBuffer, 0, source->size, (COI_MAP_TYPE) 0,
|
|
0, 0, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
/* Copy data. */
|
|
if (source->data != 0 && dest->data != 0)
|
|
memcpy ((void *) ((uintptr_t) dest->data + in_DestOffset),
|
|
(void *) ((uintptr_t) source->data + in_SourceOffset), in_Length);
|
|
else
|
|
{
|
|
assert (dest->process == source->process);
|
|
|
|
Buffer *buffer;
|
|
cmd_t cmd = CMD_BUFFER_COPY;
|
|
|
|
/* Create intermediary buffer. */
|
|
if (COIBufferCreate (in_Length, COI_BUFFER_NORMAL, 0, 0, 1,
|
|
(COIPROCESS*) &dest->process,
|
|
(COIBUFFER *) &buffer) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
int pipe_host2tgt = dest->process->pipe_host2tgt;
|
|
int pipe_tgt2host = dest->process->pipe_tgt2host;
|
|
|
|
/* Copy from source to intermediary buffer. */
|
|
if (source->data == 0)
|
|
{
|
|
assert (source->data_target != 0);
|
|
|
|
/* Send data to target. */
|
|
WRITE (pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (pipe_host2tgt, &buffer->data_target, sizeof (void *));
|
|
WRITE (pipe_host2tgt, &source->data_target, sizeof (void *));
|
|
WRITE (pipe_host2tgt, &buffer->size, sizeof (uint64_t));
|
|
|
|
/* Receive data from target. */
|
|
READ (pipe_tgt2host, &cmd, sizeof (cmd_t));
|
|
}
|
|
else
|
|
{
|
|
if (COIBufferCopy ((COIBUFFER) buffer, in_SourceBuffer, 0,
|
|
in_SourceOffset, in_Length, in_Type, 0, 0, 0)
|
|
== COI_ERROR)
|
|
return COI_ERROR;
|
|
}
|
|
|
|
/* Copy from intermediary buffer to dest. */
|
|
if (dest->data == 0)
|
|
{
|
|
assert (dest->data_target != 0);
|
|
|
|
/* Send data to target. */
|
|
WRITE (pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (pipe_host2tgt, &dest->data_target, sizeof (void *));
|
|
WRITE (pipe_host2tgt, &buffer->data_target, sizeof (void *));
|
|
WRITE (pipe_host2tgt, &buffer->size, sizeof (uint64_t));
|
|
|
|
/* Receive data from target. */
|
|
READ (pipe_tgt2host, &cmd, sizeof (cmd_t));
|
|
}
|
|
else
|
|
{
|
|
if (COIBufferCopy (in_DestBuffer, (COIBUFFER) buffer, in_DestOffset,
|
|
0, in_Length, in_Type, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
}
|
|
|
|
/* Unmap on target and destroy intermediary buffer. */
|
|
if (COIBufferDestroy ((COIBUFFER) buffer) == COI_ERROR)
|
|
return COI_ERROR;
|
|
}
|
|
|
|
/* Unmap buffers if needed. */
|
|
if (dest->type == BUFFER_NORMAL)
|
|
if (COIBufferUnmap ((COIMAPINSTANCE) dest, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
if (source->type == BUFFER_NORMAL)
|
|
if (COIBufferUnmap ((COIMAPINSTANCE) source, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
finish_critical_section ();
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferCreate, 1) (uint64_t in_Size,
|
|
COI_BUFFER_TYPE in_Type,
|
|
uint32_t in_Flags,
|
|
const void *in_pInitData,
|
|
uint32_t in_NumProcesses,
|
|
const COIPROCESS *in_pProcesses,
|
|
COIBUFFER *out_pBuffer)
|
|
{
|
|
COITRACE ("COIBufferCreate");
|
|
|
|
char *shm_name;
|
|
int shm_fd;
|
|
const int ullong_max_len = 20;
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Type == COI_BUFFER_NORMAL);
|
|
assert ((in_Flags & COI_SINK_MEMORY) == 0);
|
|
assert ((in_Flags & COI_SAME_ADDRESS_SINKS) == 0);
|
|
assert ((in_Flags & COI_SAME_ADDRESS_SINKS_AND_SOURCE) == 0);
|
|
assert (in_pInitData == NULL);
|
|
assert (in_NumProcesses == 1);
|
|
assert (in_pProcesses != NULL);
|
|
assert (out_pBuffer != NULL);
|
|
|
|
/* Create shared memory with an unique name. */
|
|
MALLOC (char *, shm_name, strlen (SHM_NAME) + ullong_max_len + 1);
|
|
for (unsigned long long i = 0; i >= 0; i++)
|
|
{
|
|
sprintf (shm_name, SHM_NAME "%lu", i);
|
|
shm_fd = shm_open (shm_name, O_CLOEXEC | O_CREAT | O_EXCL | O_RDWR,
|
|
S_IRUSR | S_IWUSR);
|
|
if (shm_fd > 0)
|
|
break;
|
|
}
|
|
if (ftruncate (shm_fd, in_Size) < 0)
|
|
COIERROR ("Cannot truncate shared memory file.");
|
|
|
|
/* Create buffer. */
|
|
Buffer *buf = new Buffer;
|
|
buf->data = 0;
|
|
buf->fd = shm_fd;
|
|
buf->process = (Process *) in_pProcesses[0];
|
|
buf->size = in_Size;
|
|
buf->type = BUFFER_NORMAL;
|
|
STRDUP (buf->name, shm_name);
|
|
|
|
/* Map buffer on target. */
|
|
size_t len = strlen (buf->name) + 1;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Send data to target. */
|
|
const cmd_t cmd = CMD_BUFFER_MAP;
|
|
int pipe_host2tgt = buf->process->pipe_host2tgt;
|
|
WRITE (pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (pipe_host2tgt, &len, sizeof (size_t));
|
|
WRITE (pipe_host2tgt, buf->name, len);
|
|
WRITE (pipe_host2tgt, &buf->size, sizeof (uint64_t));
|
|
|
|
/* Receive data from target. */
|
|
int pipe_tgt2host = buf->process->pipe_tgt2host;
|
|
READ (pipe_tgt2host, &buf->fd_target, sizeof (int));
|
|
READ (pipe_tgt2host, &buf->data_target, sizeof (void *));
|
|
|
|
finish_critical_section ();
|
|
|
|
/* Prepare output arguments. */
|
|
*out_pBuffer = (COIBUFFER) buf;
|
|
|
|
/* Clean up. */
|
|
free (shm_name);
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferCreateFromMemory, 1) (uint64_t in_Size,
|
|
COI_BUFFER_TYPE in_Type,
|
|
uint32_t in_Flags,
|
|
void *in_Memory,
|
|
uint32_t in_NumProcesses,
|
|
const COIPROCESS *in_pProcesses,
|
|
COIBUFFER *out_pBuffer)
|
|
{
|
|
COITRACE ("COIBufferCreateFromMemory");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Type == COI_BUFFER_NORMAL);
|
|
assert ((in_Flags & COI_SAME_ADDRESS_SINKS) == 0);
|
|
assert ((in_Flags & COI_SAME_ADDRESS_SINKS_AND_SOURCE) == 0);
|
|
assert (in_NumProcesses == 1);
|
|
assert (in_pProcesses != NULL);
|
|
assert (out_pBuffer != NULL);
|
|
|
|
/* Create buffer. */
|
|
Buffer *buf = new Buffer;
|
|
buf->data = (in_Flags & COI_SINK_MEMORY) == 0 ? in_Memory : 0;
|
|
buf->data_target = (in_Flags & COI_SINK_MEMORY) != 0 ? in_Memory : 0;
|
|
buf->process = (Process *) in_pProcesses[0];
|
|
buf->size = in_Size;
|
|
buf->type = BUFFER_MEMORY;
|
|
|
|
/* Prepare output argument. */
|
|
*out_pBuffer = (COIBUFFER) buf;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferDestroy, 1) (COIBUFFER in_Buffer)
|
|
{
|
|
COITRACE ("COIBufferDestroy");
|
|
|
|
cmd_t cmd = CMD_BUFFER_UNMAP;
|
|
|
|
assert (in_Buffer != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buf = (Buffer *) in_Buffer;
|
|
|
|
/* Unmap buffer on host. */
|
|
if (buf->data != 0 && buf->type == BUFFER_NORMAL)
|
|
if (COIBufferUnmap ((COIMAPINSTANCE) in_Buffer, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
/* Unmap buffer on target. */
|
|
if (buf->data_target != 0)
|
|
{
|
|
start_critical_section ();
|
|
|
|
/* Send data to target. */
|
|
int pipe_host2tgt = buf->process->pipe_host2tgt;
|
|
WRITE (pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (pipe_host2tgt, &buf->fd_target, sizeof (int));
|
|
WRITE (pipe_host2tgt, &buf->data_target, sizeof (void *));
|
|
WRITE (pipe_host2tgt, &buf->size, sizeof (uint64_t));
|
|
|
|
/* Receive data from target. */
|
|
READ (buf->process->pipe_tgt2host, &cmd, sizeof (cmd_t));
|
|
|
|
finish_critical_section ();
|
|
}
|
|
|
|
/* Unlink shared memory. */
|
|
if (buf->type == BUFFER_NORMAL)
|
|
{
|
|
if (close (buf->fd) < 0)
|
|
COIERROR ("Cannot close shared memory file.");
|
|
if (shm_unlink (buf->name) < 0)
|
|
COIERROR ("Cannot unlink shared memory.");
|
|
free (buf->name);
|
|
}
|
|
|
|
/* Clean up. */
|
|
delete buf;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferGetSinkAddress, 1) (COIBUFFER in_Buffer,
|
|
uint64_t *out_pAddress)
|
|
{
|
|
COITRACE ("COIBufferGetSinkAddress");
|
|
|
|
assert (in_Buffer != NULL);
|
|
assert (out_pAddress != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buf = (Buffer *) in_Buffer;
|
|
|
|
/* Here should come BUFFER_NORMAL buffer. */
|
|
assert (buf->type == BUFFER_NORMAL);
|
|
|
|
/* Prepare output argument. */
|
|
*out_pAddress = (uint64_t) buf->data_target;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferMap, 1) (COIBUFFER in_Buffer,
|
|
uint64_t in_Offset,
|
|
uint64_t in_Length, // Ignored
|
|
COI_MAP_TYPE in_Type, // Ignored
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion,
|
|
COIMAPINSTANCE *out_pMapInstance,
|
|
void **out_ppData)
|
|
{
|
|
COITRACE ("COIBufferMap");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Offset == 0);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buf = (Buffer *) in_Buffer;
|
|
|
|
/* Only BUFFER_NORMAL buffers should come here. */
|
|
assert (buf->type == BUFFER_NORMAL);
|
|
|
|
/* Map shared memory. */
|
|
buf->data = mmap (NULL, buf->size, PROT_READ | PROT_WRITE,
|
|
MAP_SHARED, buf->fd, 0);
|
|
if (buf->data == NULL)
|
|
COIERROR ("Cannot map shared memory.");
|
|
|
|
/* Prepare output arguments. */
|
|
if (out_pMapInstance != 0)
|
|
*out_pMapInstance = (COIMAPINSTANCE) buf;
|
|
if (out_ppData != 0)
|
|
*out_ppData = buf->data;
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferRead, 1) (COIBUFFER in_SourceBuffer,
|
|
uint64_t in_Offset,
|
|
void *in_pDestData,
|
|
uint64_t in_Length,
|
|
COI_COPY_TYPE in_Type,
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIBufferRead");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_pDestData != NULL);
|
|
assert (in_Type == COI_COPY_UNSPECIFIED);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buf = (Buffer *) in_SourceBuffer;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Map buffers if needed. */
|
|
if (buf->data == 0 && buf->type == BUFFER_NORMAL)
|
|
if (COIBufferMap (in_SourceBuffer, 0, buf->size, (COI_MAP_TYPE) 0, 0, 0, 0,
|
|
0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
/* Copy data. */
|
|
memcpy (in_pDestData, (void *) ((uintptr_t) buf->data + in_Offset),
|
|
in_Length);
|
|
|
|
/* Unmap buffers if needed. */
|
|
if (buf->type == BUFFER_NORMAL)
|
|
if (COIBufferUnmap ((COIMAPINSTANCE) buf, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
finish_critical_section ();
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferSetState, 1) (COIBUFFER in_Buffer, // Ignored
|
|
COIPROCESS in_Process, // Ignored
|
|
COI_BUFFER_STATE in_State, // Ignored
|
|
COI_BUFFER_MOVE_FLAG in_DataMove,
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIBufferSetState");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_DataMove == COI_BUFFER_NO_MOVE);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Looks like we have nothing to do here. */
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferUnmap, 1) (COIMAPINSTANCE in_MapInstance,
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIBufferUnmap");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_MapInstance != NULL);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buffer = (Buffer *) in_MapInstance;
|
|
|
|
/* Only BUFFER_NORMAL buffers should come here. */
|
|
assert (buffer->type == BUFFER_NORMAL);
|
|
|
|
/* Unmap shared memory. */
|
|
if (munmap (buffer->data, buffer->size) < 0)
|
|
COIERROR ("Cannot unmap shared memory.");
|
|
|
|
buffer->data = 0;
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIBufferWrite, 1) (COIBUFFER in_DestBuffer,
|
|
uint64_t in_Offset,
|
|
const void *in_pSourceData,
|
|
uint64_t in_Length,
|
|
COI_COPY_TYPE in_Type,
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIBufferWrite");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_DestBuffer != NULL);
|
|
assert (in_pSourceData != NULL);
|
|
assert (in_Type == COI_COPY_UNSPECIFIED);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
/* Convert input arguments. */
|
|
Buffer *buf = (Buffer *) in_DestBuffer;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Map buffers if needed. */
|
|
if (buf->data == 0 && buf->type == BUFFER_NORMAL)
|
|
if (COIBufferMap (in_DestBuffer, 0, buf->size, (COI_MAP_TYPE) 0, 0, 0, 0, 0,
|
|
0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
/* Copy data. */
|
|
memcpy ((void *) ((uintptr_t) buf->data + in_Offset), in_pSourceData,
|
|
in_Length);
|
|
|
|
/* Unmap buffers if needed. */
|
|
if (buf->type == BUFFER_NORMAL)
|
|
if (COIBufferUnmap ((COIMAPINSTANCE) buf, 0, 0, 0) == COI_ERROR)
|
|
return COI_ERROR;
|
|
|
|
finish_critical_section ();
|
|
|
|
if (out_pCompletion)
|
|
out_pCompletion->opaque[0] = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIEngineGetCount, 1) (COI_ISA_TYPE isa,
|
|
uint32_t *count)
|
|
{
|
|
COITRACE ("COIEngineGetCount");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (isa == COI_ISA_MIC);
|
|
assert (count != NULL);
|
|
|
|
/* Prepare output arguments. */
|
|
*count = num_engines;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIEngineGetHandle, 1) (COI_ISA_TYPE in_ISA,
|
|
uint32_t in_EngineIndex,
|
|
COIENGINE *out_pEngineHandle)
|
|
{
|
|
COITRACE ("COIEngineGetHandle");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_ISA == COI_ISA_MIC);
|
|
assert (out_pEngineHandle != NULL);
|
|
|
|
/* Check engine index. */
|
|
if (in_EngineIndex >= num_engines)
|
|
COIERROR ("Wrong engine index.");
|
|
|
|
/* Create engine handle. */
|
|
Engine *engine = new Engine;
|
|
engine->dir = NULL;
|
|
engine->index = in_EngineIndex;
|
|
engine->type = in_ISA;
|
|
|
|
/* Prepare output argument. */
|
|
*out_pEngineHandle = (COIENGINE) engine;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIEventWait, 1) (uint16_t in_NumEvents,
|
|
const COIEVENT *in_pEvents,
|
|
int32_t in_TimeoutMilliseconds,
|
|
uint8_t in_WaitForAll,
|
|
uint32_t *out_pNumSignaled,
|
|
uint32_t *out_pSignaledIndices)
|
|
{
|
|
COITRACE ("COIEventWait");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_pEvents != NULL);
|
|
assert (in_TimeoutMilliseconds == 0 || in_TimeoutMilliseconds == -1);
|
|
assert (in_WaitForAll == 1);
|
|
assert (out_pNumSignaled == NULL);
|
|
assert (out_pSignaledIndices == NULL);
|
|
|
|
if (in_TimeoutMilliseconds == 0)
|
|
{
|
|
/* If some event is not signalled, return timeout error. */
|
|
for (uint16_t i = 0; i < in_NumEvents; i++)
|
|
if (non_signalled_events.count (in_pEvents[i].opaque[0]) > 0)
|
|
return COI_TIME_OUT_REACHED;
|
|
else
|
|
{
|
|
/* If the event signalled with an error, return that error. */
|
|
start_critical_section ();
|
|
COIRESULT res = get_event_result (in_pEvents[i]);
|
|
finish_critical_section ();
|
|
if (res != COI_SUCCESS)
|
|
return res;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Wait indefinitely for all events. */
|
|
for (uint16_t i = 0; i < in_NumEvents; i++)
|
|
{
|
|
while (non_signalled_events.count (in_pEvents[i].opaque[0]) > 0)
|
|
usleep (1000);
|
|
|
|
/* If the event signalled with an error, return that error. */
|
|
start_critical_section ();
|
|
COIRESULT res = get_event_result (in_pEvents[i]);
|
|
finish_critical_section ();
|
|
if (res != COI_SUCCESS)
|
|
return res;
|
|
}
|
|
}
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIEventRegisterCallback, 1) (const COIEVENT in_Event,
|
|
COI_EVENT_CALLBACK in_Callback,
|
|
const void *in_UserData,
|
|
const uint64_t in_Flags)
|
|
{
|
|
COITRACE ("COIEventRegisterCallback");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Callback != NULL);
|
|
assert (in_UserData != NULL);
|
|
assert (in_Flags == 0);
|
|
|
|
start_critical_section ();
|
|
if (non_signalled_events.count (in_Event.opaque[0]) == 0)
|
|
{
|
|
/* If the event is already signalled, invoke the callback immediately. */
|
|
COIRESULT res = get_event_result (in_Event);
|
|
in_Callback (in_Event, res, in_UserData);
|
|
}
|
|
else
|
|
{
|
|
Callback callback;
|
|
callback.ptr = in_Callback;
|
|
callback.data = in_UserData;
|
|
callbacks.insert (std::pair <uint64_t, Callback> (in_Event.opaque[0],
|
|
callback));
|
|
}
|
|
finish_critical_section ();
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
/* The start routine for the COI pipeline thread. */
|
|
|
|
static void *
|
|
pipeline_thread_routine (void *in_Pipeline)
|
|
{
|
|
/* Convert input arguments. */
|
|
Pipeline *pipeline = (Pipeline *) in_Pipeline;
|
|
|
|
/* Open pipes. */
|
|
pipeline->pipe_host2tgt
|
|
= open (pipeline->pipe_host2tgt_path, O_CLOEXEC | O_WRONLY);
|
|
if (pipeline->pipe_host2tgt < 0)
|
|
COIERRORN ("Cannot open host-to-target pipe.");
|
|
pipeline->pipe_tgt2host
|
|
= open (pipeline->pipe_tgt2host_path, O_CLOEXEC | O_RDONLY);
|
|
if (pipeline->pipe_tgt2host < 0)
|
|
COIERRORN ("Cannot open target-to-host pipe.");
|
|
|
|
free (pipeline->pipe_host2tgt_path);
|
|
free (pipeline->pipe_tgt2host_path);
|
|
pipeline->pipe_host2tgt_path = NULL;
|
|
pipeline->pipe_tgt2host_path = NULL;
|
|
|
|
while (!pipeline->destroy)
|
|
if (pipeline->queue.empty ())
|
|
usleep (1000);
|
|
else
|
|
{
|
|
Function func = pipeline->queue.front ();
|
|
start_critical_section ();
|
|
pipeline->queue.pop ();
|
|
finish_critical_section ();
|
|
|
|
/* Send data to target. */
|
|
cmd_t cmd = CMD_PIPELINE_RUN_FUNCTION;
|
|
WRITEN (pipeline->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITEN (pipeline->pipe_host2tgt, &func.ptr, sizeof (void *));
|
|
WRITEN (pipeline->pipe_host2tgt, &func.num_buffers, sizeof (uint32_t));
|
|
for (uint32_t i = 0; i < func.num_buffers; i++)
|
|
{
|
|
WRITEN (pipeline->pipe_host2tgt, &func.bufs_size[i],
|
|
sizeof (uint64_t));
|
|
WRITEN (pipeline->pipe_host2tgt, &func.bufs_data_target[i],
|
|
sizeof (void *));
|
|
}
|
|
WRITEN (pipeline->pipe_host2tgt, &func.misc_data_len,
|
|
sizeof (uint16_t));
|
|
if (func.misc_data_len > 0)
|
|
WRITEN (pipeline->pipe_host2tgt, func.misc_data, func.misc_data_len);
|
|
WRITEN (pipeline->pipe_host2tgt, &func.return_value_len,
|
|
sizeof (uint16_t));
|
|
|
|
delete [] func.bufs_size;
|
|
delete [] func.bufs_data_target;
|
|
|
|
/* Receive data from target. Wait for target function to complete,
|
|
whether it has any data to return or not. */
|
|
bool has_return_value = func.return_value_len > 0;
|
|
int ret_len
|
|
= read (pipeline->pipe_tgt2host,
|
|
has_return_value ? func.return_value : &cmd,
|
|
has_return_value ? func.return_value_len : sizeof (cmd_t));
|
|
if (ret_len == 0)
|
|
{
|
|
start_critical_section ();
|
|
signal_event (func.completion_event, COI_PROCESS_DIED);
|
|
pipeline->is_destroyed = true;
|
|
finish_critical_section ();
|
|
return NULL;
|
|
}
|
|
else if (ret_len != (has_return_value ? func.return_value_len
|
|
: sizeof (cmd_t)))
|
|
COIERRORN ("Cannot read from pipe.");
|
|
|
|
start_critical_section ();
|
|
signal_event (func.completion_event, COI_SUCCESS);
|
|
finish_critical_section ();
|
|
}
|
|
|
|
/* Send data to target. */
|
|
const cmd_t cmd = CMD_PIPELINE_DESTROY;
|
|
WRITEN (pipeline->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
|
|
/* Close pipes. */
|
|
if (close (pipeline->pipe_host2tgt) < 0)
|
|
COIERRORN ("Cannot close host-to-target pipe.");
|
|
if (close (pipeline->pipe_tgt2host) < 0)
|
|
COIERRORN ("Cannot close target-to-host pipe.");
|
|
|
|
start_critical_section ();
|
|
pipeline->is_destroyed = true;
|
|
finish_critical_section ();
|
|
return NULL;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIPipelineCreate, 1) (COIPROCESS in_Process,
|
|
COI_CPU_MASK in_Mask,
|
|
uint32_t in_StackSize, // Ignored
|
|
COIPIPELINE *out_pPipeline)
|
|
{
|
|
COITRACE ("COIPipelineCreate");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Process != NULL);
|
|
assert (in_Mask == 0);
|
|
assert (out_pPipeline != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Process *proc = (Process *) in_Process;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Create pipeline handle. */
|
|
Pipeline *pipeline = new Pipeline;
|
|
pipeline->destroy = false;
|
|
pipeline->is_destroyed = false;
|
|
pipeline->process = proc;
|
|
pipelines.insert (pipeline);
|
|
|
|
/* Create pipes. */
|
|
uint32_t pipeline_num = max_pipeline_num++;
|
|
char *eng_dir = pipeline->process->engine->dir;
|
|
MALLOC (char *, pipeline->pipe_host2tgt_path,
|
|
strlen (eng_dir) + sizeof (PIPE_HOST2TGT_NAME "0000000000"));
|
|
MALLOC (char *, pipeline->pipe_tgt2host_path,
|
|
strlen (eng_dir) + sizeof (PIPE_TGT2HOST_NAME "0000000000"));
|
|
sprintf (pipeline->pipe_host2tgt_path, "%s" PIPE_HOST2TGT_NAME "%010d",
|
|
eng_dir, pipeline_num);
|
|
sprintf (pipeline->pipe_tgt2host_path, "%s" PIPE_TGT2HOST_NAME "%010d",
|
|
eng_dir, pipeline_num);
|
|
if (mkfifo (pipeline->pipe_host2tgt_path, S_IRUSR | S_IWUSR) < 0)
|
|
COIERROR ("Cannot create pipe %s.", pipeline->pipe_host2tgt_path);
|
|
if (mkfifo (pipeline->pipe_tgt2host_path, S_IRUSR | S_IWUSR) < 0)
|
|
COIERROR ("Cannot create pipe %s.", pipeline->pipe_tgt2host_path);
|
|
|
|
/* Send data to target. */
|
|
const cmd_t cmd = CMD_PIPELINE_CREATE;
|
|
WRITE (proc->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (proc->pipe_host2tgt, &pipeline_num, sizeof (pipeline_num));
|
|
|
|
/* Create a new thread for the pipeline. */
|
|
if (pthread_create (&pipeline->thread, NULL, pipeline_thread_routine,
|
|
pipeline))
|
|
COIERROR ("Cannot create new thread.");
|
|
|
|
finish_critical_section ();
|
|
|
|
/* Prepare output arguments. */
|
|
*out_pPipeline = (COIPIPELINE) pipeline;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIPipelineDestroy, 1) (COIPIPELINE in_Pipeline)
|
|
{
|
|
COITRACE ("COIPipelineDestroy");
|
|
|
|
assert (in_Pipeline != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Pipeline *pipeline = (Pipeline *) in_Pipeline;
|
|
|
|
start_critical_section ();
|
|
/* Remove pipeline from the set of undestroyed pipelines. */
|
|
pipelines.erase (pipeline);
|
|
|
|
/* Exit pipeline thread. */
|
|
pipeline->destroy = true;
|
|
finish_critical_section ();
|
|
|
|
while (!pipeline_is_destroyed (pipeline))
|
|
usleep (1000);
|
|
|
|
/* Join with a destroyed thread. */
|
|
if (pthread_join (pipeline->thread, NULL))
|
|
COIERROR ("Cannot join with a thread.");
|
|
|
|
delete pipeline;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIPipelineRunFunction, 1) (COIPIPELINE in_Pipeline,
|
|
COIFUNCTION in_Function,
|
|
uint32_t in_NumBuffers,
|
|
const COIBUFFER *in_Buffers,
|
|
const COI_ACCESS_FLAGS *in_pBufferAccessFlags, // Ignored
|
|
uint32_t in_NumDependencies,
|
|
const COIEVENT *in_pDependencies, // Ignored
|
|
const void *in_pMiscData,
|
|
uint16_t in_MiscDataLen,
|
|
void *out_pAsyncReturnValue,
|
|
uint16_t in_AsyncReturnValueLen,
|
|
COIEVENT *out_pCompletion)
|
|
{
|
|
COITRACE ("COIPipelineRunFunction");
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Pipeline != NULL);
|
|
assert (in_Function != NULL);
|
|
assert (in_NumDependencies == 0);
|
|
|
|
Function func;
|
|
func.ptr = (void *) in_Function;
|
|
func.num_buffers = in_NumBuffers;
|
|
func.bufs_size = new uint64_t [in_NumBuffers];
|
|
func.bufs_data_target = new void * [in_NumBuffers];
|
|
for (uint32_t i = 0; i < in_NumBuffers; i++)
|
|
{
|
|
Buffer **bufs = (Buffer **) in_Buffers;
|
|
func.bufs_size[i] = bufs[i]->size;
|
|
func.bufs_data_target[i] = bufs[i]->data_target;
|
|
}
|
|
func.misc_data = (void *) in_pMiscData;
|
|
func.misc_data_len = in_MiscDataLen;
|
|
func.return_value = out_pAsyncReturnValue;
|
|
func.return_value_len = in_AsyncReturnValueLen;
|
|
|
|
start_critical_section ();
|
|
func.completion_event.opaque[0] = max_event_num++;
|
|
non_signalled_events.insert (func.completion_event.opaque[0]);
|
|
((Pipeline *) in_Pipeline)->queue.push (func);
|
|
finish_critical_section ();
|
|
|
|
/* In case of synchronous execution we have to wait for target. */
|
|
if (out_pCompletion == NULL)
|
|
COIEventWait (1, &func.completion_event, -1, 1, NULL, NULL);
|
|
else
|
|
*out_pCompletion = func.completion_event;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessCreateFromMemory, 1) (COIENGINE in_Engine,
|
|
const char *in_pBinaryName,
|
|
const void *in_pBinaryBuffer,
|
|
uint64_t in_BinaryBufferLength,
|
|
int in_Argc,
|
|
const char **in_ppArgv,
|
|
uint8_t in_DupEnv,
|
|
const char **in_ppAdditionalEnv,
|
|
uint8_t in_ProxyActive, // Ignored
|
|
const char *in_Reserved, // Ignored
|
|
uint64_t in_InitialBufferSpace, // Ignored
|
|
const char *in_LibrarySearchPath,
|
|
const char *in_FileOfOrigin, // Ignored
|
|
uint64_t in_FileOfOriginOffset, // Ignored
|
|
COIPROCESS *out_pProcess)
|
|
{
|
|
COITRACE ("COIProcessCreateFromMemory");
|
|
|
|
const int run_max_args_num = 128;
|
|
char *run_argv[run_max_args_num];
|
|
char *emul_run = getenv (OFFLOAD_EMUL_RUN_ENV);
|
|
const int uint_max_len = 11;
|
|
|
|
/* Features of liboffloadmic. */
|
|
assert (in_Engine != NULL);
|
|
assert (in_pBinaryName != NULL);
|
|
assert (in_pBinaryBuffer != NULL);
|
|
assert (in_Argc == 0);
|
|
assert (in_ppArgv == NULL);
|
|
assert (in_ppAdditionalEnv == NULL);
|
|
assert (in_LibrarySearchPath != NULL);
|
|
assert (out_pProcess != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Engine *eng = (Engine *) in_Engine;
|
|
|
|
/* Create temporary directory for engine files. */
|
|
assert (eng->dir == NULL);
|
|
STRDUP (eng->dir, ENGINE_PATH);
|
|
if (mkdtemp (eng->dir) == NULL)
|
|
COIERROR ("Cannot create temporary directory %s.", eng->dir);
|
|
|
|
/* Save path to engine directory for clean up on exit. */
|
|
tmp_dirs_num++;
|
|
tmp_dirs = (char **) realloc (tmp_dirs, tmp_dirs_num * sizeof (char *));
|
|
if (!tmp_dirs)
|
|
COIERROR ("Cannot allocate memory.");
|
|
STRDUP (tmp_dirs[tmp_dirs_num - 1], eng->dir);
|
|
|
|
/* Create target executable file. */
|
|
char *target_exe;
|
|
MALLOC (char *, target_exe, strlen (eng->dir) + strlen (in_pBinaryName) + 2);
|
|
sprintf (target_exe, "%s/%s", eng->dir, in_pBinaryName);
|
|
int fd = open (target_exe, O_CLOEXEC | O_CREAT | O_WRONLY, S_IRUSR | S_IWUSR);
|
|
if (fd < 0)
|
|
COIERROR ("Cannot create file %s.", target_exe);
|
|
FILE *file = fdopen (fd, "wb");
|
|
if (file == NULL)
|
|
COIERROR ("Cannot associate stream with file descriptor.");
|
|
if (fwrite (in_pBinaryBuffer, 1, in_BinaryBufferLength, file)
|
|
!= in_BinaryBufferLength)
|
|
COIERROR ("Cannot write in file %s.", target_exe);
|
|
if (fclose (file) != 0)
|
|
COIERROR ("Cannot close file %s.", target_exe);
|
|
|
|
/* Fix file permissions. */
|
|
if (chmod (target_exe, S_IRWXU) < 0)
|
|
COIERROR ("Cannot change permissions for file %s.", target_exe);
|
|
|
|
/* Create directory for pipes to prevent names collision. */
|
|
char *pipes_path;
|
|
MALLOC (char *, pipes_path, strlen (eng->dir) + sizeof (PIPES_PATH));
|
|
sprintf (pipes_path, "%s" PIPES_PATH, eng->dir);
|
|
if (mkdir (pipes_path, S_IRWXU) < 0)
|
|
COIERROR ("Cannot create folder %s.", pipes_path);
|
|
|
|
/* Create 2 main pipes for inter-process communication. */
|
|
char *pipe_host2tgt_path, *pipe_tgt2host_path;
|
|
MALLOC (char *, pipe_host2tgt_path,
|
|
strlen (eng->dir) + sizeof (PIPE_HOST2TGT_NAME "mainpipe"));
|
|
MALLOC (char *, pipe_tgt2host_path,
|
|
strlen (eng->dir) + sizeof (PIPE_TGT2HOST_NAME "mainpipe"));
|
|
sprintf (pipe_host2tgt_path, "%s" PIPE_HOST2TGT_NAME "mainpipe", eng->dir);
|
|
sprintf (pipe_tgt2host_path, "%s" PIPE_TGT2HOST_NAME "mainpipe", eng->dir);
|
|
if (mkfifo (pipe_host2tgt_path, S_IRUSR | S_IWUSR) < 0)
|
|
COIERROR ("Cannot create main pipe %s.", pipe_host2tgt_path);
|
|
if (mkfifo (pipe_tgt2host_path, S_IRUSR | S_IWUSR) < 0)
|
|
COIERROR ("Cannot create main pipe %s.", pipe_tgt2host_path);
|
|
|
|
/* Prepare argv. */
|
|
if (emul_run == NULL || strcmp (emul_run, "") == 0)
|
|
{
|
|
STRDUP (run_argv[0], target_exe);
|
|
run_argv[1] = (char *) NULL;
|
|
}
|
|
else
|
|
{
|
|
char *ptr, *tmp;
|
|
int i = 0;
|
|
STRDUP (tmp, emul_run);
|
|
char *tok = strtok_r (tmp, " ", &ptr);
|
|
while (tok != NULL)
|
|
{
|
|
if (i >= run_max_args_num)
|
|
COIERROR ("Run command has too many arguments.");
|
|
STRDUP (run_argv[i++], tok);
|
|
tok = strtok_r (NULL, " ", &ptr);
|
|
}
|
|
STRDUP (run_argv[i], target_exe);
|
|
run_argv[i + 1] = (char *) NULL;
|
|
free (tmp);
|
|
}
|
|
|
|
/* Prepare envp. */
|
|
int env_num = 0;
|
|
if (in_DupEnv == true)
|
|
while (environ[env_num++]);
|
|
env_num += 4; // LD_LIBRARY_PATH, MIC_DIR, MIC_INDEX, NULL
|
|
|
|
char **envp;
|
|
MALLOC (char **, envp, env_num * sizeof (char *));
|
|
|
|
int env_i = 0;
|
|
if (in_DupEnv == true)
|
|
for (unsigned i = 0; environ[i] != NULL; i++)
|
|
{
|
|
unsigned j;
|
|
char *env_name;
|
|
STRDUP (env_name, environ[i]);
|
|
for (j = 0; env_name[j] != '=' && env_name[j] != '\0'; j++);
|
|
env_name[j] = '\0';
|
|
if (strcmp (env_name, "LD_LIBRARY_PATH") != 0
|
|
&& strcmp (env_name, MIC_DIR_ENV) != 0
|
|
&& strcmp (env_name, MIC_INDEX_ENV) != 0)
|
|
STRDUP (envp[env_i++], environ[i]);
|
|
free (env_name);
|
|
}
|
|
|
|
MALLOC (char *, envp[env_i], strlen (MIC_DIR_ENV) + strlen (eng->dir) + 2);
|
|
sprintf (envp[env_i], "%s=%s", MIC_DIR_ENV, eng->dir);
|
|
|
|
MALLOC (char *, envp[env_i + 1], strlen (MIC_INDEX_ENV) + uint_max_len + 1);
|
|
sprintf (envp[env_i + 1], "%s=%u", MIC_INDEX_ENV, eng->index);
|
|
|
|
MALLOC (char *, envp[env_i + 2],
|
|
strlen ("LD_LIBRARY_PATH=") + strlen (in_LibrarySearchPath) + 1);
|
|
sprintf (envp[env_i + 2], "LD_LIBRARY_PATH=%s", in_LibrarySearchPath);
|
|
|
|
envp[env_i + 3] = (char *) NULL;
|
|
|
|
/* Create target process. */
|
|
pid_t pid = vfork ();
|
|
if (pid < 0)
|
|
COIERROR ("Cannot create child process.");
|
|
|
|
if (pid == 0)
|
|
{
|
|
/* Run target executable. */
|
|
if (execvpe (run_argv[0], run_argv, envp) == -1)
|
|
COIERROR ("Cannot execute file %s.", target_exe);
|
|
}
|
|
|
|
/* Open main pipes. */
|
|
int pipe_host2tgt = open (pipe_host2tgt_path, O_CLOEXEC | O_WRONLY);
|
|
if (pipe_host2tgt < 0)
|
|
COIERROR ("Cannot open host-to-target main pipe.");
|
|
int pipe_tgt2host = open (pipe_tgt2host_path, O_CLOEXEC | O_RDONLY);
|
|
if (pipe_tgt2host < 0)
|
|
COIERROR ("Cannot open target-to-host main pipe.");
|
|
|
|
/* Create process handle. */
|
|
Process *proc = new Process;
|
|
proc->pid = pid;
|
|
proc->pipe_host2tgt = pipe_host2tgt;
|
|
proc->pipe_tgt2host = pipe_tgt2host;
|
|
proc->engine = eng;
|
|
proc->functions = NULL;
|
|
|
|
/* Prepare output arguments. */
|
|
*out_pProcess = (COIPROCESS) proc;
|
|
|
|
/* Clean up. */
|
|
for (unsigned i = 0; run_argv[i] != NULL; i++)
|
|
free (run_argv[i]);
|
|
for (unsigned i = 0; envp[i] != NULL; i++)
|
|
free (envp[i]);
|
|
free (envp);
|
|
free (pipe_host2tgt_path);
|
|
free (pipe_tgt2host_path);
|
|
free (pipes_path);
|
|
free (target_exe);
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessCreateFromFile, 1) (COIENGINE in_Engine,
|
|
const char *in_pBinaryName,
|
|
int in_Argc,
|
|
const char **in_ppArgv,
|
|
uint8_t in_DupEnv,
|
|
const char **in_ppAdditionalEnv,
|
|
uint8_t in_ProxyActive,
|
|
const char *in_Reserved,
|
|
uint64_t in_BufferSpace,
|
|
const char *in_LibrarySearchPath,
|
|
COIPROCESS *out_pProcess)
|
|
{
|
|
COITRACE ("COIProcessCreateFromFile");
|
|
|
|
/* liboffloadmic with GCC compiled binaries should never go here. */
|
|
assert (false);
|
|
return COI_ERROR;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessDestroy, 1) (COIPROCESS in_Process,
|
|
int32_t in_WaitForMainTimeout, // Ignored
|
|
uint8_t in_ForceDestroy,
|
|
int8_t *out_pProcessReturn,
|
|
uint32_t *out_pTerminationCode)
|
|
{
|
|
COITRACE ("COIProcessDestroy");
|
|
|
|
assert (in_Process != NULL);
|
|
assert (out_pProcessReturn != NULL);
|
|
assert (out_pTerminationCode != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Process *proc = (Process *) in_Process;
|
|
|
|
/* Destroy all undestroyed pipelines. */
|
|
while (!pipelines.empty ())
|
|
{
|
|
std::set<Pipeline *>::iterator p = pipelines.begin ();
|
|
COIPipelineDestroy ((COIPIPELINE) *p);
|
|
}
|
|
|
|
/* Close main pipes. */
|
|
if (close (proc->pipe_host2tgt) < 0)
|
|
COIERROR ("Cannot close host-to-target main pipe.");
|
|
if (close (proc->pipe_tgt2host) < 0)
|
|
COIERROR ("Cannot close target-to-host main pipe.");
|
|
|
|
/* Shutdown target process by force. */
|
|
if (in_ForceDestroy)
|
|
kill (proc->pid, SIGTERM);
|
|
|
|
/* Clean up. */
|
|
free (proc->engine->dir);
|
|
free (proc->functions);
|
|
delete proc->engine;
|
|
delete proc;
|
|
|
|
/* Prepare output arguments. */
|
|
*out_pProcessReturn = 0;
|
|
*out_pTerminationCode = 0;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessGetFunctionHandles, 1) (COIPROCESS in_Process,
|
|
uint32_t in_NumFunctions,
|
|
const char **in_ppFunctionNameArray,
|
|
COIFUNCTION *out_pFunctionHandleArray)
|
|
{
|
|
COITRACE ("COIProcessGetFunctionHandles");
|
|
|
|
assert (in_Process != NULL);
|
|
assert (in_ppFunctionNameArray != NULL);
|
|
assert (out_pFunctionHandleArray != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Process *proc = (Process *) in_Process;
|
|
|
|
/* This function should be called once for the process. */
|
|
assert (proc->functions == NULL);
|
|
|
|
/* Create array of function pointers. Last element is 0, what shows the end
|
|
of the array. This array is used to free memory when process is
|
|
destroyed. */
|
|
proc->functions = (void **) calloc (in_NumFunctions + 1, sizeof (void *));
|
|
if (proc->functions == NULL)
|
|
COIERROR ("Cannot allocate memory.");
|
|
|
|
/* Get handles for functions. */
|
|
for (uint32_t i = 0; i < in_NumFunctions; i++)
|
|
{
|
|
size_t len = strlen (in_ppFunctionNameArray[i]) + 1;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Send data to target. */
|
|
const cmd_t cmd = CMD_GET_FUNCTION_HANDLE;
|
|
WRITE (proc->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (proc->pipe_host2tgt, &len, sizeof (size_t));
|
|
WRITE (proc->pipe_host2tgt, in_ppFunctionNameArray[i], len);
|
|
|
|
/* Receive data from target. */
|
|
void *fn_ptr;
|
|
READ (proc->pipe_tgt2host, &fn_ptr, sizeof (void *));
|
|
|
|
finish_critical_section ();
|
|
|
|
/* Save function pointer. */
|
|
proc->functions[i] = fn_ptr;
|
|
|
|
/* Prepare output arguments. */
|
|
out_pFunctionHandleArray[i] = (COIFUNCTION) fn_ptr;
|
|
}
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessLoadLibraryFromMemory, 2) (COIPROCESS in_Process,
|
|
const void *in_pLibraryBuffer,
|
|
uint64_t in_LibraryBufferLength,
|
|
const char *in_pLibraryName,
|
|
const char *in_LibrarySearchPath, // Ignored
|
|
const char *in_FileOfOrigin, // Ignored
|
|
uint64_t in_FileOfOriginOffset, // Ignored
|
|
uint32_t in_Flags, // Ignored
|
|
COILIBRARY *out_pLibrary)
|
|
{
|
|
COITRACE ("COIProcessLoadLibraryFromMemory");
|
|
|
|
assert (in_Process != NULL);
|
|
assert (in_pLibraryBuffer != NULL);
|
|
assert (out_pLibrary != NULL);
|
|
|
|
/* Convert input arguments. */
|
|
Process *proc = (Process *) in_Process;
|
|
|
|
/* Create target library file. */
|
|
char *lib_path;
|
|
size_t len = strlen (proc->engine->dir) + strlen (in_pLibraryName) + 2;
|
|
MALLOC (char *, lib_path, len);
|
|
sprintf (lib_path, "%s/%s", proc->engine->dir, in_pLibraryName);
|
|
int fd = open (lib_path, O_CLOEXEC | O_CREAT | O_WRONLY, S_IRUSR | S_IWUSR);
|
|
if (fd < 0)
|
|
COIERROR ("Cannot create file %s.", lib_path);
|
|
FILE *file = fdopen (fd, "wb");
|
|
if (file == NULL)
|
|
COIERROR ("Cannot associate stream with file descriptor.");
|
|
if (fwrite (in_pLibraryBuffer, 1, in_LibraryBufferLength, file)
|
|
!= in_LibraryBufferLength)
|
|
COIERROR ("Cannot write in file %s.", lib_path);
|
|
if (fclose (file) != 0)
|
|
COIERROR ("Cannot close file %s.", lib_path);
|
|
|
|
start_critical_section ();
|
|
|
|
/* Make target open library. */
|
|
const cmd_t cmd = CMD_OPEN_LIBRARY;
|
|
WRITE (proc->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (proc->pipe_host2tgt, &len, sizeof (size_t));
|
|
WRITE (proc->pipe_host2tgt, lib_path, len);
|
|
|
|
/* Receive data from target. */
|
|
void *handle;
|
|
READ (proc->pipe_tgt2host, &handle, sizeof (void *));
|
|
|
|
finish_critical_section ();
|
|
|
|
/* Clean up. */
|
|
free (lib_path);
|
|
|
|
*out_pLibrary = (COILIBRARY) handle;
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessRegisterLibraries, 1) (uint32_t in_NumLibraries, // Ignored
|
|
const void **in_ppLibraryArray, // Ignored
|
|
const uint64_t *in_pLibrarySizeArray, // Ignored
|
|
const char **in_ppFileOfOriginArray, // Ignored
|
|
const uint64_t *in_pFileOfOriginOffSetArray) // Ignored
|
|
{
|
|
COITRACE ("COIProcessRegisterLibraries");
|
|
|
|
/* Looks like we have nothing to do here. */
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIProcessUnloadLibrary, 1) (COIPROCESS in_Process,
|
|
COILIBRARY in_Library)
|
|
{
|
|
COITRACE ("COIProcessUnloadLibrary");
|
|
|
|
assert (in_Process != NULL);
|
|
assert (in_Library != NULL);
|
|
|
|
const cmd_t cmd = CMD_CLOSE_LIBRARY;
|
|
|
|
/* Convert input arguments. */
|
|
Process *proc = (Process *) in_Process;
|
|
|
|
start_critical_section ();
|
|
|
|
/* Make target close library. */
|
|
WRITE (proc->pipe_host2tgt, &cmd, sizeof (cmd_t));
|
|
WRITE (proc->pipe_host2tgt, &in_Library, sizeof (void *));
|
|
|
|
finish_critical_section ();
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
uint64_t
|
|
SYMBOL_VERSION (COIPerfGetCycleFrequency, 1) ()
|
|
{
|
|
COITRACE ("COIPerfGetCycleFrequency");
|
|
|
|
return (uint64_t) CYCLE_FREQUENCY;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIPipelineClearCPUMask, 1) (COI_CPU_MASK *in_Mask)
|
|
{
|
|
COITRACE ("COIPipelineClearCPUMask");
|
|
|
|
/* Looks like we have nothing to do here. */
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIPipelineSetCPUMask, 1) (COIPROCESS in_Process,
|
|
uint32_t in_CoreID,
|
|
uint8_t in_ThreadID,
|
|
COI_CPU_MASK *out_pMask)
|
|
{
|
|
COITRACE ("COIPipelineSetCPUMask");
|
|
|
|
/* Looks like we have nothing to do here. */
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
|
|
COIRESULT
|
|
SYMBOL_VERSION (COIEngineGetInfo, 1) (COIENGINE in_EngineHandle, // Ignored
|
|
uint32_t in_EngineInfoSize, // Ignored
|
|
COI_ENGINE_INFO *out_pEngineInfo)
|
|
{
|
|
COITRACE ("COIEngineGetInfo");
|
|
|
|
assert (out_pEngineInfo != NULL);
|
|
|
|
out_pEngineInfo->ISA = COI_ISA_x86_64;
|
|
out_pEngineInfo->NumCores = 1;
|
|
out_pEngineInfo->NumThreads = 8;
|
|
out_pEngineInfo->CoreMaxFrequency = SYMBOL_VERSION(COIPerfGetCycleFrequency,1)() / 1000000;
|
|
out_pEngineInfo->PhysicalMemory = 1024;
|
|
out_pEngineInfo->PhysicalMemoryFree = 1024;
|
|
out_pEngineInfo->SwapMemory = 1024;
|
|
out_pEngineInfo->SwapMemoryFree = 1024;
|
|
out_pEngineInfo->MiscFlags = COI_ENG_ECC_DISABLED;
|
|
|
|
return COI_SUCCESS;
|
|
}
|
|
|
|
} // extern "C"
|
|
|