#include "smartpqi_includes.h"
MALLOC_DEFINE(M_SMARTPQI, "smartpqi", "Buffers for the smartpqi driver");
static void
os_dma_map(void *arg, bus_dma_segment_t *segs, int nseg, int error)
{
bus_addr_t *paddr = (bus_addr_t *)arg;
*paddr = segs[0].ds_addr;
}
int
os_dma_setup(pqisrc_softstate_t *softs)
{
DBG_FUNC("IN\n");
DBG_FUNC("OUT\n");
return PQI_STATUS_SUCCESS;
}
int
os_dma_destroy(pqisrc_softstate_t *softs)
{
DBG_FUNC("IN\n");
DBG_FUNC("OUT\n");
return PQI_STATUS_SUCCESS;
}
void
os_update_dma_attributes(pqisrc_softstate_t *softs)
{
DBG_FUNC("IN\n");
DBG_FUNC("OUT\n");
}
int
os_dma_mem_alloc(pqisrc_softstate_t *softs, struct dma_mem *dma_mem)
{
int ret = BSD_SUCCESS;
ASSERT(dma_mem->align >= 4);
if ((ret = bus_dma_tag_create(
softs->os_specific.pqi_parent_dmat,
dma_mem->align, 0,
BUS_SPACE_MAXADDR,
BUS_SPACE_MAXADDR,
NULL, NULL,
dma_mem->size,
1,
dma_mem->size,
0,
NULL, NULL,
&dma_mem->dma_tag)) != 0 ) {
DBG_ERR("can't allocate DMA tag with error = 0x%x\n", ret);
goto err_out;
}
if (!dma_mem->dma_tag) {
DBG_ERR("dma tag is NULL\n");
ret = ENOMEM;
goto err_out;
}
if ((ret = bus_dmamem_alloc(dma_mem->dma_tag, (void **)&dma_mem->virt_addr,
BUS_DMA_NOWAIT, &dma_mem->dma_map)) != 0) {
DBG_ERR("can't allocate DMA memory for required object \
with error = 0x%x\n", ret);
goto err_mem;
}
if((ret = bus_dmamap_load(dma_mem->dma_tag, dma_mem->dma_map,
dma_mem->virt_addr, dma_mem->size,
os_dma_map, &dma_mem->dma_addr, 0)) != 0) {
DBG_ERR("can't load DMA memory for required \
object with error = 0x%x\n", ret);
goto err_load;
}
memset(dma_mem->virt_addr, 0, dma_mem->size);
ret = bsd_status_to_pqi_status(ret);
return ret;
err_load:
if(dma_mem->virt_addr)
bus_dmamem_free(dma_mem->dma_tag, dma_mem->virt_addr,
dma_mem->dma_map);
err_mem:
if(dma_mem->dma_tag)
bus_dma_tag_destroy(dma_mem->dma_tag);
err_out:
DBG_FUNC("failed OUT\n");
ret = bsd_status_to_pqi_status(ret);
return ret;
}
void
os_dma_mem_free(pqisrc_softstate_t *softs, struct dma_mem *dma_mem)
{
if(dma_mem->dma_addr) {
bus_dmamap_unload(dma_mem->dma_tag, dma_mem->dma_map);
dma_mem->dma_addr = 0;
}
if(dma_mem->virt_addr) {
bus_dmamem_free(dma_mem->dma_tag, dma_mem->virt_addr,
dma_mem->dma_map);
dma_mem->virt_addr = NULL;
}
if(dma_mem->dma_tag) {
bus_dma_tag_destroy(dma_mem->dma_tag);
dma_mem->dma_tag = NULL;
}
}
void
*os_mem_alloc(pqisrc_softstate_t *softs, size_t size)
{
void *addr;
addr = malloc((unsigned long)size, M_SMARTPQI,
M_NOWAIT | M_ZERO);
return addr;
}
void
os_mem_free(pqisrc_softstate_t *softs, void *addr, size_t size)
{
free((void*)addr, M_SMARTPQI);
}
void
os_resource_free(pqisrc_softstate_t *softs)
{
if(softs->os_specific.pqi_parent_dmat)
bus_dma_tag_destroy(softs->os_specific.pqi_parent_dmat);
if (softs->os_specific.pqi_regs_res0 != NULL)
bus_release_resource(softs->os_specific.pqi_dev,
SYS_RES_MEMORY,
softs->os_specific.pqi_regs_rid0,
softs->os_specific.pqi_regs_res0);
}