media: mfc: DRV3.2: parse the additional info in DT
authorAyoung Sim <a.sim@samsung.com>
Fri, 30 Mar 2018 07:56:03 +0000 (16:56 +0900)
committerSunyoung Kang <sy0816.kang@samsung.com>
Tue, 29 May 2018 06:59:18 +0000 (15:59 +0900)
Informations that changes per every chip should be used
from device tree

Change-Id: Ib2d776372e954f7e3fe726505ec47501a9a69f80
Signed-off-by: Ayoung Sim <a.sim@samsung.com>
drivers/media/platform/exynos/mfc/s5p_mfc.c
drivers/media/platform/exynos/mfc/s5p_mfc_data_struct.h
drivers/media/platform/exynos/mfc/s5p_mfc_otf.c
drivers/media/platform/exynos/mfc/s5p_mfc_regs_v10.h

index 85233a71e1c613ef43bc5502bcb2cf0e0ce915a0..942e4f126bd165f7c733c1de0346a731b545aa05 100644 (file)
@@ -12,6 +12,7 @@
 
 #include <linux/module.h>
 #include <linux/platform_device.h>
+#include <linux/of_address.h>
 #include <linux/proc_fs.h>
 #include <video/videonode.h>
 #include <linux/of.h>
@@ -879,11 +880,13 @@ int s5p_mfc_sysmmu_fault_handler(struct iommu_domain *iodmn, struct device *devi
        dev = (struct s5p_mfc_dev *)param;
 
        /* OTF: If AxID is 1 in SYSMMU1 fault info, it is TS-MUX fault */
-       if (MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS) &&
+       if (dev->has_hwfc && dev->has_2sysmmu) {
+               if (MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS) &&
                                ((MFC_MMU1_READL(MFC_MMU_FAULT_TRANS_INFO) &
-                                MFC_MMU_FAULT_TRANS_INFO_AXID_MASK) == 1)) {
+                                 MFC_MMU_FAULT_TRANS_INFO_AXID_MASK) == 1)) {
                        mfc_err_dev("There is TS-MUX page fault. skip SFR dump.\n");
                        return 0;
+               }
        }
 
        if (MFC_MMU0_READL(MFC_MMU_INTERRUPT_STATUS)) {
@@ -893,15 +896,17 @@ int s5p_mfc_sysmmu_fault_handler(struct iommu_domain *iodmn, struct device *devi
                        dev->logging_data->cause |= (1 << MFC_CAUSE_0READ_PAGE_FAULT);
                dev->logging_data->fault_status = MFC_MMU0_READL(MFC_MMU_INTERRUPT_STATUS);
                dev->logging_data->fault_trans_info = MFC_MMU0_READL(MFC_MMU_FAULT_TRANS_INFO);
-       } else if (MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS)) {
-               if (MFC_MMU1_READL(MFC_MMU_FAULT_TRANS_INFO) & MFC_MMU_FAULT_TRANS_INFO_RW_MASK)
-                       dev->logging_data->cause |= (1 << MFC_CAUSE_1WRITE_PAGE_FAULT);
-               else
-                       dev->logging_data->cause |= (1 << MFC_CAUSE_1READ_PAGE_FAULT);
-               dev->logging_data->fault_status = MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS);
-               dev->logging_data->fault_trans_info = MFC_MMU1_READL(MFC_MMU_FAULT_TRANS_INFO);
-       } else {
-               mfc_err_dev("there isn't any fault interrupt of MFC\n");
+       }
+
+       if (dev->has_2sysmmu) {
+               if (MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS)) {
+                       if (MFC_MMU1_READL(MFC_MMU_FAULT_TRANS_INFO) & MFC_MMU_FAULT_TRANS_INFO_RW_MASK)
+                               dev->logging_data->cause |= (1 << MFC_CAUSE_1WRITE_PAGE_FAULT);
+                       else
+                               dev->logging_data->cause |= (1 << MFC_CAUSE_1READ_PAGE_FAULT);
+                       dev->logging_data->fault_status = MFC_MMU1_READL(MFC_MMU_INTERRUPT_STATUS);
+                       dev->logging_data->fault_trans_info = MFC_MMU1_READL(MFC_MMU_FAULT_TRANS_INFO);
+               }
        }
        dev->logging_data->fault_addr = (unsigned int)addr;
 
