From 25dc15f00e16b67826297222d31cc2c159a2b73e Mon Sep 17 00:00:00 2001 From: Thomas Benz Date: Thu, 14 Sep 2023 13:21:44 +0200 Subject: [PATCH] Add iDMA 2D driver support --- cheshire.mk | 1 + sw/include/dif/dma.h | 133 +++++++++++++++++++++++++++++++++++++++++ sw/include/regs/idma.h | 57 ++++++++++++++++++ sw/sw.mk | 3 +- 4 files changed, 192 insertions(+), 2 deletions(-) create mode 100644 sw/include/dif/dma.h create mode 100644 sw/include/regs/idma.h diff --git a/cheshire.mk b/cheshire.mk index 4f3dffa73..afbfcc482 100644 --- a/cheshire.mk +++ b/cheshire.mk @@ -21,6 +21,7 @@ CHS_LLC_DIR := $(shell $(BENDER) path axi_llc) OTPROOT := $(shell $(BENDER) path opentitan_peripherals) CLINTROOT := $(shell $(BENDER) path clint) AXI_VGA_ROOT := $(shell $(BENDER) path axi_vga) +IDMA_ROOT := $(shell $(BENDER) path idma) REGTOOL ?= $(CHS_REG_DIR)/vendor/lowrisc_opentitan/util/regtool.py diff --git a/sw/include/dif/dma.h b/sw/include/dif/dma.h new file mode 100644 index 000000000..f4e6f5763 --- /dev/null +++ b/sw/include/dif/dma.h @@ -0,0 +1,133 @@ +// Copyright 2022 ETH Zurich and University of Bologna. +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Alessandro Ottaviano +// Thomas Benz + +#include +#include "regs/idma.h" +#include "params.h" + +#define DMA_SRC_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_SRC_ADDR_REG_OFFSET) +#define DMA_DST_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_DST_ADDR_REG_OFFSET) +#define DMA_NUMBYTES_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_NUM_BYTES_REG_OFFSET) +#define DMA_CONF_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_CONF_REG_OFFSET) +#define DMA_STATUS_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_STATUS_REG_OFFSET) +#define DMA_NEXTID_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_NEXT_ID_REG_OFFSET) +#define DMA_DONE_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_DONE_REG_OFFSET) +#define DMA_SRC_STRIDE_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_STRIDE_SRC_REG_OFFSET) +#define DMA_DST_STRIDE_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_STRIDE_DST_REG_OFFSET) +#define DMA_NUM_REPS_ADDR(BASE) \ + (BASE + IDMA_REG64_2D_FRONTEND_NUM_REPETITIONS_REG_OFFSET) + +#define DMA_CONF_DECOUPLE 0 +#define DMA_CONF_DEBURST 0 +#define DMA_CONF_SERIALIZE 0 + +#define X(NAME, BASE_ADDR) \ + extern volatile uint64_t *NAME##_dma_src_ptr(void); \ + extern volatile uint64_t *NAME##_dma_dst_ptr(void); \ + extern volatile uint64_t *NAME##_dma_num_bytes_ptr(void); \ + extern volatile uint64_t *NAME##_dma_conf_ptr(void); \ + extern volatile uint64_t *NAME##_dma_status_ptr(void); \ + extern volatile uint64_t *NAME##_dma_nextid_ptr(void); \ + extern volatile uint64_t *NAME##_dma_done_ptr(void); \ + extern volatile uint64_t *NAME##_dma_src_stride_ptr(void); \ + extern volatile uint64_t *NAME##_dma_dst_stride_ptr(void); \ + extern volatile uint64_t *NAME##_dma_num_reps_ptr(void); \ + \ + extern uint64_t NAME##_dma_memcpy(uint64_t dst, uint64_t src, uint64_t size); \ + extern void NAME##_dma_blk_memcpy(uint64_t dst, uint64_t src, uint64_t size); \ + extern uint64_t NAME##_dma_2d_memcpy(uint64_t dst, uint64_t src, uint64_t size, uint64_t dst_stride, uint64_t src_stride, uint64_t num_reps); \ + extern void NAME##_dma_2d_blk_memcpy(uint64_t dst, uint64_t src, uint64_t size, uint64_t dst_stride, uint64_t src_stride, uint64_t num_reps); \ + \ + inline volatile uint64_t *NAME##_dma_src_ptr(void) { \ + return (volatile uint64_t *)DMA_SRC_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_dst_ptr(void) { \ + return (volatile uint64_t *)DMA_DST_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_num_bytes_ptr(void) { \ + return (volatile uint64_t *)DMA_NUMBYTES_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_conf_ptr(void) { \ + return (volatile uint64_t *)DMA_CONF_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_status_ptr(void) { \ + return (volatile uint64_t *)DMA_STATUS_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_nextid_ptr(void) { \ + return (volatile uint64_t *)DMA_NEXTID_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_done_ptr(void) { \ + return (volatile uint64_t *)DMA_DONE_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_src_stride_ptr(void) { \ + return (volatile uint64_t *)DMA_SRC_STRIDE_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_dst_stride_ptr(void) { \ + return (volatile uint64_t *)DMA_DST_STRIDE_ADDR(BASE_ADDR); \ + } \ + inline volatile uint64_t *NAME##_dma_num_reps_ptr(void) { \ + return (volatile uint64_t *)DMA_NUM_REPS_ADDR(BASE_ADDR); \ + } \ + \ + inline uint64_t NAME##_dma_memcpy(uint64_t dst, uint64_t src, uint64_t size) { \ + *(NAME##_dma_src_ptr()) = (uint64_t)src; \ + *(NAME##_dma_dst_ptr()) = (uint64_t)dst; \ + *(NAME##_dma_num_bytes_ptr()) = size; \ + *(NAME##_dma_num_reps_ptr()) = 0; \ + *(NAME##_dma_conf_ptr()) = \ + (DMA_CONF_DECOUPLE << IDMA_REG64_2D_FRONTEND_CONF_DECOUPLE_BIT) | \ + (DMA_CONF_DEBURST << IDMA_REG64_2D_FRONTEND_CONF_DEBURST_BIT) | \ + (DMA_CONF_SERIALIZE << IDMA_REG64_2D_FRONTEND_CONF_SERIALIZE_BIT); \ + return *(NAME##_dma_nextid_ptr()); \ + } \ + \ + inline void NAME##_dma_blk_memcpy(uint64_t dst, uint64_t src, uint64_t size) { \ + volatile uint64_t tf_id = NAME##_dma_memcpy(dst, src, size); \ + while (*(NAME##_dma_done_ptr()) != tf_id) { \ + asm volatile("nop"); \ + } \ + } \ + \ + inline uint64_t NAME##_dma_2d_memcpy(uint64_t dst, uint64_t src, uint64_t size, uint64_t dst_stride, \ + uint64_t src_stride, uint64_t num_reps) { \ + *(NAME##_dma_src_ptr()) = (uint64_t)src; \ + *(NAME##_dma_dst_ptr()) = (uint64_t)dst; \ + *(NAME##_dma_num_bytes_ptr()) = size; \ + *(NAME##_dma_conf_ptr()) = \ + (DMA_CONF_DECOUPLE << IDMA_REG64_2D_FRONTEND_CONF_DECOUPLE_BIT) | \ + (DMA_CONF_DEBURST << IDMA_REG64_2D_FRONTEND_CONF_DEBURST_BIT) | \ + (DMA_CONF_SERIALIZE << IDMA_REG64_2D_FRONTEND_CONF_SERIALIZE_BIT); \ + *(NAME##_dma_src_stride_ptr()) = src_stride; \ + *(NAME##_dma_dst_stride_ptr()) = dst_stride; \ + *(NAME##_dma_num_reps_ptr()) = num_reps; \ + return *(NAME##_dma_nextid_ptr()); \ + } \ + \ + inline void NAME##_dma_2d_blk_memcpy(uint64_t dst, uint64_t src, uint64_t size, uint64_t dst_stride, \ + uint64_t src_stride, uint64_t num_reps) { \ + volatile uint64_t tf_id = NAME##_dma_2d_memcpy(dst, src, size, dst_stride, src_stride, num_reps); \ + while (*(NAME##_dma_done_ptr()) != tf_id) { \ + asm volatile("nop"); \ + } \ + } \ + \ + inline uint64_t NAME##_dma_get_status(void) { \ + return *(NAME##_dma_status_ptr()); \ + } + +X(sys, __base_dma); + +#undef X diff --git a/sw/include/regs/idma.h b/sw/include/regs/idma.h new file mode 100644 index 000000000..7aa43052a --- /dev/null +++ b/sw/include/regs/idma.h @@ -0,0 +1,57 @@ +// Generated register defines for idma_reg64_2d_frontend + +// Copyright information found in source file: +// Copyright 2022 ETH Zurich and University of Bologna. + +// Licensing information found in source file: +// Licensed under Solderpad Hardware License, Version 0.51 +// SPDX-License-Identifier: SHL-0.51 + +#ifndef _IDMA_REG64_2D_FRONTEND_REG_DEFS_ +#define _IDMA_REG64_2D_FRONTEND_REG_DEFS_ + +#ifdef __cplusplus +extern "C" { +#endif +// Register width +#define IDMA_REG64_2D_FRONTEND_PARAM_REG_WIDTH 64 + +// Source Address +#define IDMA_REG64_2D_FRONTEND_SRC_ADDR_REG_OFFSET 0x0 + +// Destination Address +#define IDMA_REG64_2D_FRONTEND_DST_ADDR_REG_OFFSET 0x8 + +// Number of bytes +#define IDMA_REG64_2D_FRONTEND_NUM_BYTES_REG_OFFSET 0x10 + +// Configuration Register for DMA settings +#define IDMA_REG64_2D_FRONTEND_CONF_REG_OFFSET 0x18 +#define IDMA_REG64_2D_FRONTEND_CONF_DECOUPLE_BIT 0 +#define IDMA_REG64_2D_FRONTEND_CONF_DEBURST_BIT 1 +#define IDMA_REG64_2D_FRONTEND_CONF_SERIALIZE_BIT 2 + +// DMA Status +#define IDMA_REG64_2D_FRONTEND_STATUS_REG_OFFSET 0x20 +#define IDMA_REG64_2D_FRONTEND_STATUS_BUSY_BIT 0 + +// Next ID, launches transfer, returns 0 if transfer not set up properly. +#define IDMA_REG64_2D_FRONTEND_NEXT_ID_REG_OFFSET 0x28 + +// Get ID of finished transactions. +#define IDMA_REG64_2D_FRONTEND_DONE_REG_OFFSET 0x30 + +// Source Stride +#define IDMA_REG64_2D_FRONTEND_STRIDE_SRC_REG_OFFSET 0x38 + +// Destination Stride +#define IDMA_REG64_2D_FRONTEND_STRIDE_DST_REG_OFFSET 0x40 + +// Number of 2D repetitions +#define IDMA_REG64_2D_FRONTEND_NUM_REPETITIONS_REG_OFFSET 0x48 + +#ifdef __cplusplus +} // extern "C" +#endif +#endif // _IDMA_REG64_2D_FRONTEND_REG_DEFS_ +// End generated register defines for idma_reg64_2d_frontend \ No newline at end of file diff --git a/sw/sw.mk b/sw/sw.mk index 7c98cfc09..ed8032d6b 100644 --- a/sw/sw.mk +++ b/sw/sw.mk @@ -50,9 +50,7 @@ CHS_SW_DEPS_SRCS += $(wildcard $(OTPROOT)/sw/device/lib/dif/autogen/*.c) # Libraries # ############# -CHS_SW_INCLUDES ?= -I$(CHS_SW_DIR)/include $(CHS_SW_DEPS_INCS) CHS_SW_LIB_SRCS_S = $(wildcard $(CHS_SW_DIR)/lib/*.S $(CHS_SW_DIR)/lib/**/*.S) -CHS_SW_LIB_SRCS_C = $(wildcard $(CHS_SW_DIR)/lib/*.c $(CHS_SW_DIR)/lib/**/*.c) CHS_SW_LIB_SRCS_O = $(CHS_SW_DEPS_SRCS:.c=.o) $(CHS_SW_LIB_SRCS_S:.S=.o) $(CHS_SW_LIB_SRCS_C:.c=.o) CHS_SW_LIBS = $(CHS_SW_DIR)/lib/libcheshire.a @@ -77,6 +75,7 @@ endef $(eval $(call chs_sw_gen_hdr_rule,clint,$(CLINTROOT)/src/clint.hjson $(CLINTROOT)/.generated)) $(eval $(call chs_sw_gen_hdr_rule,serial_link,$(CHS_ROOT)/hw/serial_link.hjson $(CHS_SLINK_DIR)/.generated)) $(eval $(call chs_sw_gen_hdr_rule,axi_vga,$(AXI_VGA_ROOT)/data/axi_vga.hjson $(AXI_VGA_ROOT)/.generated)) +$(eval $(call chs_sw_gen_hdr_rule,idma,$(IDMA_ROOT)/src/frontends/register_64bit_2d/idma_reg64_2d_frontend.hjson)) $(eval $(call chs_sw_gen_hdr_rule,axi_llc,$(CHS_LLC_DIR)/data/axi_llc_regs.hjson)) $(eval $(call chs_sw_gen_hdr_rule,cheshire,$(CHS_ROOT)/hw/regs/cheshire_regs.hjson)) $(eval $(call chs_sw_gen_hdr_rule,axi_rt,$(CHS_ROOT)/hw/regs/axi_rt_regs.hjson))