summaryrefslogtreecommitdiffstats
path: root/dma.c
diff options
context:
space:
mode:
Diffstat (limited to 'dma.c')
-rw-r--r--dma.c171
1 files changed, 103 insertions, 68 deletions
diff --git a/dma.c b/dma.c
index e3bc77c..ba99ba1 100644
--- a/dma.c
+++ b/dma.c
@@ -18,115 +18,152 @@
#include "pci.h"
#include "dma.h"
-const pcilib_dma_info_t *pcilib_get_dma_info(pcilib_t *ctx) {
- if (!ctx->dma_ctx) {
- pcilib_model_description_t *model_info = pcilib_get_model_description(ctx);
-
- if ((ctx->event_ctx)&&(model_info->event_api->init_dma)) {
- pcilib_map_register_space(ctx);
- ctx->dma_ctx = model_info->event_api->init_dma(ctx->event_ctx);
- } else if ((model_info->dma_api)&&(model_info->dma_api->init)) {
- pcilib_map_register_space(ctx);
- ctx->dma_ctx = model_info->dma_api->init(ctx, PCILIB_DMA_MODIFICATION_DEFAULT, NULL);
- }
-
- if (!ctx->dma_ctx) return NULL;
+const pcilib_dma_description_t *pcilib_get_dma_info(pcilib_t *ctx) {
+ int err;
+
+ err = pcilib_init_dma(ctx);
+ if (err) {
+ pcilib_error("Error (%i) while initializing DMA", err);
+ return NULL;
}
-
- return &ctx->dma_info;
+
+ if (!ctx->dma_ctx) return NULL;
+
+ return ctx->model_info.dma;
}
+
pcilib_dma_engine_t pcilib_find_dma_by_addr(pcilib_t *ctx, pcilib_dma_direction_t direction, pcilib_dma_engine_addr_t dma) {
pcilib_dma_engine_t i;
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA Engine is not configured in the current model");
return PCILIB_ERROR_NOTSUPPORTED;
}
-
- for (i = 0; info->engines[i]; i++) {
- if ((info->engines[i]->addr == dma)&&((info->engines[i]->direction&direction)==direction)) break;
+
+ for (i = 0; info->engines[i].addr_bits; i++) {
+ if ((info->engines[i].addr == dma)&&((info->engines[i].direction&direction)==direction)) break;
}
-
- if (info->engines[i]) return i;
+
+ if (info->engines[i].addr_bits) return i;
return PCILIB_DMA_ENGINE_INVALID;
}
-int pcilib_set_dma_engine_description(pcilib_t *ctx, pcilib_dma_engine_t engine, pcilib_dma_engine_description_t *desc) {
- ctx->dma_info.engines[engine] = desc;
+
+pcilib_dma_engine_t pcilib_add_dma_engine(pcilib_t *ctx, pcilib_dma_engine_description_t *desc) {
+ pcilib_dma_engine_t engine = ctx->num_engines++;
+ memcpy (&ctx->engines[engine], desc, sizeof(pcilib_dma_engine_description_t));
+ return engine;
+}
+
+
+int pcilib_init_dma(pcilib_t *ctx) {
+ int err;
+ pcilib_dma_context_t *dma_ctx = NULL;
+ const pcilib_model_description_t *model_info = pcilib_get_model_description(ctx);
+
+ if (ctx->dma_ctx)
+ return 0;
+
+
+ if ((ctx->event_ctx)&&(model_info->api)&&(model_info->api->init_dma)) {
+ err = pcilib_init_register_banks(ctx);
+ if (err) {
+ pcilib_error("Error (%i) while initializing register banks", err);
+ return err;
+ }
+
+ dma_ctx = model_info->api->init_dma(ctx->event_ctx);
+ } else if ((model_info->dma)&&(model_info->dma->api)&&(model_info->dma->api->init)) {
+ const pcilib_dma_description_t *dma = model_info->dma;
+
+ err = pcilib_init_register_banks(ctx);
+ if (err) {
+ pcilib_error("Error (%i) while initializing register banks", err);
+ return err;
+ }
+
+ dma_ctx = dma->api->init(ctx, (dma->model?dma->model:ctx->model), dma->args);
+ }
+
+ if (dma_ctx) {
+ dma_ctx->pcilib = ctx;
+ // DS: parameters?
+ ctx->dma_ctx = dma_ctx;
+ }
return 0;
}
int pcilib_start_dma(pcilib_t *ctx, pcilib_dma_engine_t dma, pcilib_dma_flags_t flags) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return PCILIB_ERROR_NOTSUPPORTED;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return PCILIB_ERROR_NOTAVAILABLE;
}
- if (!ctx->model_info.dma_api->start_dma) {
+ if (!ctx->model_info.dma->api->start_dma) {
return 0;
}
- return ctx->model_info.dma_api->start_dma(ctx->dma_ctx, dma, flags);
+ return ctx->model_info.dma->api->start_dma(ctx->dma_ctx, dma, flags);
}
int pcilib_stop_dma(pcilib_t *ctx, pcilib_dma_engine_t dma, pcilib_dma_flags_t flags) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return PCILIB_ERROR_NOTSUPPORTED;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return PCILIB_ERROR_NOTAVAILABLE;
}
- if (!ctx->model_info.dma_api->stop_dma) {
+ if (!ctx->model_info.dma->api->stop_dma) {
return 0;
}
- return ctx->model_info.dma_api->stop_dma(ctx->dma_ctx, dma, flags);
+ return ctx->model_info.dma->api->stop_dma(ctx->dma_ctx, dma, flags);
}
int pcilib_enable_irq(pcilib_t *ctx, pcilib_irq_type_t irq_type, pcilib_dma_flags_t flags) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
- if ((!info)||(!ctx->model_info.dma_api)||(!ctx->model_info.dma_api->enable_irq)) return 0;
+ if ((!info)||(!ctx->model_info.dma->api)||(!ctx->model_info.dma->api->enable_irq)) return 0;
- return ctx->model_info.dma_api->enable_irq(ctx->dma_ctx, irq_type, flags);
+ return ctx->model_info.dma->api->enable_irq(ctx->dma_ctx, irq_type, flags);
}
int pcilib_disable_irq(pcilib_t *ctx, pcilib_dma_flags_t flags) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
- if ((!info)||(!ctx->model_info.dma_api)||(!ctx->model_info.dma_api->disable_irq)) return 0;
+ if ((!info)||(!ctx->model_info.dma->api)||(!ctx->model_info.dma->api->disable_irq)) return 0;
- return ctx->model_info.dma_api->disable_irq(ctx->dma_ctx, flags);
+ return ctx->model_info.dma->api->disable_irq(ctx->dma_ctx, flags);
}
int pcilib_acknowledge_irq(pcilib_t *ctx, pcilib_irq_type_t irq_type, pcilib_irq_source_t irq_source) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
- if ((!info)||(!ctx->model_info.dma_api)||(!ctx->model_info.dma_api->acknowledge_irq)) return 0;
+ if ((!info)||(!ctx->model_info.dma->api)||(!ctx->model_info.dma->api->acknowledge_irq)) return 0;
- return ctx->model_info.dma_api->acknowledge_irq(ctx->dma_ctx, irq_type, irq_source);
+ return ctx->model_info.dma->api->acknowledge_irq(ctx->dma_ctx, irq_type, irq_source);
}
typedef struct {
size_t size;
void *data;
size_t pos;
-
+
pcilib_dma_flags_t flags;
} pcilib_dma_read_callback_context_t;
@@ -166,33 +203,34 @@ static int pcilib_dma_skip_callback(void *arg, pcilib_dma_flags_t flags, size_t
}
int pcilib_stream_dma(pcilib_t *ctx, pcilib_dma_engine_t dma, uintptr_t addr, size_t size, pcilib_dma_flags_t flags, pcilib_timeout_t timeout, pcilib_dma_callback_t cb, void *cbattr) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return PCILIB_ERROR_NOTSUPPORTED;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return PCILIB_ERROR_NOTAVAILABLE;
}
- if (!ctx->model_info.dma_api->stream) {
+ if (!ctx->model_info.dma->api->stream) {
pcilib_error("The DMA read is not supported by configured DMA engine");
return PCILIB_ERROR_NOTSUPPORTED;
}
-
- if (!info->engines[dma]) {
+
+ // DS: We should check we are not going outside of allocated engine space
+ if (!info->engines[dma].addr_bits) {
pcilib_error("The DMA engine (%i) is not supported by device", dma);
return PCILIB_ERROR_NOTAVAILABLE;
}
- if ((info->engines[dma]->direction&PCILIB_DMA_FROM_DEVICE) == 0) {
+ if ((info->engines[dma].direction&PCILIB_DMA_FROM_DEVICE) == 0) {
pcilib_error("The selected engine (%i) is S2C-only and does not support reading", dma);
return PCILIB_ERROR_NOTSUPPORTED;
}
- return ctx->model_info.dma_api->stream(ctx->dma_ctx, dma, addr, size, flags, timeout, cb, cbattr);
+ return ctx->model_info.dma->api->stream(ctx->dma_ctx, dma, addr, size, flags, timeout, cb, cbattr);
}
int pcilib_read_dma_custom(pcilib_t *ctx, pcilib_dma_engine_t dma, uintptr_t addr, size_t size, pcilib_dma_flags_t flags, pcilib_timeout_t timeout, void *buf, size_t *read_bytes) {
@@ -242,33 +280,34 @@ int pcilib_skip_dma(pcilib_t *ctx, pcilib_dma_engine_t dma) {
int pcilib_push_dma(pcilib_t *ctx, pcilib_dma_engine_t dma, uintptr_t addr, size_t size, pcilib_dma_flags_t flags, pcilib_timeout_t timeout, void *buf, size_t *written) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return PCILIB_ERROR_NOTSUPPORTED;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return PCILIB_ERROR_NOTAVAILABLE;
}
- if (!ctx->model_info.dma_api->push) {
+ if (!ctx->model_info.dma->api->push) {
pcilib_error("The DMA write is not supported by configured DMA engine");
return PCILIB_ERROR_NOTSUPPORTED;
}
- if (!info->engines[dma]) {
+ // DS: We should check we don't exceed allocated engine range
+ if (!info->engines[dma].addr_bits) {
pcilib_error("The DMA engine (%i) is not supported by device", dma);
return PCILIB_ERROR_NOTAVAILABLE;
}
- if ((info->engines[dma]->direction&PCILIB_DMA_TO_DEVICE) == 0) {
+ if ((info->engines[dma].direction&PCILIB_DMA_TO_DEVICE) == 0) {
pcilib_error("The selected engine (%i) is C2S-only and does not support writes", dma);
return PCILIB_ERROR_NOTSUPPORTED;
}
- return ctx->model_info.dma_api->push(ctx->dma_ctx, dma, addr, size, flags, timeout, buf, written);
+ return ctx->model_info.dma->api->push(ctx->dma_ctx, dma, addr, size, flags, timeout, buf, written);
}
@@ -277,51 +316,47 @@ int pcilib_write_dma(pcilib_t *ctx, pcilib_dma_engine_t dma, uintptr_t addr, siz
}
double pcilib_benchmark_dma(pcilib_t *ctx, pcilib_dma_engine_addr_t dma, uintptr_t addr, size_t size, size_t iterations, pcilib_dma_direction_t direction) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return 0;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return -1;
}
- if (!ctx->model_info.dma_api->benchmark) {
+ if (!ctx->model_info.dma->api->benchmark) {
pcilib_error("The DMA benchmark is not supported by configured DMA engine");
return -1;
- }
-
- if (!info->engines[dma]) {
- pcilib_error("The DMA engine (%i) is not supported by device", dma);
- return -1;
}
- return ctx->model_info.dma_api->benchmark(ctx->dma_ctx, dma, addr, size, iterations, direction);
+ return ctx->model_info.dma->api->benchmark(ctx->dma_ctx, dma, addr, size, iterations, direction);
}
int pcilib_get_dma_status(pcilib_t *ctx, pcilib_dma_engine_t dma, pcilib_dma_engine_status_t *status, size_t n_buffers, pcilib_dma_buffer_status_t *buffers) {
- const pcilib_dma_info_t *info = pcilib_get_dma_info(ctx);
+ const pcilib_dma_description_t *info = pcilib_get_dma_info(ctx);
if (!info) {
pcilib_error("DMA is not supported by the device");
return 0;
}
- if (!ctx->model_info.dma_api) {
+ if (!ctx->model_info.dma->api) {
pcilib_error("DMA Engine is not configured in the current model");
return -1;
}
- if (!ctx->model_info.dma_api->status) {
+ if (!ctx->model_info.dma->api->status) {
memset(status, 0, sizeof(pcilib_dma_engine_status_t));
return -1;
}
- if (!info->engines[dma]) {
+ // DS: We should check we don't exceed allocated engine range
+ if (!info->engines[dma].addr_bits) {
pcilib_error("The DMA engine (%i) is not supported by device", dma);
return -1;
}
- return ctx->model_info.dma_api->status(ctx->dma_ctx, dma, status, n_buffers, buffers);
+ return ctx->model_info.dma->api->status(ctx->dma_ctx, dma, status, n_buffers, buffers);
}