@@ -984,6 +989,9 @@ static struct video_device *mfc_video_device_register(struct s5p_mfc_dev *dev,
 
 static int mfc_register_resource(struct platform_device *pdev, struct s5p_mfc_dev *dev)
 {
+       struct device_node *np = dev->device->of_node;
+       struct device_node *iommu;
+       struct device_node *hwfc;
        struct resource *res;
        int ret;
 
@@ -994,32 +1002,46 @@ static int mfc_register_resource(struct platform_device *pdev, struct s5p_mfc_de
                dev_err(&pdev->dev, "failed to get memory region resource\n");
                return -ENOENT;
        }
-       dev->mfc_mem = request_mem_region(res->start, resource_size(res),
-                                       pdev->name);
+       dev->mfc_mem = request_mem_region(res->start, resource_size(res), pdev->name);
        if (dev->mfc_mem == NULL) {
                dev_err(&pdev->dev, "failed to get memory region\n");
                return -ENOENT;
        }
-       dev->regs_base = ioremap(dev->mfc_mem->start,
-                               resource_size(dev->mfc_mem));
+       dev->regs_base = ioremap(dev->mfc_mem->start, resource_size(dev->mfc_mem));
        if (dev->regs_base == NULL) {
                dev_err(&pdev->dev, "failed to ioremap address region\n");
                goto err_ioremap;
        }
-       dev->sysmmu0_base = ioremap(MFC_MMU0_BASE_ADDR, MFC_MMU_SIZE);
+
+       iommu = of_get_child_by_name(np, "iommu");
+       if (!iommu) {
+               dev_err(&pdev->dev, "failed to get iommu node\n");
+               goto err_ioremap_mmu0;
+       }
+
+       dev->sysmmu0_base = of_iomap(iommu, 0);
        if (dev->sysmmu0_base == NULL) {
                dev_err(&pdev->dev, "failed to ioremap sysmmu0 address region\n");
                goto err_ioremap_mmu0;
        }
-       dev->sysmmu1_base = ioremap(MFC_MMU1_BASE_ADDR, MFC_MMU_SIZE);
+
+       dev->sysmmu1_base = of_iomap(iommu, 1);
        if (dev->sysmmu1_base == NULL) {
-               dev_err(&pdev->dev, "failed to ioremap sysmmu1 address region\n");
-               goto err_ioremap_mmu1;
+               pr_debug("there is only one MFC sysmmu\n");
+       } else {
+               dev->has_2sysmmu = 1;
        }
-       dev->hwfc_base = ioremap(HWFC_BASE_ADDR, HWFC_SIZE);
-       if (dev->hwfc_base == NULL) {
-               dev_err(&pdev->dev, "failed to ioremap hwfc adddress region\n");
-               goto err_ioremap_hwfc;
+
+       hwfc = of_get_child_by_name(np, "hwfc");
+       if (hwfc) {
+               dev->hwfc_base = of_iomap(hwfc, 0);
+               if (dev->hwfc_base == NULL) {
+                       dev->has_hwfc = 0;
+                       dev_err(&pdev->dev, "failed to iomap hwfc address region\n");
+                       goto err_res_hwfc;
+               } else {
+                       dev->has_hwfc = 1;
+               }
        }
 
        res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
@@ -1032,17 +1054,17 @@ static int mfc_register_resource(struct platform_device *pdev, struct s5p_mfc_de
                                IRQF_ONESHOT, pdev->name, dev);
        if (ret != 0) {
                dev_err(&pdev->dev, "failed to install irq (%d)\n", ret);
-               goto err_req_irq;
+               goto err_res_irq;
        }
 
        return 0;
 
-err_req_irq:
 err_res_irq:
-       iounmap(dev->hwfc_base);
-err_ioremap_hwfc:
-       iounmap(dev->sysmmu1_base);
-err_ioremap_mmu1:
+       if (dev->has_hwfc)
+               iounmap(dev->hwfc_base);
+err_res_hwfc:
+       if (dev->has_2sysmmu)
+               iounmap(dev->sysmmu1_base);
        iounmap(dev->sysmmu0_base);
 err_ioremap_mmu0:
        iounmap(dev->regs_base);
@@ -1256,8 +1278,10 @@ alloc_vdev_dec:
 err_v4l2_dev:
        mutex_destroy(&dev->mfc_mutex);
        free_irq(dev->irq, dev);
-       iounmap(dev->hwfc_base);
-       iounmap(dev->sysmmu1_base);
+       if (dev->has_hwfc)
+               iounmap(dev->hwfc_base);
+       if (dev->has_2sysmmu)
+               iounmap(dev->sysmmu1_base);
        iounmap(dev->sysmmu0_base);
        iounmap(dev->regs_base);
        release_mem_region(dev->mfc_mem->start, resource_size(dev->mfc_mem));
@@ -1295,10 +1319,12 @@ static int s5p_mfc_remove(struct platform_device *pdev)
        mfc_debug(2, "Will now deinit HW\n");
        s5p_mfc_deinit_hw(dev);
        free_irq(dev->irq, dev);
-       iounmap(dev->sysmmu1_base);
+       if (dev->has_hwfc)
+               iounmap(dev->hwfc_base);
+       if (dev->has_2sysmmu)
+               iounmap(dev->sysmmu1_base);
        iounmap(dev->sysmmu0_base);
        iounmap(dev->regs_base);
-       iounmap(dev->hwfc_base);
        release_mem_region(dev->mfc_mem->start, resource_size(dev->mfc_mem));
        s5p_mfc_pm_final(dev);
        kfree(dev);
index d3fb313f08a89cf9b2beb9509e80592fe6619278..dcc1ddd21931ead601e6fc31842e5319a77bfad7 100644 (file)
@@ -710,10 +710,8 @@ struct s5p_mfc_dev {
        wait_queue_head_t cmd_wq;
        struct s5p_mfc_listable_wq hwlock_wq;
 
-       /*
-       struct clk *clock1;
-       struct clk *clock2;
-       */
+       bool has_hwfc;
+       bool has_2sysmmu;
 
        struct s5p_mfc_special_buf common_ctx_buf;
        struct s5p_mfc_special_buf drm_common_ctx_buf;
index a54252b656b7f0ffd545decfb9cb7f7547282048..dfdfa876596b9134a6689ea5df1604492b6cce1e 100644 (file)
@@ -380,6 +380,7 @@ int s5p_mfc_otf_run_enc_init(struct s5p_mfc_ctx *ctx)
 
 int s5p_mfc_otf_run_enc_frame(struct s5p_mfc_ctx *ctx)
 {
+       struct s5p_mfc_dev *dev = ctx->dev;
        struct _otf_handle *handle = ctx->otf_handle;
        struct s5p_mfc_raw_info *raw;
 
@@ -398,6 +399,11 @@ int s5p_mfc_otf_run_enc_frame(struct s5p_mfc_ctx *ctx)
                return -EINVAL;
        }
 
+       if (!dev->has_hwfc) {
+               mfc_err_ctx("OTF: HWFC register didn't mapped\n");
+               return -EINVAL;
+       }
+
        s5p_mfc_otf_set_frame_addr(ctx, raw->num_planes);
        s5p_mfc_otf_set_stream_size(ctx, raw->total_plane_size);
        s5p_mfc_otf_set_hwfc_index(ctx, handle->otf_job_id);
index af8ce4faab72ae710a555af617511c1893ca21b1..5b48ad58016cdd2c3e833a935221184af2a43d2b 100644 (file)
 #ifndef __REGS_MFC_V10_H
 #define __REGS_MFC_V10_H __FILE__
 
-/* SYSMMU Register */
-#define MFC_MMU0_BASE_ADDR                                     0x18890000
-#define MFC_MMU1_BASE_ADDR                                     0x188B0000
-#define MFC_MMU_SIZE                                           0x9000
-
 #define MFC_MMU_INTERRUPT_STATUS                               0x0060
 #define MFC_MMU_FAULT_TRANS_INFO                               0x0078
 #define MFC_MMU_FAULT_TRANS_INFO_RW_MASK                       0x100000
 #define MFC_MMU_FAULT_TRANS_INFO_AXID_MASK                     0xFFFF
 
-#define HWFC_BASE_ADDR                                         0x18A28000
-#define HWFC_SIZE                                              0x100
-
 #define HWFC_ENCODING_IDX                                      0x4
 
 /* Codec Common Registers */