Hi All,

The following PRELIMINARY patch adds DT support to the OMAP4 ISS. It
also fixes some problems a have found along the way. It is tightly
modelled after the omap3-isp media platform driver. This patch is a
work in progress as I would like feedback. It contains debugging
messages that need to be removed, as well as disgusting abuses of the
C language as required (i.e. clk_core_fake and clk_fake).

I'm working in the latest stable mainline which as far as this patch
is concerned is compatible with media tree master. I have had this
omap4-iss working on my hardware in the 3.17 kernel, however I'm
currently having the following issue in 4.1.4 stable when I start to
stream:

[  141.612609] omap4iss 52000000.iss: CSI2: CSI2_96M_FCLK reset timeout!

Any feedback regarding this issue would really be appreciated. After
resolving this issue, we still need to do a proper implementation
using syscon and also to find a solution regarding where to put the
iss_set_constraints function. I have to give up for the next couple of
weeks as I need to submit a conference paper, which I'm going to use
my 3.17 implementation for. The following is an example of how the ISS
would be instantiated in the top level device tree:

iss_csi21_pins: pinmux_iss_csi21_pins {
    pinctrl-single,pins = <
        OMAP4_IOPAD(0x0a0, PIN_INPUT | MUX_MODE0)        /*
csi21_dx0.csi21_dx0 */
        OMAP4_IOPAD(0x0a2, PIN_INPUT | MUX_MODE0)        /*
csi21_dy0.csi21_dy0 */
        OMAP4_IOPAD(0x0a4, PIN_INPUT | MUX_MODE0)        /*
csi21_dx1.csi21_dx1 */
        OMAP4_IOPAD(0x0a6, PIN_INPUT | MUX_MODE0)        /*
csi21_dy1.csi21_dy1 */
        OMAP4_IOPAD(0x0a8, PIN_INPUT | MUX_MODE0)        /*
csi21_dx2.csi21_dx2 */
        OMAP4_IOPAD(0x0aa, PIN_INPUT | MUX_MODE0)        /*
csi21_dy2.csi21_dy2 */
    >;
};

&iss {
    status = "ok";

    pinctrl-names = "default";
    pinctrl-0 = <&iss_csi21_pins>;

    ports {
        port@0 {
            reg = <0>;
            csi2a_ep: endpoint {
                remote-endpoint = <&ov5640_1_cam_ep>;
                clock-lanes = <1>;
                data-lanes = <2>;
                crc = <0>;
                lane-polarities = <0 0>;
            };
        };
    };
};

and for the connected camera:

ov5640_1_camera: ov5640@3c {
    compatible = "omnivision,ov5640";
    status = "ok";
    reg = <0x3c>;

    pwdn-gpios = <&ov5640_1_gpio 5 GPIO_ACTIVE_HIGH>;
    reset-gpios = <&ov5640_1_gpio 6 GPIO_ACTIVE_LOW>;

    avdd-supply = <&switch_ov5640_1_avdd>;
    dvdd-supply = <&switch_ov5640_1_dvdd>;

    clocks = <&ov5640_1_camera_clk>;

    port {
        ov5640_1_cam_ep: endpoint {
            clock-lanes = <0>;
            data-lanes = <1>;
            remote-endpoint = <&csi2a_ep>;
        };
    };
};

>From 919995491fb34cf7e2bd8a331c47e45cad677ce6 Mon Sep 17 00:00:00 2001
From: Michael Allwright <allse...@gmail.com>
Date: Mon, 10 Aug 2015 16:55:57 +0200
Subject: [PATCH] omap4-iss: Add device support (WIP)

---
 arch/arm/boot/dts/omap4.dtsi                |  33 +++
 drivers/staging/media/omap4iss/iss.c        | 419 +++++++++++++++++++++-------
 drivers/staging/media/omap4iss/iss.h        |  11 +
 drivers/staging/media/omap4iss/iss_csi2.c   |   4 +-
 drivers/staging/media/omap4iss/iss_csiphy.c |  16 +-
 drivers/staging/media/omap4iss/iss_video.c  |   6 +-
 include/media/omap4iss.h                    |  18 +-
 7 files changed, 393 insertions(+), 114 deletions(-)

diff --git a/arch/arm/boot/dts/omap4.dtsi b/arch/arm/boot/dts/omap4.dtsi
index f884d6a..bd37437 100644
--- a/arch/arm/boot/dts/omap4.dtsi
+++ b/arch/arm/boot/dts/omap4.dtsi
@@ -923,6 +923,39 @@
             status = "disabled";
         };

