Commit 92955ea0 authored by Marek Szyprowski's avatar Marek Szyprowski Committed by Mauro Carvalho Chehab

[media] exynos-gsc: Add support for Exynos5433 specific version

This patch adds support for Exynos5433 specific version of the GScaler
module. The main difference between Exynos 5433 and earlier is addition
of new clocks that have to be controlled.
Signed-off-by: default avatarMarek Szyprowski <m.szyprowski@samsung.com>
Reviewed-by: default avatarJavier Martinez Canillas <javier@osg.samsung.com>
Tested-by: default avatarJavier Martinez Canillas <javier@osg.samsung.com>
Acked-by: default avatarKrzysztof Kozlowski <krzk@kernel.org>
Signed-off-by: default avatarSylwester Nawrocki <s.nawrocki@samsung.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@s-opensource.com>
parent 6f99e1b8
...@@ -3,7 +3,8 @@ ...@@ -3,7 +3,8 @@
G-Scaler is used for scaling and color space conversion on EXYNOS5 SoCs. G-Scaler is used for scaling and color space conversion on EXYNOS5 SoCs.
Required properties: Required properties:
- compatible: should be "samsung,exynos5-gsc" - compatible: should be "samsung,exynos5-gsc" (for Exynos 5250, 5420 and
5422 SoCs) or "samsung,exynos5433-gsc" (Exynos 5433)
- reg: should contain G-Scaler physical address location and length. - reg: should contain G-Scaler physical address location and length.
- interrupts: should contain G-Scaler interrupt number - interrupts: should contain G-Scaler interrupt number
......
...@@ -29,8 +29,6 @@ ...@@ -29,8 +29,6 @@
#include "gsc-core.h" #include "gsc-core.h"
#define GSC_CLOCK_GATE_NAME "gscl"
static const struct gsc_fmt gsc_formats[] = { static const struct gsc_fmt gsc_formats[] = {
{ {
.name = "RGB565", .name = "RGB565",
...@@ -978,6 +976,19 @@ static struct gsc_driverdata gsc_v_100_drvdata = { ...@@ -978,6 +976,19 @@ static struct gsc_driverdata gsc_v_100_drvdata = {
[3] = &gsc_v_100_variant, [3] = &gsc_v_100_variant,
}, },
.num_entities = 4, .num_entities = 4,
.clk_names = { "gscl" },
.num_clocks = 1,
};
static struct gsc_driverdata gsc_5433_drvdata = {
.variant = {
[0] = &gsc_v_100_variant,
[1] = &gsc_v_100_variant,
[2] = &gsc_v_100_variant,
},
.num_entities = 3,
.clk_names = { "pclk", "aclk", "aclk_xiu", "aclk_gsclbend" },
.num_clocks = 4,
}; };
static const struct of_device_id exynos_gsc_match[] = { static const struct of_device_id exynos_gsc_match[] = {
...@@ -985,6 +996,10 @@ static const struct of_device_id exynos_gsc_match[] = { ...@@ -985,6 +996,10 @@ static const struct of_device_id exynos_gsc_match[] = {
.compatible = "samsung,exynos5-gsc", .compatible = "samsung,exynos5-gsc",
.data = &gsc_v_100_drvdata, .data = &gsc_v_100_drvdata,
}, },
{
.compatible = "samsung,exynos5433-gsc",
.data = &gsc_5433_drvdata,
},
{}, {},
}; };
MODULE_DEVICE_TABLE(of, exynos_gsc_match); MODULE_DEVICE_TABLE(of, exynos_gsc_match);
...@@ -996,6 +1011,7 @@ static int gsc_probe(struct platform_device *pdev) ...@@ -996,6 +1011,7 @@ static int gsc_probe(struct platform_device *pdev)
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
const struct gsc_driverdata *drv_data = of_device_get_match_data(dev); const struct gsc_driverdata *drv_data = of_device_get_match_data(dev);
int ret; int ret;
int i;
gsc = devm_kzalloc(dev, sizeof(struct gsc_dev), GFP_KERNEL); gsc = devm_kzalloc(dev, sizeof(struct gsc_dev), GFP_KERNEL);
if (!gsc) if (!gsc)
...@@ -1011,6 +1027,7 @@ static int gsc_probe(struct platform_device *pdev) ...@@ -1011,6 +1027,7 @@ static int gsc_probe(struct platform_device *pdev)
return -EINVAL; return -EINVAL;
} }
gsc->num_clocks = drv_data->num_clocks;
gsc->variant = drv_data->variant[gsc->id]; gsc->variant = drv_data->variant[gsc->id];
gsc->pdev = pdev; gsc->pdev = pdev;
...@@ -1029,18 +1046,24 @@ static int gsc_probe(struct platform_device *pdev) ...@@ -1029,18 +1046,24 @@ static int gsc_probe(struct platform_device *pdev)
return -ENXIO; return -ENXIO;
} }
gsc->clock = devm_clk_get(dev, GSC_CLOCK_GATE_NAME); for (i = 0; i < gsc->num_clocks; i++) {
if (IS_ERR(gsc->clock)) { gsc->clock[i] = devm_clk_get(dev, drv_data->clk_names[i]);
dev_err(dev, "failed to get clock~~~: %s\n", if (IS_ERR(gsc->clock[i])) {
GSC_CLOCK_GATE_NAME); dev_err(dev, "failed to get clock: %s\n",
return PTR_ERR(gsc->clock); drv_data->clk_names[i]);
return PTR_ERR(gsc->clock[i]);
}
} }
ret = clk_prepare_enable(gsc->clock); for (i = 0; i < gsc->num_clocks; i++) {
if (ret) { ret = clk_prepare_enable(gsc->clock[i]);
dev_err(&gsc->pdev->dev, "clock prepare failed for clock: %s\n", if (ret) {
GSC_CLOCK_GATE_NAME); dev_err(dev, "clock prepare failed for clock: %s\n",
return ret; drv_data->clk_names[i]);
while (--i >= 0)
clk_disable_unprepare(gsc->clock[i]);
return ret;
}
} }
ret = devm_request_irq(dev, res->start, gsc_irq_handler, ret = devm_request_irq(dev, res->start, gsc_irq_handler,
...@@ -1075,13 +1098,15 @@ static int gsc_probe(struct platform_device *pdev) ...@@ -1075,13 +1098,15 @@ static int gsc_probe(struct platform_device *pdev)
err_v4l2: err_v4l2:
v4l2_device_unregister(&gsc->v4l2_dev); v4l2_device_unregister(&gsc->v4l2_dev);
err_clk: err_clk:
clk_disable_unprepare(gsc->clock); for (i = gsc->num_clocks - 1; i >= 0; i--)
clk_disable_unprepare(gsc->clock[i]);
return ret; return ret;
} }
static int gsc_remove(struct platform_device *pdev) static int gsc_remove(struct platform_device *pdev)
{ {
struct gsc_dev *gsc = platform_get_drvdata(pdev); struct gsc_dev *gsc = platform_get_drvdata(pdev);
int i;
pm_runtime_get_sync(&pdev->dev); pm_runtime_get_sync(&pdev->dev);
...@@ -1089,7 +1114,8 @@ static int gsc_remove(struct platform_device *pdev) ...@@ -1089,7 +1114,8 @@ static int gsc_remove(struct platform_device *pdev)
v4l2_device_unregister(&gsc->v4l2_dev); v4l2_device_unregister(&gsc->v4l2_dev);
vb2_dma_contig_clear_max_seg_size(&pdev->dev); vb2_dma_contig_clear_max_seg_size(&pdev->dev);
clk_disable_unprepare(gsc->clock); for (i = 0; i < gsc->num_clocks; i++)
clk_disable_unprepare(gsc->clock[i]);
pm_runtime_put_noidle(&pdev->dev); pm_runtime_put_noidle(&pdev->dev);
...@@ -1139,12 +1165,18 @@ static int gsc_runtime_resume(struct device *dev) ...@@ -1139,12 +1165,18 @@ static int gsc_runtime_resume(struct device *dev)
{ {
struct gsc_dev *gsc = dev_get_drvdata(dev); struct gsc_dev *gsc = dev_get_drvdata(dev);
int ret = 0; int ret = 0;
int i;
pr_debug("gsc%d: state: 0x%lx\n", gsc->id, gsc->state); pr_debug("gsc%d: state: 0x%lx\n", gsc->id, gsc->state);
ret = clk_prepare_enable(gsc->clock); for (i = 0; i < gsc->num_clocks; i++) {
if (ret) ret = clk_prepare_enable(gsc->clock[i]);
return ret; if (ret) {
while (--i >= 0)
clk_disable_unprepare(gsc->clock[i]);
return ret;
}
}
gsc_hw_set_sw_reset(gsc); gsc_hw_set_sw_reset(gsc);
gsc_wait_reset(gsc); gsc_wait_reset(gsc);
...@@ -1157,10 +1189,14 @@ static int gsc_runtime_suspend(struct device *dev) ...@@ -1157,10 +1189,14 @@ static int gsc_runtime_suspend(struct device *dev)
{ {
struct gsc_dev *gsc = dev_get_drvdata(dev); struct gsc_dev *gsc = dev_get_drvdata(dev);
int ret = 0; int ret = 0;
int i;
ret = gsc_m2m_suspend(gsc); ret = gsc_m2m_suspend(gsc);
if (!ret) if (ret)
clk_disable_unprepare(gsc->clock); return ret;
for (i = gsc->num_clocks - 1; i >= 0; i--)
clk_disable_unprepare(gsc->clock[i]);
pr_debug("gsc%d: state: 0x%lx\n", gsc->id, gsc->state); pr_debug("gsc%d: state: 0x%lx\n", gsc->id, gsc->state);
return ret; return ret;
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#define GSC_SHUTDOWN_TIMEOUT ((100*HZ)/1000) #define GSC_SHUTDOWN_TIMEOUT ((100*HZ)/1000)
#define GSC_MAX_DEVS 4 #define GSC_MAX_DEVS 4
#define GSC_MAX_CLOCKS 4
#define GSC_M2M_BUF_NUM 0 #define GSC_M2M_BUF_NUM 0
#define GSC_MAX_CTRL_NUM 10 #define GSC_MAX_CTRL_NUM 10
#define GSC_SC_ALIGN_4 4 #define GSC_SC_ALIGN_4 4
...@@ -307,6 +308,8 @@ struct gsc_variant { ...@@ -307,6 +308,8 @@ struct gsc_variant {
*/ */
struct gsc_driverdata { struct gsc_driverdata {
struct gsc_variant *variant[GSC_MAX_DEVS]; struct gsc_variant *variant[GSC_MAX_DEVS];
const char *clk_names[GSC_MAX_CLOCKS];
int num_clocks;
int num_entities; int num_entities;
}; };
...@@ -330,7 +333,8 @@ struct gsc_dev { ...@@ -330,7 +333,8 @@ struct gsc_dev {
struct platform_device *pdev; struct platform_device *pdev;
struct gsc_variant *variant; struct gsc_variant *variant;
u16 id; u16 id;
struct clk *clock; int num_clocks;
struct clk *clock[GSC_MAX_CLOCKS];
void __iomem *regs; void __iomem *regs;
wait_queue_head_t irq_queue; wait_queue_head_t irq_queue;
struct gsc_m2m_device m2m; struct gsc_m2m_device m2m;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment