123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436 |
- /*
- * Copyright (c) 2010 Sascha Hauer <s.hauer@pengutronix.de>
- * Copyright (C) 2005-2009 Freescale Semiconductor, Inc.
- *
- * This program is free software; you can redistribute it and/or modify it
- * under the terms of the GNU General Public License as published by the
- * Free Software Foundation; either version 2 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 GNU General Public License
- * for more details.
- */
- #include <linux/export.h>
- #include <linux/types.h>
- #include <linux/errno.h>
- #include <linux/io.h>
- #include <video/imx-ipu-v3.h>
- #include "ipu-prv.h"
- #define DMFC_RD_CHAN 0x0000
- #define DMFC_WR_CHAN 0x0004
- #define DMFC_WR_CHAN_DEF 0x0008
- #define DMFC_DP_CHAN 0x000c
- #define DMFC_DP_CHAN_DEF 0x0010
- #define DMFC_GENERAL1 0x0014
- #define DMFC_GENERAL2 0x0018
- #define DMFC_IC_CTRL 0x001c
- #define DMFC_WR_CHAN_ALT 0x0020
- #define DMFC_WR_CHAN_DEF_ALT 0x0024
- #define DMFC_DP_CHAN_ALT 0x0028
- #define DMFC_DP_CHAN_DEF_ALT 0x002c
- #define DMFC_GENERAL1_ALT 0x0030
- #define DMFC_STAT 0x0034
- #define DMFC_WR_CHAN_1_28 0
- #define DMFC_WR_CHAN_2_41 8
- #define DMFC_WR_CHAN_1C_42 16
- #define DMFC_WR_CHAN_2C_43 24
- #define DMFC_DP_CHAN_5B_23 0
- #define DMFC_DP_CHAN_5F_27 8
- #define DMFC_DP_CHAN_6B_24 16
- #define DMFC_DP_CHAN_6F_29 24
- #define DMFC_FIFO_SIZE_64 (3 << 3)
- #define DMFC_FIFO_SIZE_128 (2 << 3)
- #define DMFC_FIFO_SIZE_256 (1 << 3)
- #define DMFC_FIFO_SIZE_512 (0 << 3)
- #define DMFC_SEGMENT(x) ((x & 0x7) << 0)
- #define DMFC_BURSTSIZE_128 (0 << 6)
- #define DMFC_BURSTSIZE_64 (1 << 6)
- #define DMFC_BURSTSIZE_32 (2 << 6)
- #define DMFC_BURSTSIZE_16 (3 << 6)
- struct dmfc_channel_data {
- int ipu_channel;
- unsigned long channel_reg;
- unsigned long shift;
- unsigned eot_shift;
- unsigned max_fifo_lines;
- };
- static const struct dmfc_channel_data dmfcdata[] = {
- {
- .ipu_channel = IPUV3_CHANNEL_MEM_BG_SYNC,
- .channel_reg = DMFC_DP_CHAN,
- .shift = DMFC_DP_CHAN_5B_23,
- .eot_shift = 20,
- .max_fifo_lines = 3,
- }, {
- .ipu_channel = 24,
- .channel_reg = DMFC_DP_CHAN,
- .shift = DMFC_DP_CHAN_6B_24,
- .eot_shift = 22,
- .max_fifo_lines = 1,
- }, {
- .ipu_channel = IPUV3_CHANNEL_MEM_FG_SYNC,
- .channel_reg = DMFC_DP_CHAN,
- .shift = DMFC_DP_CHAN_5F_27,
- .eot_shift = 21,
- .max_fifo_lines = 2,
- }, {
- .ipu_channel = IPUV3_CHANNEL_MEM_DC_SYNC,
- .channel_reg = DMFC_WR_CHAN,
- .shift = DMFC_WR_CHAN_1_28,
- .eot_shift = 16,
- .max_fifo_lines = 2,
- }, {
- .ipu_channel = 29,
- .channel_reg = DMFC_DP_CHAN,
- .shift = DMFC_DP_CHAN_6F_29,
- .eot_shift = 23,
- .max_fifo_lines = 1,
- },
- };
- #define DMFC_NUM_CHANNELS ARRAY_SIZE(dmfcdata)
- struct ipu_dmfc_priv;
- struct dmfc_channel {
- unsigned slots;
- unsigned slotmask;
- unsigned segment;
- int burstsize;
- struct ipu_soc *ipu;
- struct ipu_dmfc_priv *priv;
- const struct dmfc_channel_data *data;
- };
- struct ipu_dmfc_priv {
- struct ipu_soc *ipu;
- struct device *dev;
- struct dmfc_channel channels[DMFC_NUM_CHANNELS];
- struct mutex mutex;
- unsigned long bandwidth_per_slot;
- void __iomem *base;
- int use_count;
- };
- int ipu_dmfc_enable_channel(struct dmfc_channel *dmfc)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- mutex_lock(&priv->mutex);
- if (!priv->use_count)
- ipu_module_enable(priv->ipu, IPU_CONF_DMFC_EN);
- priv->use_count++;
- mutex_unlock(&priv->mutex);
- return 0;
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_enable_channel);
- static void ipu_dmfc_wait_fifos(struct ipu_dmfc_priv *priv)
- {
- unsigned long timeout = jiffies + msecs_to_jiffies(1000);
- while ((readl(priv->base + DMFC_STAT) & 0x02fff000) != 0x02fff000) {
- if (time_after(jiffies, timeout)) {
- dev_warn(priv->dev,
- "Timeout waiting for DMFC FIFOs to clear\n");
- break;
- }
- cpu_relax();
- }
- }
- void ipu_dmfc_disable_channel(struct dmfc_channel *dmfc)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- mutex_lock(&priv->mutex);
- priv->use_count--;
- if (!priv->use_count) {
- ipu_dmfc_wait_fifos(priv);
- ipu_module_disable(priv->ipu, IPU_CONF_DMFC_EN);
- }
- if (priv->use_count < 0)
- priv->use_count = 0;
- mutex_unlock(&priv->mutex);
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_disable_channel);
- static int ipu_dmfc_setup_channel(struct dmfc_channel *dmfc, int slots,
- int segment, int burstsize)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- u32 val, field;
- dev_dbg(priv->dev,
- "dmfc: using %d slots starting from segment %d for IPU channel %d\n",
- slots, segment, dmfc->data->ipu_channel);
- switch (slots) {
- case 1:
- field = DMFC_FIFO_SIZE_64;
- break;
- case 2:
- field = DMFC_FIFO_SIZE_128;
- break;
- case 4:
- field = DMFC_FIFO_SIZE_256;
- break;
- case 8:
- field = DMFC_FIFO_SIZE_512;
- break;
- default:
- return -EINVAL;
- }
- switch (burstsize) {
- case 16:
- field |= DMFC_BURSTSIZE_16;
- break;
- case 32:
- field |= DMFC_BURSTSIZE_32;
- break;
- case 64:
- field |= DMFC_BURSTSIZE_64;
- break;
- case 128:
- field |= DMFC_BURSTSIZE_128;
- break;
- }
- field |= DMFC_SEGMENT(segment);
- val = readl(priv->base + dmfc->data->channel_reg);
- val &= ~(0xff << dmfc->data->shift);
- val |= field << dmfc->data->shift;
- writel(val, priv->base + dmfc->data->channel_reg);
- dmfc->slots = slots;
- dmfc->segment = segment;
- dmfc->burstsize = burstsize;
- dmfc->slotmask = ((1 << slots) - 1) << segment;
- return 0;
- }
- static int dmfc_bandwidth_to_slots(struct ipu_dmfc_priv *priv,
- unsigned long bandwidth)
- {
- int slots = 1;
- while (slots * priv->bandwidth_per_slot < bandwidth)
- slots *= 2;
- return slots;
- }
- static int dmfc_find_slots(struct ipu_dmfc_priv *priv, int slots)
- {
- unsigned slotmask_need, slotmask_used = 0;
- int i, segment = 0;
- slotmask_need = (1 << slots) - 1;
- for (i = 0; i < DMFC_NUM_CHANNELS; i++)
- slotmask_used |= priv->channels[i].slotmask;
- while (slotmask_need <= 0xff) {
- if (!(slotmask_used & slotmask_need))
- return segment;
- slotmask_need <<= 1;
- segment++;
- }
- return -EBUSY;
- }
- void ipu_dmfc_free_bandwidth(struct dmfc_channel *dmfc)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- int i;
- dev_dbg(priv->dev, "dmfc: freeing %d slots starting from segment %d\n",
- dmfc->slots, dmfc->segment);
- mutex_lock(&priv->mutex);
- if (!dmfc->slots)
- goto out;
- dmfc->slotmask = 0;
- dmfc->slots = 0;
- dmfc->segment = 0;
- for (i = 0; i < DMFC_NUM_CHANNELS; i++)
- priv->channels[i].slotmask = 0;
- for (i = 0; i < DMFC_NUM_CHANNELS; i++) {
- if (priv->channels[i].slots > 0) {
- priv->channels[i].segment =
- dmfc_find_slots(priv, priv->channels[i].slots);
- priv->channels[i].slotmask =
- ((1 << priv->channels[i].slots) - 1) <<
- priv->channels[i].segment;
- }
- }
- for (i = 0; i < DMFC_NUM_CHANNELS; i++) {
- if (priv->channels[i].slots > 0)
- ipu_dmfc_setup_channel(&priv->channels[i],
- priv->channels[i].slots,
- priv->channels[i].segment,
- priv->channels[i].burstsize);
- }
- out:
- mutex_unlock(&priv->mutex);
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_free_bandwidth);
- int ipu_dmfc_alloc_bandwidth(struct dmfc_channel *dmfc,
- unsigned long bandwidth_pixel_per_second, int burstsize)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- int slots = dmfc_bandwidth_to_slots(priv, bandwidth_pixel_per_second);
- int segment = -1, ret = 0;
- dev_dbg(priv->dev, "dmfc: trying to allocate %ldMpixel/s for IPU channel %d\n",
- bandwidth_pixel_per_second / 1000000,
- dmfc->data->ipu_channel);
- ipu_dmfc_free_bandwidth(dmfc);
- mutex_lock(&priv->mutex);
- if (slots > 8) {
- ret = -EBUSY;
- goto out;
- }
- /* For the MEM_BG channel, first try to allocate twice the slots */
- if (dmfc->data->ipu_channel == IPUV3_CHANNEL_MEM_BG_SYNC)
- segment = dmfc_find_slots(priv, slots * 2);
- else if (slots < 2)
- /* Always allocate at least 128*4 bytes (2 slots) */
- slots = 2;
- if (segment >= 0)
- slots *= 2;
- else
- segment = dmfc_find_slots(priv, slots);
- if (segment < 0) {
- ret = -EBUSY;
- goto out;
- }
- ipu_dmfc_setup_channel(dmfc, slots, segment, burstsize);
- out:
- mutex_unlock(&priv->mutex);
- return ret;
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_alloc_bandwidth);
- int ipu_dmfc_init_channel(struct dmfc_channel *dmfc, int width)
- {
- struct ipu_dmfc_priv *priv = dmfc->priv;
- u32 dmfc_gen1;
- dmfc_gen1 = readl(priv->base + DMFC_GENERAL1);
- if ((dmfc->slots * 64 * 4) / width > dmfc->data->max_fifo_lines)
- dmfc_gen1 |= 1 << dmfc->data->eot_shift;
- else
- dmfc_gen1 &= ~(1 << dmfc->data->eot_shift);
- writel(dmfc_gen1, priv->base + DMFC_GENERAL1);
- return 0;
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_init_channel);
- struct dmfc_channel *ipu_dmfc_get(struct ipu_soc *ipu, int ipu_channel)
- {
- struct ipu_dmfc_priv *priv = ipu->dmfc_priv;
- int i;
- for (i = 0; i < DMFC_NUM_CHANNELS; i++)
- if (dmfcdata[i].ipu_channel == ipu_channel)
- return &priv->channels[i];
- return ERR_PTR(-ENODEV);
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_get);
- void ipu_dmfc_put(struct dmfc_channel *dmfc)
- {
- ipu_dmfc_free_bandwidth(dmfc);
- }
- EXPORT_SYMBOL_GPL(ipu_dmfc_put);
- int ipu_dmfc_init(struct ipu_soc *ipu, struct device *dev, unsigned long base,
- struct clk *ipu_clk)
- {
- struct ipu_dmfc_priv *priv;
- int i;
- priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
- if (!priv)
- return -ENOMEM;
- priv->base = devm_ioremap(dev, base, PAGE_SIZE);
- if (!priv->base)
- return -ENOMEM;
- priv->dev = dev;
- priv->ipu = ipu;
- mutex_init(&priv->mutex);
- ipu->dmfc_priv = priv;
- for (i = 0; i < DMFC_NUM_CHANNELS; i++) {
- priv->channels[i].priv = priv;
- priv->channels[i].ipu = ipu;
- priv->channels[i].data = &dmfcdata[i];
- }
- writel(0x0, priv->base + DMFC_WR_CHAN);
- writel(0x0, priv->base + DMFC_DP_CHAN);
- /*
- * We have a total bandwidth of clkrate * 4pixel divided
- * into 8 slots.
- */
- priv->bandwidth_per_slot = clk_get_rate(ipu_clk) * 4 / 8;
- dev_dbg(dev, "dmfc: 8 slots with %ldMpixel/s bandwidth each\n",
- priv->bandwidth_per_slot / 1000000);
- writel(0x202020f6, priv->base + DMFC_WR_CHAN_DEF);
- writel(0x2020f6f6, priv->base + DMFC_DP_CHAN_DEF);
- writel(0x00000003, priv->base + DMFC_GENERAL1);
- return 0;
- }
- void ipu_dmfc_exit(struct ipu_soc *ipu)
- {
- }
|