+        iss: iss@52000000 {
+            compatible = "ti,omap4-iss";
+            reg = <0x52000000 0x100>, /* top */
+                  <0x52001000 0x170>, /* csi2_a_regs1 */
+                  <0x52001170 0x020>, /* camerarx_core1 */
+                  <0x52001400 0x170>, /* csi2_b_regs1 */
+                  <0x52001570 0x020>, /* camerarx_core2 */
+                  <0x52002000 0x200>, /* bte */
+                  <0x52010000 0x0a0>, /* isp_sys1 */
+                  <0x52010400 0x400>, /* isp_resizer */
+                  <0x52010800 0x800>, /* isp_ipipe */
+                  <0x52011000 0x200>, /* isp_isif */
+                  <0x52011200 0x080>; /* isp_ipipeif */
+            reg-names = "top",
+                        "csi2_a_regs1",
+                        "camerarx_core1",
+                        "csi2_b_regs1",
+                        "camerarx_core2",
+                        "bte",
+                        "isp_sys1",
+                        "isp_resizer",
+                        "isp_ipipe",
+                        "isp_isif",
+                        "isp_ipipeif";
+            status = "ok";
+            ti,hwmods = "iss";
+            interrupts = <GIC_SPI 24 IRQ_TYPE_LEVEL_HIGH>;
+            clocks = <&ducati_clk_mux_ck>, <&iss_ctrlclk>;
+            clock-names = "iss_fck", "iss_ctrlclk";
+            dmas = <&sdma 9>, <&sdma 10>, <&sdma 12>, <&sdma 13>;
+            dma-names = "1", "2", "3", "4";
+        };
+
         dss: dss@58000000 {
             compatible = "ti,omap4-dss";
             reg = <0x58000000 0x80>;
diff --git a/drivers/staging/media/omap4iss/iss.c
b/drivers/staging/media/omap4iss/iss.c
index 7ced940..0ad1206 100644
--- a/drivers/staging/media/omap4iss/iss.c
+++ b/drivers/staging/media/omap4iss/iss.c
@@ -14,6 +14,7 @@
 #include <linux/clk.h>
 #include <linux/delay.h>
 #include <linux/device.h>
+#include <linux/of_device.h>
 #include <linux/dma-mapping.h>
 #include <linux/i2c.h>
 #include <linux/interrupt.h>
@@ -28,6 +29,8 @@
 #include <media/v4l2-device.h>
 #include <media/v4l2-ctrls.h>

+#include <linux/pm_runtime.h>
+
 #include "iss.h"
 #include "iss_regs.h"

@@ -129,7 +132,8 @@ int omap4iss_get_external_info(struct iss_pipeline *pipe,
     struct iss_device *iss =
         container_of(pipe, struct iss_video, pipe)->iss;
     struct v4l2_subdev_format fmt;
-    struct v4l2_ctrl *ctrl;
+    struct v4l2_ext_controls ctrls;
+    struct v4l2_ext_control ctrl;
     int ret;

     if (!pipe->external)
@@ -149,15 +153,23 @@ int omap4iss_get_external_info(struct iss_pipeline *pipe,

     pipe->external_bpp = omap4iss_video_format_info(fmt.format.code)->bpp;

-    ctrl = v4l2_ctrl_find(pipe->external->ctrl_handler,
-                  V4L2_CID_PIXEL_RATE);
-    if (ctrl == NULL) {
-        dev_warn(iss->dev, "no pixel rate control in subdev %s\n",
+    memset(&ctrls, 0, sizeof(ctrls));
+    memset(&ctrl, 0, sizeof(ctrl));
+
+    ctrl.id = V4L2_CID_PIXEL_RATE;
+    ctrls.count = 1;
+    ctrls.controls = &ctrl;
+
+    ret = v4l2_g_ext_ctrls(pipe->external->ctrl_handler, &ctrls);
+    if (ret < 0) {
+        dev_warn(iss->dev, "no pixel rate control in subdev %s\n",
              pipe->external->name);
-        return -EPIPE;
+        return ret;
     }

-    pipe->external_rate = v4l2_ctrl_g_ctrl_int64(ctrl);
+    pipe->external_rate = ctrl.value64;
+    dev_info(iss->dev, "subdev %s pixel rate = %u\n",
+         pipe->external->name, pipe->external_rate);

     return 0;
 }
@@ -993,13 +1005,13 @@ static int iss_enable_clocks(struct iss_device *iss)
 {
     int ret;

-    ret = clk_enable(iss->iss_fck);
+    ret = clk_prepare_enable(iss->iss_fck);
     if (ret) {
         dev_err(iss->dev, "clk_enable iss_fck failed\n");
         return ret;
     }

-    ret = clk_enable(iss->iss_ctrlclk);
+    ret = clk_prepare_enable(iss->iss_ctrlclk);
     if (ret) {
         dev_err(iss->dev, "clk_enable iss_ctrlclk failed\n");
         clk_disable(iss->iss_fck);
@@ -1015,15 +1027,26 @@ static int iss_enable_clocks(struct iss_device *iss)
  */
 static void iss_disable_clocks(struct iss_device *iss)
 {
-    clk_disable(iss->iss_ctrlclk);
-    clk_disable(iss->iss_fck);
+    clk_disable_unprepare(iss->iss_ctrlclk);
+    clk_disable_unprepare(iss->iss_fck);
 }

+struct clk_core_fake {
+    const char        *name;
+};
+
+struct clk_fake {
+    struct clk_core_fake    *core;
+    const char *dev_id;
+    const char *con_id;
+};
+
+
 static int iss_get_clocks(struct iss_device *iss)
 {
-    iss->iss_fck = devm_clk_get(iss->dev, "iss_fck");
+    iss->iss_fck = devm_clk_get(iss->dev, "ducati_clk_mux_ck");
     if (IS_ERR(iss->iss_fck)) {
-        dev_err(iss->dev, "Unable to get iss_fck clock info\n");
+        dev_err(iss->dev, "Unable to get ducati_clk_mux_ck clock info\n");
         return PTR_ERR(iss->iss_fck);
     }

@@ -1033,6 +1056,11 @@ static int iss_get_clocks(struct iss_device *iss)
         return PTR_ERR(iss->iss_ctrlclk);
     }

+    dev_info(iss->dev, "Got clocks\n%s(%s:%s)\n%s(%s:%s)\n", ((struct
clk_core_fake*)((struct clk_fake*)iss->iss_fck)->core)->name,
+                                 ((struct
clk_fake*)iss->iss_fck)->dev_id, ((struct
clk_fake*)iss->iss_fck)->con_id,
+                                 ((struct clk_core_fake*)((struct
clk_fake*)iss->iss_ctrlclk)->core)->name,
+                                 ((struct
clk_fake*)iss->iss_ctrlclk)->dev_id, ((struct
clk_fake*)iss->iss_ctrlclk)->con_id);
+
     return 0;
 }

@@ -1125,58 +1153,221 @@ static void iss_unregister_entities(struct
iss_device *iss)
 }

 /*
- * iss_register_subdev_group - Register a group of subdevices
+ * iss_register_subdev - Register a sub-device
  * @iss: OMAP4 ISS device
- * @board_info: I2C subdevs board information array
+ * @iss_subdev: platform data related to a sub-device
  *
- * Register all I2C subdevices in the board_info array. The array must be
- * terminated by a NULL entry, and the first entry must be the sensor.
+ * Register an I2C sub-device which has not been registered by other
+ * means (such as the Device Tree).
  *
- * Return a pointer to the sensor media entity if it has been successfully
+ * Return a pointer to the sub-device if it has been successfully
  * registered, or NULL otherwise.
  */
 static struct v4l2_subdev *
-iss_register_subdev_group(struct iss_device *iss,
-             struct iss_subdev_i2c_board_info *board_info)
+iss_register_subdev(struct iss_device *iss,
+            struct iss_platform_subdev *iss_subdev)
 {
-    struct v4l2_subdev *sensor = NULL;
-    unsigned int first;
+    struct i2c_adapter *adapter;
+    struct v4l2_subdev *sd;

-    if (board_info->board_info == NULL)
+    if (iss_subdev->board_info == NULL)
         return NULL;

-    for (first = 1; board_info->board_info; ++board_info, first = 0) {
-        struct v4l2_subdev *subdev;
-        struct i2c_adapter *adapter;
+    adapter = i2c_get_adapter(iss_subdev->i2c_adapter_id);
+    if (adapter == NULL) {
+        dev_err(iss->dev,
+            "%s: Unable to get I2C adapter %d for device %s\n",
+            __func__, iss_subdev->i2c_adapter_id,
+            iss_subdev->board_info->type);
+        return NULL;
+    }

-        adapter = i2c_get_adapter(board_info->i2c_adapter_id);
-        if (adapter == NULL) {
-            dev_err(iss->dev,
-                "%s: Unable to get I2C adapter %d for device %s\n",
-                __func__, board_info->i2c_adapter_id,
-                board_info->board_info->type);
-            continue;
+    sd = v4l2_i2c_new_subdev_board(&iss->v4l2_dev, adapter,
+                       iss_subdev->board_info, NULL);
+    if (sd == NULL) {
+        dev_err(iss->dev, "%s: Unable to register subdev %s\n",
+            __func__, iss_subdev->board_info->type);
+        return NULL;
+    }
+
+    return sd;
+}
+
+
+static int iss_link_entity(
+    struct iss_device *iss, struct media_entity *entity,
+    enum iss_interface_type interface)
+{
+    struct media_entity *input;
+    unsigned int flags;
+    unsigned int pad;
+    unsigned int i;
+
+    /* Connect the sensor to the correct interface module.
+     * serial sensors are connected to the CSI2a or CSI2b
+     */
+    switch (interface) {
+
+    case ISS_INTERFACE_CSI2A_PHY1:
+        input = &iss->csi2a.subdev.entity;
+        pad = CSI2_PAD_SINK;
+        flags = MEDIA_LNK_FL_IMMUTABLE | MEDIA_LNK_FL_ENABLED;
+        break;
+
+    case ISS_INTERFACE_CSI2B_PHY2:
+        input = &iss->csi2b.subdev.entity;
+        pad = CSI2_PAD_SINK;
+        flags = MEDIA_LNK_FL_IMMUTABLE | MEDIA_LNK_FL_ENABLED;
+        break;
+
+    default:
+        dev_err(iss->dev, "%s: invalid interface type %u\n", __func__,
+            interface);
+        return -EINVAL;
+    }
+
+    for (i = 0; i < entity->num_pads; i++) {
+        if (entity->pads[i].flags & MEDIA_PAD_FL_SOURCE)
+            break;
+    }
+    if (i == entity->num_pads) {
+        dev_err(iss->dev, "%s: no source pad in external entity\n",
+            __func__);
+        return -EINVAL;
+    }
+
+    return media_entity_create_link(entity, i, input, pad, flags);
+}
+
+static int iss_of_parse_node(struct device *dev, struct device_node *node,
+                 struct iss_async_subdev *isd)
+{
+    struct iss_bus_cfg *buscfg = &isd->bus;
+    struct v4l2_of_endpoint vep;
+    unsigned int i, lanes;
+
+    v4l2_of_parse_endpoint(node, &vep);
+
+    dev_info(dev, "parsing endpoint %s, interface %u\n", node->full_name,
+        vep.base.port);
+
+    switch(vep.base.port) {
+    case ISS_INTERFACE_CSI2A_PHY1:
+        buscfg->interface = ISS_INTERFACE_CSI2A_PHY1;
+        lanes = ISS_CSIPHY1_NUM_DATA_LANES;
+        break;
+
+    case ISS_INTERFACE_CSI2B_PHY2:
+        buscfg->interface = ISS_INTERFACE_CSI2B_PHY2;
+        lanes = ISS_CSIPHY2_NUM_DATA_LANES;
+        break;
+
+    default:
+        dev_warn(dev, "%s: invalid interface %u\n", node->full_name,
+             vep.base.port);
+        return -EINVAL;
+    }
+
+    buscfg->bus.csi2.lanecfg.clk.pos = vep.bus.mipi_csi2.clock_lane;
+    buscfg->bus.csi2.lanecfg.clk.pol =
+        vep.bus.mipi_csi2.lane_polarities[0];
+    dev_info(dev, "clock lane polarity %u, pos %u\n",
+        buscfg->bus.csi2.lanecfg.clk.pol,
+        buscfg->bus.csi2.lanecfg.clk.pos);
+
+    for (i = 0; i < lanes; i++) {
+        buscfg->bus.csi2.lanecfg.data[i].pos =
+            vep.bus.mipi_csi2.data_lanes[i];
+        buscfg->bus.csi2.lanecfg.data[i].pol =
+            vep.bus.mipi_csi2.lane_polarities[i + 1];
+        dev_info(dev, "data lane %u polarity %u, pos %u\n", i,
+            buscfg->bus.csi2.lanecfg.data[i].pol,
+            buscfg->bus.csi2.lanecfg.data[i].pos);
+    }
+
+    /*
+     * FIXME: now we assume the CRC is always there.
+     * Implement a way to obtain this information from the
+     * sensor. Frame descriptors, perhaps?
+     */
+    buscfg->bus.csi2.crc = 0;
+
+    return 0;
+}
+
+static int iss_of_parse_nodes(struct device *dev,
+                  struct v4l2_async_notifier *notifier)
+{
+    struct device_node *node = NULL;
+
+    notifier->subdevs = devm_kcalloc(
+        dev, ISS_MAX_SUBDEVS, sizeof(*notifier->subdevs), GFP_KERNEL);
+    if (!notifier->subdevs)
+        return -ENOMEM;
+
+    while (notifier->num_subdevs < ISS_MAX_SUBDEVS &&
+           (node = of_graph_get_next_endpoint(dev->of_node, node))) {
+        struct iss_async_subdev *isd;
+
+        isd = devm_kzalloc(dev, sizeof(*isd), GFP_KERNEL);
+        if (!isd) {
+            of_node_put(node);
+            return -ENOMEM;
         }

-        subdev = v4l2_i2c_new_subdev_board(&iss->v4l2_dev, adapter,
-                board_info->board_info, NULL);
-        if (subdev == NULL) {
-            dev_err(iss->dev, "Unable to register subdev %s\n",
-                board_info->board_info->type);
-            continue;
+        notifier->subdevs[notifier->num_subdevs] = &isd->asd;
+
+        if (iss_of_parse_node(dev, node, isd)) {
+            of_node_put(node);
+            return -EINVAL;
         }

-        if (first)
-            sensor = subdev;
+        isd->asd.match.of.node = of_graph_get_remote_port_parent(node);
+        of_node_put(node);
+        if (!isd->asd.match.of.node) {
+            dev_warn(dev, "bad remote port parent\n");
+            return -EINVAL;
+        }
+
+        isd->asd.match_type = V4L2_ASYNC_MATCH_OF;
+        notifier->num_subdevs++;
     }

-    return sensor;
+    return notifier->num_subdevs;
+}
+
+static int iss_subdev_notifier_bound(struct v4l2_async_notifier *async,
+                     struct v4l2_subdev *subdev,
+                     struct v4l2_async_subdev *asd)
+{
+    struct iss_device *iss = container_of(async, struct iss_device,
+                          notifier);
+    struct iss_async_subdev *isd =
+        container_of(asd, struct iss_async_subdev, asd);
+    int ret;
+
+    ret = iss_link_entity(iss, &subdev->entity, isd->bus.interface);
+    if (ret < 0)
+        return ret;
+
+    isd->sd = subdev;
+    isd->sd->host_priv = &isd->bus;
+
+    return ret;
+}
+
+static int iss_subdev_notifier_complete(struct v4l2_async_notifier *async)
+{
+    struct iss_device *iss = container_of(async, struct iss_device,
+                          notifier);
+
+    return v4l2_device_register_subdev_nodes(&iss->v4l2_dev);
 }

 static int iss_register_entities(struct iss_device *iss)
 {
     struct iss_platform_data *pdata = iss->pdata;
-    struct iss_v4l2_subdevs_group *subdevs;
+    struct iss_platform_subdev *iss_subdev;
     int ret;

     iss->media_dev.dev = iss->dev;
@@ -1220,56 +1411,40 @@ static int iss_register_entities(struct iss_device *iss)
     if (ret < 0)
         goto done;

+    /*
+     * Device Tree --- the external sub-devices will be registered
+     * later. The same goes for the sub-device node registration.
+     */
+    if (iss->dev->of_node)
+        return 0;
+
     /* Register external entities */
-    for (subdevs = pdata->subdevs; subdevs && subdevs->subdevs; ++subdevs) {
-        struct v4l2_subdev *sensor;
-        struct media_entity *input;
-        unsigned int flags;
-        unsigned int pad;
-
-        sensor = iss_register_subdev_group(iss, subdevs->subdevs);
-        if (sensor == NULL)
-            continue;
+    for (iss_subdev = pdata ? pdata->subdevs : NULL;
+         iss_subdev && iss_subdev->board_info; iss_subdev++) {
+        struct v4l2_subdev *sd;

-        sensor->host_priv = subdevs;
+        sd = iss_register_subdev(iss, iss_subdev);

-        /* Connect the sensor to the correct interface module.
-         * CSI2a receiver through CSIPHY1, or
-         * CSI2b receiver through CSIPHY2
+        /*
+         * No bus information --- this is either a flash or a
+         * lens subdev.
          */
-        switch (subdevs->interface) {
-        case ISS_INTERFACE_CSI2A_PHY1:
-            input = &iss->csi2a.subdev.entity;
-            pad = CSI2_PAD_SINK;
-            flags = MEDIA_LNK_FL_IMMUTABLE
-                  | MEDIA_LNK_FL_ENABLED;
-            break;
+        if (!sd || !iss_subdev->bus)
+            continue;

-        case ISS_INTERFACE_CSI2B_PHY2:
-            input = &iss->csi2b.subdev.entity;
-            pad = CSI2_PAD_SINK;
-            flags = MEDIA_LNK_FL_IMMUTABLE
-                  | MEDIA_LNK_FL_ENABLED;
-            break;
+        sd->host_priv = iss_subdev->bus;

-        default:
-            dev_err(iss->dev, "invalid interface type %u\n",
-                subdevs->interface);
-            ret = -EINVAL;
-            goto done;
-        }
-
-        ret = media_entity_create_link(&sensor->entity, 0, input, pad,
-                           flags);
+        ret = iss_link_entity(iss, &sd->entity,
+                      iss_subdev->bus->interface);
         if (ret < 0)
             goto done;
     }

     ret = v4l2_device_register_subdev_nodes(&iss->v4l2_dev);
-
 done:
-    if (ret < 0)
+    if (ret < 0) {
         iss_unregister_entities(iss);
+    }

     return ret;
 }
@@ -1362,24 +1537,62 @@ error_csiphy:
     return ret;
 }

+
+/*
+We need a better solution for this
+*/
+#include <../arch/arm/mach-omap2/omap-pm.h>
+
+static void iss_set_constraints(struct iss_device *iss, bool enable)
+{
+    if (!iss)
+        return;
+
+    /* FIXME: Look for something more precise as a good throughtput limit */
+    omap_pm_set_min_bus_tput(iss->dev, OCP_INITIATOR_AGENT,
+                 enable ? 800000 : -1);
+}
+
+static struct iss_platform_data iss_dummy_pdata = {
+    .set_constraints = iss_set_constraints,
+};
+
 static int iss_probe(struct platform_device *pdev)
 {
-    struct iss_platform_data *pdata = pdev->dev.platform_data;
     struct iss_device *iss;
     unsigned int i;
-    int ret;
-
-    if (pdata == NULL)
-        return -EINVAL;
+    int ret, r;

     iss = devm_kzalloc(&pdev->dev, sizeof(*iss), GFP_KERNEL);
-    if (!iss)
+    if (!iss) {
+        dev_err(&pdev->dev, "could not allocate memory\n");
         return -ENOMEM;
+    }
+
+    if (IS_ENABLED(CONFIG_OF) && pdev->dev.of_node) {
+        ret = iss_of_parse_nodes(&pdev->dev, &iss->notifier);
+        if (ret < 0)
+            return ret;
+        ret = v4l2_async_notifier_register(&iss->v4l2_dev,
+                           &iss->notifier);
+        if (ret)
+            return ret;
+
+        /* use dummy pdata with set constraints function */
+        iss->pdata = &iss_dummy_pdata;
+    } else {
+        iss->pdata = pdev->dev.platform_data;
+        dev_warn(&pdev->dev,
+             "Platform data support is deprecated! Please move to DT now!\n");
+    }
+
+    pm_runtime_enable(&pdev->dev);
+    r = pm_runtime_get_sync(&pdev->dev);

     mutex_init(&iss->iss_mutex);

     iss->dev = &pdev->dev;
-    iss->pdata = pdata;
+    iss->ref_count = 0;

     iss->raw_dmamask = DMA_BIT_MASK(32);
     iss->dev->dma_mask = &iss->raw_dmamask;
@@ -1415,30 +1628,30 @@ static int iss_probe(struct platform_device *pdev)

     iss->revision = iss_reg_read(iss, OMAP4_ISS_MEM_TOP, ISS_HL_REVISION);
     dev_info(iss->dev, "Revision %08x found\n", iss->revision);
-
+dev_info(iss->dev, "A\n");
     for (i = 1; i < OMAP4_ISS_MEM_LAST; i++) {
         ret = iss_map_mem_resource(pdev, iss, i);
         if (ret)
             goto error_iss;
     }
-
+dev_info(iss->dev, "B\n");
     /* Configure BTE BW_LIMITER field to max recommended value (1 GB) */
     iss_reg_update(iss, OMAP4_ISS_MEM_BTE, BTE_CTRL,
                BTE_CTRL_BW_LIMITER_MASK,
                18 << BTE_CTRL_BW_LIMITER_SHIFT);
-
+dev_info(iss->dev, "C\n");
     /* Perform ISP reset */
     ret = omap4iss_subclk_enable(iss, OMAP4_ISS_SUBCLK_ISP);
     if (ret < 0)
         goto error_iss;
-
+dev_info(iss->dev, "D\n");
     ret = iss_isp_reset(iss);
     if (ret < 0)
         goto error_iss;
-
+dev_info(iss->dev, "E\n");
     dev_info(iss->dev, "ISP Revision %08x found\n",
          iss_reg_read(iss, OMAP4_ISS_MEM_ISP_SYS1, ISP5_REVISION));
-
+dev_info(iss->dev, "F\n");
     /* Interrupt */
     iss->irq_num = platform_get_irq(pdev, 0);
     if (iss->irq_num <= 0) {
@@ -1446,28 +1659,32 @@ static int iss_probe(struct platform_device *pdev)
         ret = -ENODEV;
         goto error_iss;
     }
-
+dev_info(iss->dev, "G\n");
     if (devm_request_irq(iss->dev, iss->irq_num, iss_isr, IRQF_SHARED,
                  "OMAP4 ISS", iss)) {
         dev_err(iss->dev, "Unable to request IRQ\n");
         ret = -EINVAL;
         goto error_iss;
     }
-
+dev_info(iss->dev, "H\n");
     /* Entities */
     ret = iss_initialize_modules(iss);
     if (ret < 0)
         goto error_iss;
+dev_info(iss->dev, "I\n");
+    iss->notifier.bound = iss_subdev_notifier_bound;
+    iss->notifier.complete = iss_subdev_notifier_complete;

     ret = iss_register_entities(iss);
     if (ret < 0)
         goto error_modules;
-
+dev_info(iss->dev, "J\n");
     omap4iss_put(iss);
-
+dev_info(iss->dev, "K\n");
     return 0;

 error_modules:
+    v4l2_async_notifier_unregister(&iss->notifier);
     iss_cleanup_modules(iss);
 error_iss:
     omap4iss_put(iss);
@@ -1495,12 +1712,20 @@ static struct platform_device_id omap4iss_id_table[] = {
 };
 MODULE_DEVICE_TABLE(platform, omap4iss_id_table);

+static struct of_device_id omap4iss_of_table[] = {
+    { .compatible = "ti,omap4-iss" },
+    { },
+};
+MODULE_DEVICE_TABLE(of, omap4iss_of_table);
+
+
 static struct platform_driver iss_driver = {
     .probe        = iss_probe,
     .remove        = iss_remove,
     .id_table    = omap4iss_id_table,
     .driver = {
         .name    = "omap4iss",
+        .of_match_table = omap4iss_of_table,
     },
 };

diff --git a/drivers/staging/media/omap4iss/iss.h
b/drivers/staging/media/omap4iss/iss.h
index 35df8b4..7830061 100644
--- a/drivers/staging/media/omap4iss/iss.h
+++ b/drivers/staging/media/omap4iss/iss.h
@@ -15,6 +15,7 @@
 #define _OMAP4_ISS_H_

 #include <media/v4l2-device.h>
+#include <media/v4l2-of.h>
 #include <linux/device.h>
 #include <linux/io.h>
 #include <linux/platform_device.h>
@@ -86,6 +87,7 @@ struct iss_reg {
  */
 struct iss_device {
     struct v4l2_device v4l2_dev;
+    struct v4l2_async_notifier notifier;
     struct media_device media_dev;
     struct device *dev;
     u32 revision;
@@ -119,6 +121,15 @@ struct iss_device {

     unsigned int subclk_resources;
     unsigned int isp_subclk_resources;
+
+#define ISS_MAX_SUBDEVS        2
+    struct v4l2_subdev *subdevs[ISS_MAX_SUBDEVS];
+};
+
+struct iss_async_subdev {
+    struct v4l2_subdev *sd;
+    struct iss_bus_cfg bus;
+    struct v4l2_async_subdev asd;
 };

 #define v4l2_dev_to_iss_device(dev) \
diff --git a/drivers/staging/media/omap4iss/iss_csi2.c
b/drivers/staging/media/omap4iss/iss_csi2.c
index d7ff769..0002869 100644
--- a/drivers/staging/media/omap4iss/iss_csi2.c
+++ b/drivers/staging/media/omap4iss/iss_csi2.c
@@ -479,6 +479,7 @@ static void csi2_irq_status_set(struct
iss_csi2_device *csi2, int enable)
         iss_reg_write(csi2->iss, csi2->regs1, CSI2_IRQENABLE, 0);
 }

+static void csi2_print_status(struct iss_csi2_device *csi2);
 /*
  * omap4iss_csi2_reset - Resets the CSI2 module.
  *
@@ -515,6 +516,7 @@ int omap4iss_csi2_reset(struct iss_csi2_device *csi2)
         REGISTER1_RESET_DONE_CTRLCLK, 10000, 100, 500);
     if (timeout) {
         dev_err(csi2->iss->dev, "CSI2: CSI2_96M_FCLK reset timeout!\n");
+csi2_print_status(csi2);
         return -EBUSY;
     }

@@ -528,7 +530,7 @@ int omap4iss_csi2_reset(struct iss_csi2_device *csi2)

 static int csi2_configure(struct iss_csi2_device *csi2)
 {
-    const struct iss_v4l2_subdevs_group *pdata;
+    const struct iss_bus_cfg *pdata;
     struct iss_csi2_timing_cfg *timing = &csi2->timing[0];
     struct v4l2_subdev *sensor;
     struct media_pad *pad;
diff --git a/drivers/staging/media/omap4iss/iss_csiphy.c
b/drivers/staging/media/omap4iss/iss_csiphy.c
index 748607f..da59dca 100644
--- a/drivers/staging/media/omap4iss/iss_csiphy.c
+++ b/drivers/staging/media/omap4iss/iss_csiphy.c
@@ -121,7 +121,7 @@ int omap4iss_csiphy_config(struct iss_device *iss,
 {
     struct iss_csi2_device *csi2 = v4l2_get_subdevdata(csi2_subdev);
     struct iss_pipeline *pipe = to_iss_pipeline(&csi2_subdev->entity);
-    struct iss_v4l2_subdevs_group *subdevs = pipe->external->host_priv;
+    struct iss_bus_cfg *buscfg = pipe->external->host_priv;
     struct iss_csiphy_dphy_cfg csi2phy;
     int csi2_ddrclk_khz;
     struct iss_csiphy_lanes_cfg *lanes;
@@ -129,7 +129,14 @@ int omap4iss_csiphy_config(struct iss_device *iss,
     u32 cam_rx_ctrl;
     unsigned int i;

-    lanes = &subdevs->bus.csi2.lanecfg;
+    if (!buscfg) {
+        struct iss_async_subdev *isd =
+            container_of(pipe->external->asd,
+                     struct iss_async_subdev, asd);
+        buscfg = &isd->bus;
+    }
+
+    lanes = &buscfg->bus.csi2.lanecfg;

     /*
      * SCM.CONTROL_CAMERA_RX
@@ -147,7 +154,8 @@ int omap4iss_csiphy_config(struct iss_device *iss,
      */
     regmap_read(iss->syscon, 0x68, &cam_rx_ctrl);

-    if (subdevs->interface == ISS_INTERFACE_CSI2A_PHY1) {
+
+    if (buscfg->interface == ISS_INTERFACE_CSI2A_PHY1) {
         cam_rx_ctrl &= ~(OMAP4_CAMERARX_CSI21_LANEENABLE_MASK |
                 OMAP4_CAMERARX_CSI21_CAMMODE_MASK);
         /* NOTE: Leave CSIPHY1 config to 0x0: D-PHY mode */
@@ -158,7 +166,7 @@ int omap4iss_csiphy_config(struct iss_device *iss,
         cam_rx_ctrl |= OMAP4_CAMERARX_CSI21_CTRLCLKEN_MASK;
     }

-    if (subdevs->interface == ISS_INTERFACE_CSI2B_PHY2) {
+    if (buscfg->interface == ISS_INTERFACE_CSI2B_PHY2) {
         cam_rx_ctrl &= ~(OMAP4_CAMERARX_CSI22_LANEENABLE_MASK |
                 OMAP4_CAMERARX_CSI22_CAMMODE_MASK);
         /* NOTE: Leave CSIPHY2 config to 0x0: D-PHY mode */
diff --git a/drivers/staging/media/omap4iss/iss_video.c
b/drivers/staging/media/omap4iss/iss_video.c
index 85c54fe..daf85c4 100644
--- a/drivers/staging/media/omap4iss/iss_video.c
+++ b/drivers/staging/media/omap4iss/iss_video.c
@@ -844,7 +844,7 @@ iss_video_streamon(struct file *file, void *fh,
enum v4l2_buf_type type)
     pipe->external_bpp = 0;
     pipe->entities = 0;

-    if (video->iss->pdata->set_constraints)
+    if (video->iss->pdata && video->iss->pdata->set_constraints)
         video->iss->pdata->set_constraints(video->iss, true);

     ret = media_entity_pipeline_start(&video->video.entity, &pipe->pipe);
@@ -932,7 +932,7 @@ err_omap4iss_set_stream:
 err_iss_video_check_format:
     media_entity_pipeline_stop(&video->video.entity);
 err_media_entity_pipeline_start:
-    if (video->iss->pdata->set_constraints)
+    if (video->iss->pdata && video->iss->pdata->set_constraints)
         video->iss->pdata->set_constraints(video->iss, false);
     video->queue = NULL;

@@ -974,7 +974,7 @@ iss_video_streamoff(struct file *file, void *fh,
enum v4l2_buf_type type)
     vb2_streamoff(&vfh->queue, type);
     video->queue = NULL;

-    if (video->iss->pdata->set_constraints)
+    if (video->iss->pdata && video->iss->pdata->set_constraints)
         video->iss->pdata->set_constraints(video->iss, false);
     media_entity_pipeline_stop(&video->video.entity);

diff --git a/include/media/omap4iss.h b/include/media/omap4iss.h
index 0d7620d..8f25cf1 100644
--- a/include/media/omap4iss.h
+++ b/include/media/omap4iss.h
@@ -6,7 +6,7 @@
 struct iss_device;

 enum iss_interface_type {
-    ISS_INTERFACE_CSI2A_PHY1,
+    ISS_INTERFACE_CSI2A_PHY1 = 0,
     ISS_INTERFACE_CSI2B_PHY2,
 };

@@ -44,21 +44,21 @@ struct iss_csi2_platform_data {
     struct iss_csiphy_lanes_cfg lanecfg;
 };

-struct iss_subdev_i2c_board_info {
-    struct i2c_board_info *board_info;
-    int i2c_adapter_id;
-};
-
-struct iss_v4l2_subdevs_group {
-    struct iss_subdev_i2c_board_info *subdevs;
+struct iss_bus_cfg {
     enum iss_interface_type interface;
     union {
         struct iss_csi2_platform_data csi2;
     } bus; /* gcc < 4.6.0 chokes on anonymous union initializers */
 };

+struct iss_platform_subdev {
+    struct i2c_board_info *board_info;
+    int i2c_adapter_id;
+    struct iss_bus_cfg *bus;
+};
+
 struct iss_platform_data {
-    struct iss_v4l2_subdevs_group *subdevs;
+    struct iss_platform_subdev *subdevs;
     void (*set_constraints)(struct iss_device *iss, bool enable);
 };

-- 
1.9.1
--
To unsubscribe from this list: send the line "unsubscribe linux-media" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to