/*
* Copyright (c) 2010, Citrix Systems, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation;
* version 2.1 of the License.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; If not, see .
*/
#include
#include
#include "xc_private.h"
#include "xg_private.h"
xc_hypercall_buffer_t XC__HYPERCALL_BUFFER_NAME(HYPERCALL_BUFFER_NULL) = {
.hbuf = NULL,
.param_shadow = NULL,
HYPERCALL_BUFFER_INIT_NO_BOUNCE
};
void *xc__hypercall_buffer_alloc_pages(xc_interface *xch, xc_hypercall_buffer_t *b, int nr_pages)
{
void *p = xencall_alloc_buffer_pages(xch->xcall, nr_pages);
if (!p)
return NULL;
b->hbuf = p;
return b->hbuf;
}
void xc__hypercall_buffer_free_pages(xc_interface *xch, xc_hypercall_buffer_t *b, int nr_pages)
{
xencall_free_buffer_pages(xch->xcall, b->hbuf, nr_pages);
}
void *xc__hypercall_buffer_alloc(xc_interface *xch, xc_hypercall_buffer_t *b, size_t size)
{
void *p = xencall_alloc_buffer(xch->xcall, size);
if (!p)
return NULL;
b->hbuf = p;
return b->hbuf;
}
void xc__hypercall_buffer_free(xc_interface *xch, xc_hypercall_buffer_t *b)
{
xencall_free_buffer(xch->xcall, b->hbuf);
}
int xc__hypercall_bounce_pre(xc_interface *xch, xc_hypercall_buffer_t *b)
{
void *p;
/*
* Catch hypercall buffer declared other than with DECLARE_HYPERCALL_BOUNCE.
*/
if ( b->ubuf == (void *)-1 || b->dir == XC_HYPERCALL_BUFFER_BOUNCE_NONE )
abort();
/*
* Don't need to bounce a NULL buffer.
*/
if ( b->ubuf == NULL )
{
b->hbuf = NULL;
return 0;
}
p = xc__hypercall_buffer_alloc(xch, b, b->sz);
if ( p == NULL )
return -1;
if ( b->dir == XC_HYPERCALL_BUFFER_BOUNCE_IN || b->dir == XC_HYPERCALL_BUFFER_BOUNCE_BOTH )
memcpy(b->hbuf, b->ubuf, b->sz);
return 0;
}
void xc__hypercall_bounce_post(xc_interface *xch, xc_hypercall_buffer_t *b)
{
/*
* Catch hypercall buffer declared other than with DECLARE_HYPERCALL_BOUNCE.
*/
if ( b->ubuf == (void *)-1 || b->dir == XC_HYPERCALL_BUFFER_BOUNCE_NONE )
abort();
if ( b->hbuf == NULL )
return;
if ( b->dir == XC_HYPERCALL_BUFFER_BOUNCE_OUT || b->dir == XC_HYPERCALL_BUFFER_BOUNCE_BOTH )
memcpy(b->ubuf, b->hbuf, b->sz);
xc__hypercall_buffer_free(xch, b);
}
struct xc_hypercall_buffer_array {
unsigned max_bufs;
xc_hypercall_buffer_t *bufs;
};
xc_hypercall_buffer_array_t *xc_hypercall_buffer_array_create(xc_interface *xch,
unsigned n)
{
xc_hypercall_buffer_array_t *array;
xc_hypercall_buffer_t *bufs = NULL;
array = malloc(sizeof(*array));
if ( array == NULL )
goto error;
bufs = calloc(n, sizeof(*bufs));
if ( bufs == NULL )
goto error;
array->max_bufs = n;
array->bufs = bufs;
return array;
error:
free(bufs);
free(array);
return NULL;
}
void *xc__hypercall_buffer_array_alloc(xc_interface *xch,
xc_hypercall_buffer_array_t *array,
unsigned index,
xc_hypercall_buffer_t *hbuf,
size_t size)
{
void *buf;
if ( index >= array->max_bufs || array->bufs[index].hbuf )
abort();
buf = xc__hypercall_buffer_alloc(xch, hbuf, size);
if ( buf )
array->bufs[index] = *hbuf;
return buf;
}
void *xc__hypercall_buffer_array_get(xc_interface *xch,
xc_hypercall_buffer_array_t *array,
unsigned index,
xc_hypercall_buffer_t *hbuf)
{
if ( index >= array->max_bufs || array->bufs[index].hbuf == NULL )
abort();
*hbuf = array->bufs[index];
return array->bufs[index].hbuf;
}
void xc_hypercall_buffer_array_destroy(xc_interface *xc,
xc_hypercall_buffer_array_t *array)
{
unsigned i;
if ( array == NULL )
return;
for (i = 0; i < array->max_bufs; i++ )
xc__hypercall_buffer_free(xc, &array->bufs[i]);
free(array->bufs);
free(array);
}
/*
* Local variables:
* mode: C
* c-file-style: "BSD"
* c-basic-offset: 4
* tab-width: 4
* indent-tabs-mode: nil
* End:
*/