Loading arch/arm/mach-ux500/devices-db8500.c +1 −11 Original line number Original line Diff line number Diff line Loading @@ -132,35 +132,25 @@ static struct resource dma40_resources[] = { /* Default configuration for physcial memcpy */ /* Default configuration for physcial memcpy */ struct stedma40_chan_cfg dma40_memcpy_conf_phy = { struct stedma40_chan_cfg dma40_memcpy_conf_phy = { .channel_type = (STEDMA40_CHANNEL_IN_PHY_MODE | .mode = STEDMA40_MODE_PHYSICAL, STEDMA40_LOW_PRIORITY_CHANNEL | STEDMA40_PCHAN_BASIC_MODE), .dir = STEDMA40_MEM_TO_MEM, .dir = STEDMA40_MEM_TO_MEM, .src_info.endianess = STEDMA40_LITTLE_ENDIAN, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.psize = STEDMA40_PSIZE_PHY_1, .src_info.psize = STEDMA40_PSIZE_PHY_1, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.endianess = STEDMA40_LITTLE_ENDIAN, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.psize = STEDMA40_PSIZE_PHY_1, .dst_info.psize = STEDMA40_PSIZE_PHY_1, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, }; }; /* Default configuration for logical memcpy */ /* Default configuration for logical memcpy */ struct stedma40_chan_cfg dma40_memcpy_conf_log = { struct stedma40_chan_cfg dma40_memcpy_conf_log = { .channel_type = (STEDMA40_CHANNEL_IN_LOG_MODE | STEDMA40_LOW_PRIORITY_CHANNEL | STEDMA40_LCHAN_SRC_LOG_DST_LOG | STEDMA40_NO_TIM_FOR_LINK), .dir = STEDMA40_MEM_TO_MEM, .dir = STEDMA40_MEM_TO_MEM, .src_info.endianess = STEDMA40_LITTLE_ENDIAN, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.psize = STEDMA40_PSIZE_LOG_1, .src_info.psize = STEDMA40_PSIZE_LOG_1, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.endianess = STEDMA40_LITTLE_ENDIAN, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.psize = STEDMA40_PSIZE_LOG_1, .dst_info.psize = STEDMA40_PSIZE_LOG_1, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, Loading arch/arm/plat-nomadik/include/plat/ste_dma40.h +21 −39 Original line number Original line Diff line number Diff line Loading @@ -18,37 +18,20 @@ #define STEDMA40_DEV_DST_MEMORY (-1) #define STEDMA40_DEV_DST_MEMORY (-1) #define STEDMA40_DEV_SRC_MEMORY (-1) #define STEDMA40_DEV_SRC_MEMORY (-1) /* enum stedma40_mode { * Description of bitfields of channel_type variable is available in STEDMA40_MODE_LOGICAL = 0, * the info structure. STEDMA40_MODE_PHYSICAL, */ STEDMA40_MODE_OPERATION, }; /* Priority */ enum stedma40_mode_opt { #define STEDMA40_INFO_PRIO_TYPE_POS 2 STEDMA40_PCHAN_BASIC_MODE = 0, #define STEDMA40_HIGH_PRIORITY_CHANNEL (0x1 << STEDMA40_INFO_PRIO_TYPE_POS) STEDMA40_LCHAN_SRC_LOG_DST_LOG = 0, #define STEDMA40_LOW_PRIORITY_CHANNEL (0x2 << STEDMA40_INFO_PRIO_TYPE_POS) STEDMA40_PCHAN_MODULO_MODE, STEDMA40_PCHAN_DOUBLE_DST_MODE, /* Mode */ STEDMA40_LCHAN_SRC_PHY_DST_LOG, #define STEDMA40_INFO_CH_MODE_TYPE_POS 6 STEDMA40_LCHAN_SRC_LOG_DST_PHY, #define STEDMA40_CHANNEL_IN_PHY_MODE (0x1 << STEDMA40_INFO_CH_MODE_TYPE_POS) }; #define STEDMA40_CHANNEL_IN_LOG_MODE (0x2 << STEDMA40_INFO_CH_MODE_TYPE_POS) #define STEDMA40_CHANNEL_IN_OPER_MODE (0x3 << STEDMA40_INFO_CH_MODE_TYPE_POS) /* Mode options */ #define STEDMA40_INFO_CH_MODE_OPT_POS 8 #define STEDMA40_PCHAN_BASIC_MODE (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_PCHAN_MODULO_MODE (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_PCHAN_DOUBLE_DST_MODE (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_PHY_DST_LOG (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_LOG_DST_PHS (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_LOG_DST_LOG (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) /* Interrupt */ #define STEDMA40_INFO_TIM_POS 10 #define STEDMA40_NO_TIM_FOR_LINK (0x0 << STEDMA40_INFO_TIM_POS) #define STEDMA40_TIM_FOR_LINK (0x1 << STEDMA40_INFO_TIM_POS) /* End of channel_type configuration */ #define STEDMA40_ESIZE_8_BIT 0x0 #define STEDMA40_ESIZE_8_BIT 0x0 #define STEDMA40_ESIZE_16_BIT 0x1 #define STEDMA40_ESIZE_16_BIT 0x1 Loading Loading @@ -79,11 +62,6 @@ enum stedma40_flow_ctrl { STEDMA40_FLOW_CTRL, STEDMA40_FLOW_CTRL, }; }; enum stedma40_endianess { STEDMA40_LITTLE_ENDIAN, STEDMA40_BIG_ENDIAN }; enum stedma40_periph_data_width { enum stedma40_periph_data_width { STEDMA40_BYTE_WIDTH = STEDMA40_ESIZE_8_BIT, STEDMA40_BYTE_WIDTH = STEDMA40_ESIZE_8_BIT, STEDMA40_HALFWORD_WIDTH = STEDMA40_ESIZE_16_BIT, STEDMA40_HALFWORD_WIDTH = STEDMA40_ESIZE_16_BIT, Loading @@ -102,13 +80,13 @@ enum stedma40_xfer_dir { /** /** * struct stedma40_chan_cfg - dst/src channel configuration * struct stedma40_chan_cfg - dst/src channel configuration * * * @endianess: Endianess of the src/dst hardware * @big_endian: true if the src/dst should be read as big endian * @data_width: Data width of the src/dst hardware * @data_width: Data width of the src/dst hardware * @p_size: Burst size * @p_size: Burst size * @flow_ctrl: Flow control on/off. * @flow_ctrl: Flow control on/off. */ */ struct stedma40_half_channel_info { struct stedma40_half_channel_info { enum stedma40_endianess endianess; bool big_endian; enum stedma40_periph_data_width data_width; enum stedma40_periph_data_width data_width; int psize; int psize; enum stedma40_flow_ctrl flow_ctrl; enum stedma40_flow_ctrl flow_ctrl; Loading @@ -118,7 +96,9 @@ struct stedma40_half_channel_info { * struct stedma40_chan_cfg - Structure to be filled by client drivers. * struct stedma40_chan_cfg - Structure to be filled by client drivers. * * * @dir: MEM 2 MEM, PERIPH 2 MEM , MEM 2 PERIPH, PERIPH 2 PERIPH * @dir: MEM 2 MEM, PERIPH 2 MEM , MEM 2 PERIPH, PERIPH 2 PERIPH * @channel_type: priority, mode, mode options and interrupt configuration. * @high_priority: true if high-priority * @mode: channel mode: physical, logical, or operation * @mode_opt: options for the chosen channel mode * @src_dev_type: Src device type * @src_dev_type: Src device type * @dst_dev_type: Dst device type * @dst_dev_type: Dst device type * @src_info: Parameters for dst half channel * @src_info: Parameters for dst half channel Loading @@ -131,7 +111,9 @@ struct stedma40_half_channel_info { */ */ struct stedma40_chan_cfg { struct stedma40_chan_cfg { enum stedma40_xfer_dir dir; enum stedma40_xfer_dir dir; unsigned int channel_type; bool high_priority; enum stedma40_mode mode; enum stedma40_mode_opt mode_opt; int src_dev_type; int src_dev_type; int dst_dev_type; int dst_dev_type; struct stedma40_half_channel_info src_info; struct stedma40_half_channel_info src_info; Loading drivers/dma/ste_dma40.c +38 −18 Original line number Original line Diff line number Diff line Loading @@ -175,6 +175,7 @@ struct d40_base; * @active: Active descriptor. * @active: Active descriptor. * @queue: Queued jobs. * @queue: Queued jobs. * @dma_cfg: The client configuration of this dma channel. * @dma_cfg: The client configuration of this dma channel. * @configured: whether the dma_cfg configuration is valid * @base: Pointer to the device instance struct. * @base: Pointer to the device instance struct. * @src_def_cfg: Default cfg register setting for src. * @src_def_cfg: Default cfg register setting for src. * @dst_def_cfg: Default cfg register setting for dst. * @dst_def_cfg: Default cfg register setting for dst. Loading @@ -198,6 +199,7 @@ struct d40_chan { struct list_head active; struct list_head active; struct list_head queue; struct list_head queue; struct stedma40_chan_cfg dma_cfg; struct stedma40_chan_cfg dma_cfg; bool configured; struct d40_base *base; struct d40_base *base; /* Default register configurations */ /* Default register configurations */ u32 src_def_cfg; u32 src_def_cfg; Loading Loading @@ -691,6 +693,31 @@ static u32 d40_chan_has_events(struct d40_chan *d40c) return val; return val; } } static u32 d40_get_prmo(struct d40_chan *d40c) { static const unsigned int phy_map[] = { [STEDMA40_PCHAN_BASIC_MODE] = D40_DREG_PRMO_PCHAN_BASIC, [STEDMA40_PCHAN_MODULO_MODE] = D40_DREG_PRMO_PCHAN_MODULO, [STEDMA40_PCHAN_DOUBLE_DST_MODE] = D40_DREG_PRMO_PCHAN_DOUBLE_DST, }; static const unsigned int log_map[] = { [STEDMA40_LCHAN_SRC_PHY_DST_LOG] = D40_DREG_PRMO_LCHAN_SRC_PHY_DST_LOG, [STEDMA40_LCHAN_SRC_LOG_DST_PHY] = D40_DREG_PRMO_LCHAN_SRC_LOG_DST_PHY, [STEDMA40_LCHAN_SRC_LOG_DST_LOG] = D40_DREG_PRMO_LCHAN_SRC_LOG_DST_LOG, }; if (d40c->log_num == D40_PHY_CHAN) return phy_map[d40c->dma_cfg.mode_opt]; else return log_map[d40c->dma_cfg.mode_opt]; } static void d40_config_write(struct d40_chan *d40c) static void d40_config_write(struct d40_chan *d40c) { { u32 addr_base; u32 addr_base; Loading @@ -704,8 +731,7 @@ static void d40_config_write(struct d40_chan *d40c) writel(var, d40c->base->virtbase + D40_DREG_PRMSE + addr_base); writel(var, d40c->base->virtbase + D40_DREG_PRMSE + addr_base); /* Setup operational mode option register */ /* Setup operational mode option register */ var = ((d40c->dma_cfg.channel_type >> STEDMA40_INFO_CH_MODE_OPT_POS) & var = d40_get_prmo(d40c) << D40_CHAN_POS(d40c->phy_chan->num); 0x3) << D40_CHAN_POS(d40c->phy_chan->num); writel(var, d40c->base->virtbase + D40_DREG_PRMOE + addr_base); writel(var, d40c->base->virtbase + D40_DREG_PRMOE + addr_base); Loading Loading @@ -1149,8 +1175,7 @@ static int d40_validate_conf(struct d40_chan *d40c, int res = 0; int res = 0; u32 dst_event_group = D40_TYPE_TO_GROUP(conf->dst_dev_type); u32 dst_event_group = D40_TYPE_TO_GROUP(conf->dst_dev_type); u32 src_event_group = D40_TYPE_TO_GROUP(conf->src_dev_type); u32 src_event_group = D40_TYPE_TO_GROUP(conf->src_dev_type); bool is_log = (conf->channel_type & STEDMA40_CHANNEL_IN_OPER_MODE) bool is_log = conf->mode == STEDMA40_MODE_LOGICAL; == STEDMA40_CHANNEL_IN_LOG_MODE; if (!conf->dir) { if (!conf->dir) { dev_err(&d40c->chan.dev->device, "[%s] Invalid direction.\n", dev_err(&d40c->chan.dev->device, "[%s] Invalid direction.\n", Loading Loading @@ -1314,10 +1339,7 @@ static int d40_allocate_channel(struct d40_chan *d40c) int j; int j; int log_num; int log_num; bool is_src; bool is_src; bool is_log = (d40c->dma_cfg.channel_type & bool is_log = d40c->dma_cfg.mode == STEDMA40_MODE_LOGICAL; STEDMA40_CHANNEL_IN_OPER_MODE) == STEDMA40_CHANNEL_IN_LOG_MODE; phys = d40c->base->phy_res; phys = d40c->base->phy_res; Loading Loading @@ -1518,8 +1540,7 @@ static int d40_free_dma(struct d40_chan *d40c) return res; return res; } } d40c->phy_chan = NULL; d40c->phy_chan = NULL; /* Invalidate channel type */ d40c->configured = false; d40c->dma_cfg.channel_type = 0; d40c->base->lookup_phy_chans[phy->num] = NULL; d40c->base->lookup_phy_chans[phy->num] = NULL; return 0; return 0; Loading Loading @@ -1704,6 +1725,9 @@ bool stedma40_filter(struct dma_chan *chan, void *data) } else } else err = d40_config_memcpy(d40c); err = d40_config_memcpy(d40c); if (!err) d40c->configured = true; return err == 0; return err == 0; } } EXPORT_SYMBOL(stedma40_filter); EXPORT_SYMBOL(stedma40_filter); Loading @@ -1720,12 +1744,8 @@ static int d40_alloc_chan_resources(struct dma_chan *chan) d40c->completed = chan->cookie = 1; d40c->completed = chan->cookie = 1; /* /* If no dma configuration is set use default configuration (memcpy) */ * If no dma configuration is set (channel_type == 0) if (!d40c->configured) { * use default configuration (memcpy) */ if (d40c->dma_cfg.channel_type == 0) { err = d40_config_memcpy(d40c); err = d40_config_memcpy(d40c); if (err) { if (err) { dev_err(&d40c->chan.dev->device, dev_err(&d40c->chan.dev->device, Loading Loading @@ -2231,11 +2251,11 @@ static void d40_set_runtime_config(struct dma_chan *chan, /* Set up all the endpoint configs */ /* Set up all the endpoint configs */ cfg->src_info.data_width = addr_width; cfg->src_info.data_width = addr_width; cfg->src_info.psize = psize; cfg->src_info.psize = psize; cfg->src_info.endianess = STEDMA40_LITTLE_ENDIAN; cfg->src_info.big_endian = false; cfg->src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->dst_info.data_width = addr_width; cfg->dst_info.data_width = addr_width; cfg->dst_info.psize = psize; cfg->dst_info.psize = psize; cfg->dst_info.endianess = STEDMA40_LITTLE_ENDIAN; cfg->dst_info.big_endian = false; cfg->dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; /* Fill in register values */ /* Fill in register values */ Loading drivers/dma/ste_dma40_ll.c +5 −3 Original line number Original line Diff line number Diff line Loading @@ -108,13 +108,15 @@ void d40_phy_cfg(struct stedma40_chan_cfg *cfg, src |= 1 << D40_SREG_CFG_LOG_GIM_POS; src |= 1 << D40_SREG_CFG_LOG_GIM_POS; } } if (cfg->channel_type & STEDMA40_HIGH_PRIORITY_CHANNEL) { if (cfg->high_priority) { src |= 1 << D40_SREG_CFG_PRI_POS; src |= 1 << D40_SREG_CFG_PRI_POS; dst |= 1 << D40_SREG_CFG_PRI_POS; dst |= 1 << D40_SREG_CFG_PRI_POS; } } src |= cfg->src_info.endianess << D40_SREG_CFG_LBE_POS; if (cfg->src_info.big_endian) dst |= cfg->dst_info.endianess << D40_SREG_CFG_LBE_POS; src |= 1 << D40_SREG_CFG_LBE_POS; if (cfg->dst_info.big_endian) dst |= 1 << D40_SREG_CFG_LBE_POS; *src_cfg = src; *src_cfg = src; *dst_cfg = dst; *dst_cfg = dst; Loading drivers/dma/ste_dma40_ll.h +7 −0 Original line number Original line Diff line number Diff line Loading @@ -130,6 +130,13 @@ #define D40_DREG_PRMSO 0x014 #define D40_DREG_PRMSO 0x014 #define D40_DREG_PRMOE 0x018 #define D40_DREG_PRMOE 0x018 #define D40_DREG_PRMOO 0x01C #define D40_DREG_PRMOO 0x01C #define D40_DREG_PRMO_PCHAN_BASIC 0x1 #define D40_DREG_PRMO_PCHAN_MODULO 0x2 #define D40_DREG_PRMO_PCHAN_DOUBLE_DST 0x3 #define D40_DREG_PRMO_LCHAN_SRC_PHY_DST_LOG 0x1 #define D40_DREG_PRMO_LCHAN_SRC_LOG_DST_PHY 0x2 #define D40_DREG_PRMO_LCHAN_SRC_LOG_DST_LOG 0x3 #define D40_DREG_LCPA 0x020 #define D40_DREG_LCPA 0x020 #define D40_DREG_LCLA 0x024 #define D40_DREG_LCLA 0x024 #define D40_DREG_ACTIVE 0x050 #define D40_DREG_ACTIVE 0x050 Loading Loading
arch/arm/mach-ux500/devices-db8500.c +1 −11 Original line number Original line Diff line number Diff line Loading @@ -132,35 +132,25 @@ static struct resource dma40_resources[] = { /* Default configuration for physcial memcpy */ /* Default configuration for physcial memcpy */ struct stedma40_chan_cfg dma40_memcpy_conf_phy = { struct stedma40_chan_cfg dma40_memcpy_conf_phy = { .channel_type = (STEDMA40_CHANNEL_IN_PHY_MODE | .mode = STEDMA40_MODE_PHYSICAL, STEDMA40_LOW_PRIORITY_CHANNEL | STEDMA40_PCHAN_BASIC_MODE), .dir = STEDMA40_MEM_TO_MEM, .dir = STEDMA40_MEM_TO_MEM, .src_info.endianess = STEDMA40_LITTLE_ENDIAN, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.psize = STEDMA40_PSIZE_PHY_1, .src_info.psize = STEDMA40_PSIZE_PHY_1, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.endianess = STEDMA40_LITTLE_ENDIAN, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.psize = STEDMA40_PSIZE_PHY_1, .dst_info.psize = STEDMA40_PSIZE_PHY_1, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, }; }; /* Default configuration for logical memcpy */ /* Default configuration for logical memcpy */ struct stedma40_chan_cfg dma40_memcpy_conf_log = { struct stedma40_chan_cfg dma40_memcpy_conf_log = { .channel_type = (STEDMA40_CHANNEL_IN_LOG_MODE | STEDMA40_LOW_PRIORITY_CHANNEL | STEDMA40_LCHAN_SRC_LOG_DST_LOG | STEDMA40_NO_TIM_FOR_LINK), .dir = STEDMA40_MEM_TO_MEM, .dir = STEDMA40_MEM_TO_MEM, .src_info.endianess = STEDMA40_LITTLE_ENDIAN, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.data_width = STEDMA40_BYTE_WIDTH, .src_info.psize = STEDMA40_PSIZE_LOG_1, .src_info.psize = STEDMA40_PSIZE_LOG_1, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.endianess = STEDMA40_LITTLE_ENDIAN, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.data_width = STEDMA40_BYTE_WIDTH, .dst_info.psize = STEDMA40_PSIZE_LOG_1, .dst_info.psize = STEDMA40_PSIZE_LOG_1, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, .dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL, Loading
arch/arm/plat-nomadik/include/plat/ste_dma40.h +21 −39 Original line number Original line Diff line number Diff line Loading @@ -18,37 +18,20 @@ #define STEDMA40_DEV_DST_MEMORY (-1) #define STEDMA40_DEV_DST_MEMORY (-1) #define STEDMA40_DEV_SRC_MEMORY (-1) #define STEDMA40_DEV_SRC_MEMORY (-1) /* enum stedma40_mode { * Description of bitfields of channel_type variable is available in STEDMA40_MODE_LOGICAL = 0, * the info structure. STEDMA40_MODE_PHYSICAL, */ STEDMA40_MODE_OPERATION, }; /* Priority */ enum stedma40_mode_opt { #define STEDMA40_INFO_PRIO_TYPE_POS 2 STEDMA40_PCHAN_BASIC_MODE = 0, #define STEDMA40_HIGH_PRIORITY_CHANNEL (0x1 << STEDMA40_INFO_PRIO_TYPE_POS) STEDMA40_LCHAN_SRC_LOG_DST_LOG = 0, #define STEDMA40_LOW_PRIORITY_CHANNEL (0x2 << STEDMA40_INFO_PRIO_TYPE_POS) STEDMA40_PCHAN_MODULO_MODE, STEDMA40_PCHAN_DOUBLE_DST_MODE, /* Mode */ STEDMA40_LCHAN_SRC_PHY_DST_LOG, #define STEDMA40_INFO_CH_MODE_TYPE_POS 6 STEDMA40_LCHAN_SRC_LOG_DST_PHY, #define STEDMA40_CHANNEL_IN_PHY_MODE (0x1 << STEDMA40_INFO_CH_MODE_TYPE_POS) }; #define STEDMA40_CHANNEL_IN_LOG_MODE (0x2 << STEDMA40_INFO_CH_MODE_TYPE_POS) #define STEDMA40_CHANNEL_IN_OPER_MODE (0x3 << STEDMA40_INFO_CH_MODE_TYPE_POS) /* Mode options */ #define STEDMA40_INFO_CH_MODE_OPT_POS 8 #define STEDMA40_PCHAN_BASIC_MODE (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_PCHAN_MODULO_MODE (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_PCHAN_DOUBLE_DST_MODE (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_PHY_DST_LOG (0x1 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_LOG_DST_PHS (0x2 << STEDMA40_INFO_CH_MODE_OPT_POS) #define STEDMA40_LCHAN_SRC_LOG_DST_LOG (0x3 << STEDMA40_INFO_CH_MODE_OPT_POS) /* Interrupt */ #define STEDMA40_INFO_TIM_POS 10 #define STEDMA40_NO_TIM_FOR_LINK (0x0 << STEDMA40_INFO_TIM_POS) #define STEDMA40_TIM_FOR_LINK (0x1 << STEDMA40_INFO_TIM_POS) /* End of channel_type configuration */ #define STEDMA40_ESIZE_8_BIT 0x0 #define STEDMA40_ESIZE_8_BIT 0x0 #define STEDMA40_ESIZE_16_BIT 0x1 #define STEDMA40_ESIZE_16_BIT 0x1 Loading Loading @@ -79,11 +62,6 @@ enum stedma40_flow_ctrl { STEDMA40_FLOW_CTRL, STEDMA40_FLOW_CTRL, }; }; enum stedma40_endianess { STEDMA40_LITTLE_ENDIAN, STEDMA40_BIG_ENDIAN }; enum stedma40_periph_data_width { enum stedma40_periph_data_width { STEDMA40_BYTE_WIDTH = STEDMA40_ESIZE_8_BIT, STEDMA40_BYTE_WIDTH = STEDMA40_ESIZE_8_BIT, STEDMA40_HALFWORD_WIDTH = STEDMA40_ESIZE_16_BIT, STEDMA40_HALFWORD_WIDTH = STEDMA40_ESIZE_16_BIT, Loading @@ -102,13 +80,13 @@ enum stedma40_xfer_dir { /** /** * struct stedma40_chan_cfg - dst/src channel configuration * struct stedma40_chan_cfg - dst/src channel configuration * * * @endianess: Endianess of the src/dst hardware * @big_endian: true if the src/dst should be read as big endian * @data_width: Data width of the src/dst hardware * @data_width: Data width of the src/dst hardware * @p_size: Burst size * @p_size: Burst size * @flow_ctrl: Flow control on/off. * @flow_ctrl: Flow control on/off. */ */ struct stedma40_half_channel_info { struct stedma40_half_channel_info { enum stedma40_endianess endianess; bool big_endian; enum stedma40_periph_data_width data_width; enum stedma40_periph_data_width data_width; int psize; int psize; enum stedma40_flow_ctrl flow_ctrl; enum stedma40_flow_ctrl flow_ctrl; Loading @@ -118,7 +96,9 @@ struct stedma40_half_channel_info { * struct stedma40_chan_cfg - Structure to be filled by client drivers. * struct stedma40_chan_cfg - Structure to be filled by client drivers. * * * @dir: MEM 2 MEM, PERIPH 2 MEM , MEM 2 PERIPH, PERIPH 2 PERIPH * @dir: MEM 2 MEM, PERIPH 2 MEM , MEM 2 PERIPH, PERIPH 2 PERIPH * @channel_type: priority, mode, mode options and interrupt configuration. * @high_priority: true if high-priority * @mode: channel mode: physical, logical, or operation * @mode_opt: options for the chosen channel mode * @src_dev_type: Src device type * @src_dev_type: Src device type * @dst_dev_type: Dst device type * @dst_dev_type: Dst device type * @src_info: Parameters for dst half channel * @src_info: Parameters for dst half channel Loading @@ -131,7 +111,9 @@ struct stedma40_half_channel_info { */ */ struct stedma40_chan_cfg { struct stedma40_chan_cfg { enum stedma40_xfer_dir dir; enum stedma40_xfer_dir dir; unsigned int channel_type; bool high_priority; enum stedma40_mode mode; enum stedma40_mode_opt mode_opt; int src_dev_type; int src_dev_type; int dst_dev_type; int dst_dev_type; struct stedma40_half_channel_info src_info; struct stedma40_half_channel_info src_info; Loading
drivers/dma/ste_dma40.c +38 −18 Original line number Original line Diff line number Diff line Loading @@ -175,6 +175,7 @@ struct d40_base; * @active: Active descriptor. * @active: Active descriptor. * @queue: Queued jobs. * @queue: Queued jobs. * @dma_cfg: The client configuration of this dma channel. * @dma_cfg: The client configuration of this dma channel. * @configured: whether the dma_cfg configuration is valid * @base: Pointer to the device instance struct. * @base: Pointer to the device instance struct. * @src_def_cfg: Default cfg register setting for src. * @src_def_cfg: Default cfg register setting for src. * @dst_def_cfg: Default cfg register setting for dst. * @dst_def_cfg: Default cfg register setting for dst. Loading @@ -198,6 +199,7 @@ struct d40_chan { struct list_head active; struct list_head active; struct list_head queue; struct list_head queue; struct stedma40_chan_cfg dma_cfg; struct stedma40_chan_cfg dma_cfg; bool configured; struct d40_base *base; struct d40_base *base; /* Default register configurations */ /* Default register configurations */ u32 src_def_cfg; u32 src_def_cfg; Loading Loading @@ -691,6 +693,31 @@ static u32 d40_chan_has_events(struct d40_chan *d40c) return val; return val; } } static u32 d40_get_prmo(struct d40_chan *d40c) { static const unsigned int phy_map[] = { [STEDMA40_PCHAN_BASIC_MODE] = D40_DREG_PRMO_PCHAN_BASIC, [STEDMA40_PCHAN_MODULO_MODE] = D40_DREG_PRMO_PCHAN_MODULO, [STEDMA40_PCHAN_DOUBLE_DST_MODE] = D40_DREG_PRMO_PCHAN_DOUBLE_DST, }; static const unsigned int log_map[] = { [STEDMA40_LCHAN_SRC_PHY_DST_LOG] = D40_DREG_PRMO_LCHAN_SRC_PHY_DST_LOG, [STEDMA40_LCHAN_SRC_LOG_DST_PHY] = D40_DREG_PRMO_LCHAN_SRC_LOG_DST_PHY, [STEDMA40_LCHAN_SRC_LOG_DST_LOG] = D40_DREG_PRMO_LCHAN_SRC_LOG_DST_LOG, }; if (d40c->log_num == D40_PHY_CHAN) return phy_map[d40c->dma_cfg.mode_opt]; else return log_map[d40c->dma_cfg.mode_opt]; } static void d40_config_write(struct d40_chan *d40c) static void d40_config_write(struct d40_chan *d40c) { { u32 addr_base; u32 addr_base; Loading @@ -704,8 +731,7 @@ static void d40_config_write(struct d40_chan *d40c) writel(var, d40c->base->virtbase + D40_DREG_PRMSE + addr_base); writel(var, d40c->base->virtbase + D40_DREG_PRMSE + addr_base); /* Setup operational mode option register */ /* Setup operational mode option register */ var = ((d40c->dma_cfg.channel_type >> STEDMA40_INFO_CH_MODE_OPT_POS) & var = d40_get_prmo(d40c) << D40_CHAN_POS(d40c->phy_chan->num); 0x3) << D40_CHAN_POS(d40c->phy_chan->num); writel(var, d40c->base->virtbase + D40_DREG_PRMOE + addr_base); writel(var, d40c->base->virtbase + D40_DREG_PRMOE + addr_base); Loading Loading @@ -1149,8 +1175,7 @@ static int d40_validate_conf(struct d40_chan *d40c, int res = 0; int res = 0; u32 dst_event_group = D40_TYPE_TO_GROUP(conf->dst_dev_type); u32 dst_event_group = D40_TYPE_TO_GROUP(conf->dst_dev_type); u32 src_event_group = D40_TYPE_TO_GROUP(conf->src_dev_type); u32 src_event_group = D40_TYPE_TO_GROUP(conf->src_dev_type); bool is_log = (conf->channel_type & STEDMA40_CHANNEL_IN_OPER_MODE) bool is_log = conf->mode == STEDMA40_MODE_LOGICAL; == STEDMA40_CHANNEL_IN_LOG_MODE; if (!conf->dir) { if (!conf->dir) { dev_err(&d40c->chan.dev->device, "[%s] Invalid direction.\n", dev_err(&d40c->chan.dev->device, "[%s] Invalid direction.\n", Loading Loading @@ -1314,10 +1339,7 @@ static int d40_allocate_channel(struct d40_chan *d40c) int j; int j; int log_num; int log_num; bool is_src; bool is_src; bool is_log = (d40c->dma_cfg.channel_type & bool is_log = d40c->dma_cfg.mode == STEDMA40_MODE_LOGICAL; STEDMA40_CHANNEL_IN_OPER_MODE) == STEDMA40_CHANNEL_IN_LOG_MODE; phys = d40c->base->phy_res; phys = d40c->base->phy_res; Loading Loading @@ -1518,8 +1540,7 @@ static int d40_free_dma(struct d40_chan *d40c) return res; return res; } } d40c->phy_chan = NULL; d40c->phy_chan = NULL; /* Invalidate channel type */ d40c->configured = false; d40c->dma_cfg.channel_type = 0; d40c->base->lookup_phy_chans[phy->num] = NULL; d40c->base->lookup_phy_chans[phy->num] = NULL; return 0; return 0; Loading Loading @@ -1704,6 +1725,9 @@ bool stedma40_filter(struct dma_chan *chan, void *data) } else } else err = d40_config_memcpy(d40c); err = d40_config_memcpy(d40c); if (!err) d40c->configured = true; return err == 0; return err == 0; } } EXPORT_SYMBOL(stedma40_filter); EXPORT_SYMBOL(stedma40_filter); Loading @@ -1720,12 +1744,8 @@ static int d40_alloc_chan_resources(struct dma_chan *chan) d40c->completed = chan->cookie = 1; d40c->completed = chan->cookie = 1; /* /* If no dma configuration is set use default configuration (memcpy) */ * If no dma configuration is set (channel_type == 0) if (!d40c->configured) { * use default configuration (memcpy) */ if (d40c->dma_cfg.channel_type == 0) { err = d40_config_memcpy(d40c); err = d40_config_memcpy(d40c); if (err) { if (err) { dev_err(&d40c->chan.dev->device, dev_err(&d40c->chan.dev->device, Loading Loading @@ -2231,11 +2251,11 @@ static void d40_set_runtime_config(struct dma_chan *chan, /* Set up all the endpoint configs */ /* Set up all the endpoint configs */ cfg->src_info.data_width = addr_width; cfg->src_info.data_width = addr_width; cfg->src_info.psize = psize; cfg->src_info.psize = psize; cfg->src_info.endianess = STEDMA40_LITTLE_ENDIAN; cfg->src_info.big_endian = false; cfg->src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->src_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->dst_info.data_width = addr_width; cfg->dst_info.data_width = addr_width; cfg->dst_info.psize = psize; cfg->dst_info.psize = psize; cfg->dst_info.endianess = STEDMA40_LITTLE_ENDIAN; cfg->dst_info.big_endian = false; cfg->dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; cfg->dst_info.flow_ctrl = STEDMA40_NO_FLOW_CTRL; /* Fill in register values */ /* Fill in register values */ Loading
drivers/dma/ste_dma40_ll.c +5 −3 Original line number Original line Diff line number Diff line Loading @@ -108,13 +108,15 @@ void d40_phy_cfg(struct stedma40_chan_cfg *cfg, src |= 1 << D40_SREG_CFG_LOG_GIM_POS; src |= 1 << D40_SREG_CFG_LOG_GIM_POS; } } if (cfg->channel_type & STEDMA40_HIGH_PRIORITY_CHANNEL) { if (cfg->high_priority) { src |= 1 << D40_SREG_CFG_PRI_POS; src |= 1 << D40_SREG_CFG_PRI_POS; dst |= 1 << D40_SREG_CFG_PRI_POS; dst |= 1 << D40_SREG_CFG_PRI_POS; } } src |= cfg->src_info.endianess << D40_SREG_CFG_LBE_POS; if (cfg->src_info.big_endian) dst |= cfg->dst_info.endianess << D40_SREG_CFG_LBE_POS; src |= 1 << D40_SREG_CFG_LBE_POS; if (cfg->dst_info.big_endian) dst |= 1 << D40_SREG_CFG_LBE_POS; *src_cfg = src; *src_cfg = src; *dst_cfg = dst; *dst_cfg = dst; Loading
drivers/dma/ste_dma40_ll.h +7 −0 Original line number Original line Diff line number Diff line Loading @@ -130,6 +130,13 @@ #define D40_DREG_PRMSO 0x014 #define D40_DREG_PRMSO 0x014 #define D40_DREG_PRMOE 0x018 #define D40_DREG_PRMOE 0x018 #define D40_DREG_PRMOO 0x01C #define D40_DREG_PRMOO 0x01C #define D40_DREG_PRMO_PCHAN_BASIC 0x1 #define D40_DREG_PRMO_PCHAN_MODULO 0x2 #define D40_DREG_PRMO_PCHAN_DOUBLE_DST 0x3 #define D40_DREG_PRMO_LCHAN_SRC_PHY_DST_LOG 0x1 #define D40_DREG_PRMO_LCHAN_SRC_LOG_DST_PHY 0x2 #define D40_DREG_PRMO_LCHAN_SRC_LOG_DST_LOG 0x3 #define D40_DREG_LCPA 0x020 #define D40_DREG_LCPA 0x020 #define D40_DREG_LCLA 0x024 #define D40_DREG_LCLA 0x024 #define D40_DREG_ACTIVE 0x050 #define D40_DREG_ACTIVE 0x050 Loading