Lines Matching +full:en +full:- +full:active +full:- +full:lanes

1 // SPDX-License-Identifier: GPL-2.0
3 * Driver for Xilinx MIPI CSI-2 Rx Subsystem
5 * Copyright (C) 2016 - 2020 Xilinx, Inc.
19 #include <linux/v4l2-subdev.h>
20 #include <media/media-entity.h>
21 #include <media/v4l2-common.h>
22 #include <media/v4l2-ctrls.h>
23 #include <media/v4l2-fwnode.h>
24 #include <media/v4l2-subdev.h>
25 #include "xilinx-vip.h"
118 /* MIPI CSI-2 Data Types from spec */
142 #define XCSI_VCX_NUM_EVENTS ((XCSI_MAX_VCX - XCSI_MAX_VC) * 2)
145 * struct xcsi2rxss_event - Event log structure
182 * This table provides a mapping between CSI-2 Data type
215 * struct xcsi2rxss_state - CSI-2 Rx Subsystem device structure
217 * @format: Active V4L2 formats on each pad
226 * @max_num_lanes: Maximum number of lanes present
231 * @enable_active_lanes: If number of active lanes can be modified
273 return ioread32(xcsi2rxss->iomem + addr); in xcsi2rxss_read()
279 iowrite32(value, xcsi2rxss->iomem + addr); in xcsi2rxss_write()
305 if (n-- == 0) in xcsi2rxss_get_nth_mbus()
327 * xcsi2rxss_soft_reset - Does a soft reset of the MIPI CSI-2 Rx Subsystem
328 * @state: Xilinx CSI-2 Rx Subsystem structure pointer
333 * Return: 0 - on success OR -ETIME if reset times out
343 dev_err(state->dev, "soft reset timed out!\n"); in xcsi2rxss_soft_reset()
344 return -ETIME; in xcsi2rxss_soft_reset()
347 timeout--; in xcsi2rxss_soft_reset()
357 if (!state->rst_gpio) in xcsi2rxss_hard_reset()
361 gpiod_set_value_cansleep(state->rst_gpio, 1); in xcsi2rxss_hard_reset()
363 gpiod_set_value_cansleep(state->rst_gpio, 0); in xcsi2rxss_hard_reset()
371 state->events[i] = 0; in xcsi2rxss_reset_event_counters()
374 state->vcx_events[i] = 0; in xcsi2rxss_reset_event_counters()
380 struct device *dev = state->dev; in xcsi2rxss_log_counters()
384 if (state->events[i] > 0) { in xcsi2rxss_log_counters()
387 state->events[i]); in xcsi2rxss_log_counters()
391 if (state->en_vcx) { in xcsi2rxss_log_counters()
393 if (state->vcx_events[i] > 0) { in xcsi2rxss_log_counters()
398 state->vcx_events[i]); in xcsi2rxss_log_counters()
405 * xcsi2rxss_log_status - Logs the status of the CSI-2 Receiver
408 * This function prints the current status of Xilinx MIPI CSI-2
415 struct device *dev = xcsi2rxss->dev; in xcsi2rxss_log_status()
419 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_log_status()
457 if (xcsi2rxss->en_vcx) in xcsi2rxss_log_status()
484 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_log_status()
494 if (!remote || !is_media_entity_v4l2_subdev(remote->entity)) in xcsi2rxss_get_remote_subdev()
497 return media_entity_to_v4l2_subdev(remote->entity); in xcsi2rxss_get_remote_subdev()
509 state->streaming = false; in xcsi2rxss_start_stream()
518 state->streaming = true; in xcsi2rxss_start_stream()
520 state->rsubdev = in xcsi2rxss_start_stream()
521 xcsi2rxss_get_remote_subdev(&state->pads[XVIP_PAD_SINK]); in xcsi2rxss_start_stream()
523 ret = v4l2_subdev_call(state->rsubdev, video, s_stream, 1); in xcsi2rxss_start_stream()
531 state->streaming = false; in xcsi2rxss_start_stream()
539 v4l2_subdev_call(state->rsubdev, video, s_stream, 0); in xcsi2rxss_stop_stream()
547 state->streaming = false; in xcsi2rxss_stop_stream()
551 * xcsi2rxss_irq_handler - Interrupt handler for CSI-2
563 struct device *dev = state->dev; in xcsi2rxss_irq_handler()
626 state->events[i]++; in xcsi2rxss_irq_handler()
629 state->events[i]); in xcsi2rxss_irq_handler()
632 if (status & XCSI_ISR_VCXFE && state->en_vcx) { in xcsi2rxss_irq_handler()
640 state->vcx_events[i]++; in xcsi2rxss_irq_handler()
650 * xcsi2rxss_s_stream - It is used to start/stop the streaming.
655 * Xilinx MIPI CSI-2 Rx Subsystem.
664 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_s_stream()
666 if (enable == xcsi2rxss->streaming) in xcsi2rxss_s_stream()
678 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_s_stream()
689 return v4l2_subdev_get_try_format(&xcsi2rxss->subdev, cfg, pad); in __xcsi2rxss_get_pad_format()
691 return &xcsi2rxss->format; in __xcsi2rxss_get_pad_format()
698 * xcsi2rxss_init_cfg - Initialise the pad format config to default
714 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_init_cfg()
717 *format = xcsi2rxss->default_format; in xcsi2rxss_init_cfg()
719 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_init_cfg()
725 * xcsi2rxss_get_format - Get the pad format
740 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_get_format()
741 fmt->format = *__xcsi2rxss_get_pad_format(xcsi2rxss, cfg, fmt->pad, in xcsi2rxss_get_format()
742 fmt->which); in xcsi2rxss_get_format()
743 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_get_format()
749 * xcsi2rxss_set_format - This is used to set the pad format
769 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
772 * Only the format->code parameter matters for CSI as the in xcsi2rxss_set_format()
777 fmt->pad, fmt->which); in xcsi2rxss_set_format()
780 if (fmt->pad == XVIP_PAD_SOURCE) { in xcsi2rxss_set_format()
781 fmt->format = *__format; in xcsi2rxss_set_format()
782 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
791 dt = xcsi2rxss_get_dt(fmt->format.code); in xcsi2rxss_set_format()
792 if (dt != xcsi2rxss->datatype && dt != XCSI_DT_RAW8) { in xcsi2rxss_set_format()
793 dev_dbg(xcsi2rxss->dev, "Unsupported media bus format"); in xcsi2rxss_set_format()
795 fmt->format.code = xcsi2rxss_get_nth_mbus(xcsi2rxss->datatype, in xcsi2rxss_set_format()
799 *__format = fmt->format; in xcsi2rxss_set_format()
800 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
806 * xcsi2rxss_enum_mbus_code - Handle pixel format enumeration
811 * Return: -EINVAL or zero on success
822 if (code->index < 4) { in xcsi2rxss_enum_mbus_code()
823 n = code->index; in xcsi2rxss_enum_mbus_code()
825 } else if (state->datatype != XCSI_DT_RAW8) { in xcsi2rxss_enum_mbus_code()
826 n = code->index - 4; in xcsi2rxss_enum_mbus_code()
827 dt = state->datatype; in xcsi2rxss_enum_mbus_code()
829 return -EINVAL; in xcsi2rxss_enum_mbus_code()
832 code->code = xcsi2rxss_get_nth_mbus(dt, n); in xcsi2rxss_enum_mbus_code()
833 if (!code->code) in xcsi2rxss_enum_mbus_code()
834 ret = -EINVAL; in xcsi2rxss_enum_mbus_code()
839 /* -----------------------------------------------------------------------------
871 struct device *dev = xcsi2rxss->dev; in xcsi2rxss_parse_of()
872 struct device_node *node = dev->of_node; in xcsi2rxss_parse_of()
881 en_csi_v20 = of_property_read_bool(node, "xlnx,en-csi-v2-0"); in xcsi2rxss_parse_of()
883 xcsi2rxss->en_vcx = of_property_read_bool(node, "xlnx,en-vcx"); in xcsi2rxss_parse_of()
885 xcsi2rxss->enable_active_lanes = in xcsi2rxss_parse_of()
886 of_property_read_bool(node, "xlnx,en-active-lanes"); in xcsi2rxss_parse_of()
888 ret = of_property_read_u32(node, "xlnx,csi-pxl-format", in xcsi2rxss_parse_of()
889 &xcsi2rxss->datatype); in xcsi2rxss_parse_of()
891 dev_err(dev, "missing xlnx,csi-pxl-format property\n"); in xcsi2rxss_parse_of()
895 switch (xcsi2rxss->datatype) { in xcsi2rxss_parse_of()
913 ret = -EINVAL; in xcsi2rxss_parse_of()
918 ret = -EINVAL; in xcsi2rxss_parse_of()
921 dev_err(dev, "invalid csi-pxl-format property!\n"); in xcsi2rxss_parse_of()
928 return -EINVAL; in xcsi2rxss_parse_of()
936 return -EINVAL; in xcsi2rxss_parse_of()
946 dev_dbg(dev, "mipi number lanes = %d\n", in xcsi2rxss_parse_of()
949 xcsi2rxss->max_num_lanes = vep.bus.mipi_csi2.num_data_lanes; in xcsi2rxss_parse_of()
956 return -EINVAL; in xcsi2rxss_parse_of()
961 dev_dbg(dev, "vcx %s, %u data lanes (%s), data type 0x%02x\n", in xcsi2rxss_parse_of()
962 xcsi2rxss->en_vcx ? "enabled" : "disabled", in xcsi2rxss_parse_of()
963 xcsi2rxss->max_num_lanes, in xcsi2rxss_parse_of()
964 xcsi2rxss->enable_active_lanes ? "dynamic" : "static", in xcsi2rxss_parse_of()
965 xcsi2rxss->datatype); in xcsi2rxss_parse_of()
975 struct device *dev = &pdev->dev; in xcsi2rxss_probe()
980 return -ENOMEM; in xcsi2rxss_probe()
982 xcsi2rxss->dev = dev; in xcsi2rxss_probe()
984 xcsi2rxss->clks = devm_kmemdup(dev, xcsi2rxss_clks, in xcsi2rxss_probe()
986 if (!xcsi2rxss->clks) in xcsi2rxss_probe()
987 return -ENOMEM; in xcsi2rxss_probe()
990 xcsi2rxss->rst_gpio = devm_gpiod_get_optional(dev, "video-reset", in xcsi2rxss_probe()
992 if (IS_ERR(xcsi2rxss->rst_gpio)) { in xcsi2rxss_probe()
993 if (PTR_ERR(xcsi2rxss->rst_gpio) != -EPROBE_DEFER) in xcsi2rxss_probe()
995 return PTR_ERR(xcsi2rxss->rst_gpio); in xcsi2rxss_probe()
1002 xcsi2rxss->iomem = devm_platform_ioremap_resource(pdev, 0); in xcsi2rxss_probe()
1003 if (IS_ERR(xcsi2rxss->iomem)) in xcsi2rxss_probe()
1004 return PTR_ERR(xcsi2rxss->iomem); in xcsi2rxss_probe()
1018 ret = clk_bulk_get(dev, num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1023 ret = clk_bulk_prepare_enable(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1027 mutex_init(&xcsi2rxss->lock); in xcsi2rxss_probe()
1033 xcsi2rxss->pads[XVIP_PAD_SINK].flags = MEDIA_PAD_FL_SINK; in xcsi2rxss_probe()
1034 xcsi2rxss->pads[XVIP_PAD_SOURCE].flags = MEDIA_PAD_FL_SOURCE; in xcsi2rxss_probe()
1037 xcsi2rxss->default_format.code = in xcsi2rxss_probe()
1038 xcsi2rxss_get_nth_mbus(xcsi2rxss->datatype, 0); in xcsi2rxss_probe()
1039 xcsi2rxss->default_format.field = V4L2_FIELD_NONE; in xcsi2rxss_probe()
1040 xcsi2rxss->default_format.colorspace = V4L2_COLORSPACE_SRGB; in xcsi2rxss_probe()
1041 xcsi2rxss->default_format.width = XCSI_DEFAULT_WIDTH; in xcsi2rxss_probe()
1042 xcsi2rxss->default_format.height = XCSI_DEFAULT_HEIGHT; in xcsi2rxss_probe()
1043 xcsi2rxss->format = xcsi2rxss->default_format; in xcsi2rxss_probe()
1046 subdev = &xcsi2rxss->subdev; in xcsi2rxss_probe()
1048 subdev->dev = dev; in xcsi2rxss_probe()
1049 strscpy(subdev->name, dev_name(dev), sizeof(subdev->name)); in xcsi2rxss_probe()
1050 subdev->flags |= V4L2_SUBDEV_FL_HAS_EVENTS | V4L2_SUBDEV_FL_HAS_DEVNODE; in xcsi2rxss_probe()
1051 subdev->entity.ops = &xcsi2rxss_media_ops; in xcsi2rxss_probe()
1054 ret = media_entity_pads_init(&subdev->entity, XCSI_MEDIA_PADS, in xcsi2rxss_probe()
1055 xcsi2rxss->pads); in xcsi2rxss_probe()
1069 media_entity_cleanup(&subdev->entity); in xcsi2rxss_probe()
1070 mutex_destroy(&xcsi2rxss->lock); in xcsi2rxss_probe()
1071 clk_bulk_disable_unprepare(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1073 clk_bulk_put(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1080 struct v4l2_subdev *subdev = &xcsi2rxss->subdev; in xcsi2rxss_remove()
1084 media_entity_cleanup(&subdev->entity); in xcsi2rxss_remove()
1085 mutex_destroy(&xcsi2rxss->lock); in xcsi2rxss_remove()
1086 clk_bulk_disable_unprepare(num_clks, xcsi2rxss->clks); in xcsi2rxss_remove()
1087 clk_bulk_put(num_clks, xcsi2rxss->clks); in xcsi2rxss_remove()
1093 { .compatible = "xlnx,mipi-csi2-rx-subsystem-5.0", },
1100 .name = "xilinx-csi2rxss",
1110 MODULE_DESCRIPTION("Xilinx MIPI CSI-2 Rx Subsystem Driver");