268 lines
10 KiB
C
268 lines
10 KiB
C
/*
|
||
* Copyright : (C) 2022 Phytium Information Technology, Inc.
|
||
* All Rights Reserved.
|
||
*
|
||
* This program is OPEN SOURCE software: you can redistribute it and/or modify it
|
||
* under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
|
||
* either version 1.0 of the License, or (at your option) any later version.
|
||
*
|
||
* This program 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 Phytium Public License for more details.
|
||
*
|
||
*
|
||
* FilePath: fgdma.h
|
||
* Date: 2022-02-10 14:53:42
|
||
* LastEditTime: 2022-02-18 08:25:35
|
||
* Description: This file is for gdma user api implementation
|
||
*
|
||
* Modify History:
|
||
* Ver Who Date Changes
|
||
* ----- ------ -------- --------------------------------------
|
||
* 1.0 huanghe 2021/11/5 init commit
|
||
* 1.1 zhugengyu 2022/5/16 modify according to tech manual.
|
||
*/
|
||
|
||
#ifndef FGDMA_H
|
||
#define FGDMA_H
|
||
|
||
#include "ftypes.h"
|
||
#include "fassert.h"
|
||
#include "ferror_code.h"
|
||
#include "fkernel.h"
|
||
|
||
#ifdef __cplusplus
|
||
extern "C"
|
||
{
|
||
#endif
|
||
|
||
/***************************** Include Files *********************************/
|
||
|
||
|
||
|
||
/************************** Constant Definitions *****************************/
|
||
|
||
/* gdma capacity mask */
|
||
|
||
#define FGDMA_IRQ1_MASK BIT(0) /* All Gdma channel share a single interrupt */
|
||
#define FGDMA_IRQ2_MASK BIT(1) /* Each gdma channel owns an independent interrupt */
|
||
#define FGDMA_TRANS_NEED_RESET_MASK BIT(2) /* Gdma needs to be reset before transmission */
|
||
|
||
typedef enum
|
||
{
|
||
FGDMA_OPER_NONE_PRIORITY_POLL = -1, /* Priority = -1 表示读写请求仲裁模式为Poll */
|
||
FGDMA_OPER_PRIORITY0 = 0, /* Priority >= 0 表示读写请求仲裁模式为Qos */
|
||
FGDMA_OPER_PRIORITY1 = 1,
|
||
FGDMA_OPER_PRIORITY2 = 2,
|
||
FGDMA_OPER_PRIORITY3 = 3,
|
||
FGDMA_OPER_PRIORITY4 = 4,
|
||
FGDMA_OPER_PRIORITY5 = 5,
|
||
FGDMA_OPER_PRIORITY6 = 6,
|
||
FGDMA_OPER_PRIORITY7 = 7,
|
||
FGDMA_OPER_PRIORITY8 = 8,
|
||
FGDMA_OPER_PRIORITY9 = 9,
|
||
FGDMA_OPER_PRIORITY10 = 10,
|
||
FGDMA_OPER_PRIORITY11 = 11,
|
||
FGDMA_OPER_PRIORITY12 = 12,
|
||
FGDMA_OPER_PRIORITY13 = 13,
|
||
FGDMA_OPER_PRIORITY14 = 14,
|
||
FGDMA_OPER_PRIORITY15 = 15
|
||
} FGdmaOperPriority; /* Qos配置,值越高,优先级越高 */
|
||
|
||
typedef enum
|
||
{
|
||
FGDMA_OPER_DIRECT = 0, /* 直接操作模式 */
|
||
FGDMA_OPER_BDL /* BDL操作模式 */
|
||
} FGdmaOperMode; /* 支持的操作模式 */
|
||
|
||
typedef enum
|
||
{
|
||
FGDMA_WAIT_INTR = 0, /* 中断模式 */
|
||
FGDMA_WAIT_POLL /* 轮询模式 */
|
||
} FGdmaWaitEnd; /* 支持的等待传输完成的模式 */
|
||
|
||
typedef enum
|
||
{
|
||
FGDMA_BURST_SIZE_1_BYTE = 0,
|
||
FGDMA_BURST_SIZE_2_BYTE = 1,
|
||
FGDMA_BURST_SIZE_4_BYTE = 2,
|
||
FGDMA_BURST_SIZE_8_BYTE = 3,
|
||
FGDMA_BURST_SIZE_16_BYTE = 4
|
||
} FGdmaBurstSize; /* 所支持的读写请求burst size范围,其值本身是一个二次幂指数
|
||
* 即,单次burst传输的最大数据字节数 = 2^burst size */
|
||
|
||
#define FGDMA_GET_BURST_BYTE(burst_size) (1U << burst_size) /* 获取burst size所代表的单次burst传输的最大数据字节数 */
|
||
|
||
typedef enum
|
||
{
|
||
FGDMA_CHAN_EVT_FIFO_EMPTY = 0, /* 通道Fifo空事件 */
|
||
FGDMA_CHAN_EVT_FIFO_FULL, /* 通道Fifo满事件 */
|
||
FGDMA_CHAN_EVT_BDL_END, /* BDL模式下一个BDL条目传输完成 */
|
||
FGDMA_CHAN_EVT_TRANS_END, /* 所有传输数据完成 */
|
||
FGDMA_CHAN_EVT_BUSY, /* 前一次传输未完成,当前还处于传输中 */
|
||
|
||
FGDMA_CHAN_NUM_OF_EVT
|
||
} FGdmaChanEvtType; /* 通道中断事件 */
|
||
|
||
#define FGDMA_SUCCESS FT_SUCCESS
|
||
#define FGDMA_ERR_NOT_INIT FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 0)
|
||
#define FGDMA_ERR_CHAN_IN_USE FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 1)
|
||
#define FGDMA_ERR_CHAN_NOT_INIT FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 2)
|
||
#define FGDMA_ERR_INVALID_ADDR FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 3)
|
||
#define FGDMA_ERR_INVALID_SIZE FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 4)
|
||
#define FGDMA_ERR_BDL_NOT_ENOUGH FT_MAKE_ERRCODE(ErrModBsp, ErrGdma, 5)
|
||
|
||
#define FGDMA_ADDR_ALIGMENT 128U /* BDL链表的地址需要按128字节对齐 */
|
||
|
||
/**************************** Type Definitions *******************************/
|
||
typedef struct _FGdma FGdma;
|
||
typedef struct _FGdmaChan FGdmaChan;
|
||
|
||
typedef struct
|
||
{
|
||
u32 instance_id; /* GDMA控制器ID */
|
||
u32 irq_num[FGDMA_NUM_OF_CHAN]; /* GDMA控制器中断号 */
|
||
u32 irq_prority; /* GDMA控制器中断优先级 */
|
||
volatile uintptr_t base_addr; /* GDMA控制器基地址 */
|
||
FGdmaOperPriority rd_qos; /* 读操作优先级 */
|
||
FGdmaOperPriority wr_qos; /* 写操作优先级 */
|
||
u32 caps; /* driver capacity */
|
||
} FGdmaConfig; /* GDMA控制器配置 */
|
||
|
||
typedef struct
|
||
{
|
||
u32 src_addr_l; /* 0x0, 数据源地址低32位 */
|
||
u32 src_addr_h; /* 0x4, 数据源地址高32位 */
|
||
u32 dst_addr_l; /* 0x8, 数据目的地址低32位 */
|
||
u32 dst_addr_h; /* 0xc, 数据目的地址高32位 */
|
||
#define FGDMA_SRC_TC_BDL_BURST_SET(x) SET_REG32_BITS((x), 1U, 0U)
|
||
#define FGDMA_SRC_TC_BDL_SIZE_SET(x) SET_REG32_BITS((x), 6U, 4U)
|
||
#define FGDMA_SRC_TC_BDL_LEN_SET(x) SET_REG32_BITS((x), 15U, 8U)
|
||
u32 src_tc; /* 0x10, 源传输控制位 */
|
||
#define FGDMA_DST_TC_BDL_BURST_SET(x) SET_REG32_BITS((x), 1U, 0U)
|
||
#define FGDMA_DST_TC_BDL_SIZE_SET(x) SET_REG32_BITS((x), 6U, 4U)
|
||
#define FGDMA_DST_TC_BDL_LEN_SET(x) SET_REG32_BITS((x), 15U, 8U)
|
||
u32 dst_tc; /* 0x14, 目的传输控制 */
|
||
u32 total_bytes;/* 0x18, 传输数据总量,以Byte为单位 */
|
||
u32 ioc; /* 0x1c, 该条目传输完成中断产生控制位 */
|
||
} __attribute__((__packed__)) FGdmaBdlDesc; /* BDL描述符 */
|
||
|
||
FASSERT_STATIC(0x20U == sizeof(FGdmaBdlDesc));
|
||
|
||
typedef u32 FGdmaChanIndex;
|
||
|
||
typedef struct
|
||
{
|
||
FGdmaChanIndex chan_id; /* DMA通道ID */
|
||
FGdmaOperPriority rd_qos; /* DMA通道读Qos配置 */
|
||
FGdmaOperPriority wr_qos; /* DMA通道写Qos配置 */
|
||
FGdmaOperMode trans_mode; /* DMA通道的操作模式,直接模式或者BDL模式 */
|
||
FGdmaWaitEnd wait_mode; /* 等待传输完成信号的模式,中断模式或轮询模式 */
|
||
/* Direct模式有效 */
|
||
FGdmaBurstSize rd_size; /* DMA读请求的burst size */
|
||
FGdmaBurstSize wr_size; /* DMA写请求的burst size */
|
||
/* BDL模式有效 */
|
||
boolean roll_back; /* 循环模式,TRUE: 当前BDL列表完成后,从第一个BDL项从新开始传输 */
|
||
FGdmaBdlDesc *descs;
|
||
u32 total_desc_num;
|
||
u32 valid_desc_num;
|
||
} FGdmaChanConfig; /* DMA通道配置 */
|
||
|
||
typedef void (*FGdmaChanEvtHandler)(FGdmaChan *const chan, void *args);
|
||
|
||
|
||
typedef struct _FGdmaChan
|
||
{
|
||
FGdmaChanConfig config; /* DMA通道配置 */
|
||
FGdma *gdma; /* DMA控制器实例 */
|
||
FGdmaChanEvtHandler evt_handlers[FGDMA_CHAN_NUM_OF_EVT]; /* DMA通道事件回调函数 */
|
||
void *evt_handler_args[FGDMA_CHAN_NUM_OF_EVT]; /* DMA通道事件回调函数入参 */
|
||
} FGdmaChan; /* GDMA通道实例 */
|
||
|
||
typedef struct _FGdma
|
||
{
|
||
FGdmaConfig config; /* GDMA控制器配置 */
|
||
u32 is_ready; /* GDMA控制器初始化是否完成 */
|
||
FGdmaChan *chans[FGDMA_NUM_OF_CHAN]; /* GDMA通道实例,如果通道没有分配,值为NULL */
|
||
} FGdma; /* GDMA控制器实例 */
|
||
|
||
/************************** Variable Definitions *****************************/
|
||
|
||
/***************** Macros (Inline Functions) Definitions *********************/
|
||
/* 获取默认的通道配置 */
|
||
#define FGDMA_DEFAULT_DIRECT_CHAN_CONFIG(_chan_id)\
|
||
(FGdmaChanConfig){ \
|
||
.chan_id = (_chan_id),\
|
||
.rd_size = FGDMA_BURST_SIZE_4_BYTE,\
|
||
.wr_size = FGDMA_BURST_SIZE_4_BYTE,\
|
||
.rd_qos = FGDMA_OPER_NONE_PRIORITY_POLL,\
|
||
.wr_qos = FGDMA_OPER_NONE_PRIORITY_POLL,\
|
||
.trans_mode = FGDMA_OPER_DIRECT,\
|
||
.wait_mode = FGDMA_WAIT_INTR,\
|
||
.roll_back = FALSE\
|
||
}
|
||
|
||
#define FGDMA_DEFAULT_BDL_CHAN_CONFIG(_chan_id, _bdl_descs, _bdl_desc_num)\
|
||
(FGdmaChanConfig){ \
|
||
.chan_id = (_chan_id),\
|
||
.rd_size = FGDMA_BURST_SIZE_4_BYTE,\
|
||
.wr_size = FGDMA_BURST_SIZE_4_BYTE,\
|
||
.rd_qos = FGDMA_OPER_NONE_PRIORITY_POLL,\
|
||
.wr_qos = FGDMA_OPER_NONE_PRIORITY_POLL,\
|
||
.trans_mode = FGDMA_OPER_BDL,\
|
||
.wait_mode = FGDMA_WAIT_INTR,\
|
||
.roll_back = FALSE,\
|
||
.descs = _bdl_descs,\
|
||
.total_desc_num = _bdl_desc_num,\
|
||
.valid_desc_num = 0U\
|
||
}
|
||
|
||
/************************** Function Prototypes ******************************/
|
||
/* 获取GDMA控制器默认配置 */
|
||
const FGdmaConfig *FGdmaLookupConfig(u32 instance_id);
|
||
|
||
/* 初始化GDMA控制器实例 */
|
||
FError FGdmaCfgInitialize(FGdma *const instance_p, const FGdmaConfig *config_p);
|
||
|
||
/* 去初始化GDMA控制器实例 */
|
||
void FGdmaDeInitialize(FGdma *const instance_p);
|
||
|
||
/* 分配指定GDMA通道 */
|
||
FError FGdmaAllocateChan(FGdma *const instance_p, FGdmaChan *const chan_p,
|
||
const FGdmaChanConfig *config_p);
|
||
|
||
/* 释放GDMA通道 */
|
||
FError FGdmaDellocateChan(FGdmaChan *const chan_p);
|
||
|
||
/* 直接操作模式下发起DMA传输 */
|
||
FError FGdmaDirectTransfer(FGdmaChan *const chan_p, uintptr src_addr, uintptr dst_addr, fsize_t data_len);
|
||
|
||
/* 设置BDL描述符的一个条目 */
|
||
FError FGdmaAppendBDLEntry(FGdmaChan *const chan_p, uintptr src_addr, uintptr dst_addr, fsize_t data_len);
|
||
|
||
/* BDL操作模式下发起DMA传输 */
|
||
FError FGdmaBDLTransfer(FGdmaChan *const chan_p);
|
||
|
||
/* 使能启动GDMA控制器 */
|
||
FError FGdmaStart(FGdma *const instance_p);
|
||
|
||
/* 停止GDMA控制器 */
|
||
FError FGdmaStop(FGdma *const instance_p);
|
||
|
||
/* GDMA中断处理函数 */
|
||
void FGdmaIrqHandler(s32 vector, void *args);
|
||
void FGdmaIrqHandlerPrivateChannel(s32 vector, void *args);
|
||
|
||
/* 注册GDMA通道事件回调函数 */
|
||
void FGdmaChanRegisterEvtHandler(FGdmaChan *const chan_p, FGdmaChanEvtType evt,
|
||
FGdmaChanEvtHandler handler, void *handler_arg);
|
||
|
||
/* 打印当前的GDMA寄存器值, DEBUG模式下有效 */
|
||
void FGdmaDumpRegisterVals(uintptr base_addr, u32 max_chan);
|
||
|
||
#ifdef __cplusplus
|
||
}
|
||
#endif
|
||
|
||
#endif
|