Lines Matching +full:csi +full:- +full:pxl +full:- +full:format

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/mipi-csi2.h>
22 #include <media/v4l2-common.h>
23 #include <media/v4l2-ctrls.h>
24 #include <media/v4l2-fwnode.h>
25 #include <media/v4l2-subdev.h>
26 #include "xilinx-vip.h"
126 #define XCSI_VCX_NUM_EVENTS ((XCSI_MAX_VCX - XCSI_MAX_VC) * 2)
129 * struct xcsi2rxss_event - Event log structure
166 * This table provides a mapping between CSI-2 Data type
200 * struct xcsi2rxss_state - CSI-2 Rx Subsystem device structure
202 * @format: Active V4L2 formats on each pad
203 * @default_format: Default V4L2 format
223 struct v4l2_mbus_framefmt format; member
258 return ioread32(xcsi2rxss->iomem + addr); in xcsi2rxss_read()
264 iowrite32(value, xcsi2rxss->iomem + addr); in xcsi2rxss_write()
290 if (n-- == 0) in xcsi2rxss_get_nth_mbus()
298 /* This returns the data type for a media bus format else 0 */
312 * xcsi2rxss_soft_reset - Does a soft reset of the MIPI CSI-2 Rx Subsystem
313 * @state: Xilinx CSI-2 Rx Subsystem structure pointer
318 * Return: 0 - on success OR -ETIME if reset times out
328 dev_err(state->dev, "soft reset timed out!\n"); in xcsi2rxss_soft_reset()
329 return -ETIME; in xcsi2rxss_soft_reset()
332 timeout--; in xcsi2rxss_soft_reset()
342 if (!state->rst_gpio) in xcsi2rxss_hard_reset()
346 gpiod_set_value_cansleep(state->rst_gpio, 1); in xcsi2rxss_hard_reset()
348 gpiod_set_value_cansleep(state->rst_gpio, 0); in xcsi2rxss_hard_reset()
356 state->events[i] = 0; in xcsi2rxss_reset_event_counters()
359 state->vcx_events[i] = 0; in xcsi2rxss_reset_event_counters()
365 struct device *dev = state->dev; in xcsi2rxss_log_counters()
369 if (state->events[i] > 0) { in xcsi2rxss_log_counters()
372 state->events[i]); in xcsi2rxss_log_counters()
376 if (state->en_vcx) { in xcsi2rxss_log_counters()
378 if (state->vcx_events[i] > 0) { in xcsi2rxss_log_counters()
383 state->vcx_events[i]); in xcsi2rxss_log_counters()
390 * xcsi2rxss_log_status - Logs the status of the CSI-2 Receiver
393 * This function prints the current status of Xilinx MIPI CSI-2
400 struct device *dev = xcsi2rxss->dev; in xcsi2rxss_log_status()
404 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_log_status()
442 if (xcsi2rxss->en_vcx) in xcsi2rxss_log_status()
469 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_log_status()
479 if (!remote || !is_media_entity_v4l2_subdev(remote->entity)) in xcsi2rxss_get_remote_subdev()
482 return media_entity_to_v4l2_subdev(remote->entity); in xcsi2rxss_get_remote_subdev()
494 state->streaming = false; in xcsi2rxss_start_stream()
503 state->streaming = true; in xcsi2rxss_start_stream()
505 state->rsubdev = in xcsi2rxss_start_stream()
506 xcsi2rxss_get_remote_subdev(&state->pads[XVIP_PAD_SINK]); in xcsi2rxss_start_stream()
508 ret = v4l2_subdev_call(state->rsubdev, video, s_stream, 1); in xcsi2rxss_start_stream()
516 state->streaming = false; in xcsi2rxss_start_stream()
524 v4l2_subdev_call(state->rsubdev, video, s_stream, 0); in xcsi2rxss_stop_stream()
532 state->streaming = false; in xcsi2rxss_stop_stream()
536 * xcsi2rxss_irq_handler - Interrupt handler for CSI-2
548 struct device *dev = state->dev; in xcsi2rxss_irq_handler()
611 state->events[i]++; in xcsi2rxss_irq_handler()
614 state->events[i]); in xcsi2rxss_irq_handler()
617 if (status & XCSI_ISR_VCXFE && state->en_vcx) { in xcsi2rxss_irq_handler()
625 state->vcx_events[i]++; in xcsi2rxss_irq_handler()
635 * xcsi2rxss_s_stream - It is used to start/stop the streaming.
640 * Xilinx MIPI CSI-2 Rx Subsystem.
649 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_s_stream()
651 if (enable == xcsi2rxss->streaming) in xcsi2rxss_s_stream()
663 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_s_stream()
674 return v4l2_subdev_get_try_format(&xcsi2rxss->subdev, in __xcsi2rxss_get_pad_format()
677 return &xcsi2rxss->format; in __xcsi2rxss_get_pad_format()
684 * xcsi2rxss_init_cfg - Initialise the pad format config to default
688 * This function is used to initialize the pad format with the default
697 struct v4l2_mbus_framefmt *format; in xcsi2rxss_init_cfg() local
700 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_init_cfg()
702 format = v4l2_subdev_get_try_format(sd, sd_state, i); in xcsi2rxss_init_cfg()
703 *format = xcsi2rxss->default_format; in xcsi2rxss_init_cfg()
705 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_init_cfg()
711 * xcsi2rxss_get_format - Get the pad format
714 * @fmt: Pointer to pad level media bus format
716 * This function is used to get the pad format information.
726 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_get_format()
727 fmt->format = *__xcsi2rxss_get_pad_format(xcsi2rxss, sd_state, in xcsi2rxss_get_format()
728 fmt->pad, in xcsi2rxss_get_format()
729 fmt->which); in xcsi2rxss_get_format()
730 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_get_format()
736 * xcsi2rxss_set_format - This is used to set the pad format
739 * @fmt: Pointer to pad level media bus format
741 * This function is used to set the pad format. Since the pad format is fixed
742 * in hardware, it can't be modified on run time. So when a format set is
743 * requested by application, all parameters except the format type is saved
744 * for the pad and the original pad format is sent back to the application.
756 mutex_lock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
759 * Only the format->code parameter matters for CSI as the in xcsi2rxss_set_format()
760 * CSI format cannot be changed at runtime. in xcsi2rxss_set_format()
761 * Ensure that format to set is copied to over to CSI pad format in xcsi2rxss_set_format()
764 fmt->pad, fmt->which); in xcsi2rxss_set_format()
766 /* only sink pad format can be updated */ in xcsi2rxss_set_format()
767 if (fmt->pad == XVIP_PAD_SOURCE) { in xcsi2rxss_set_format()
768 fmt->format = *__format; in xcsi2rxss_set_format()
769 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
774 * RAW8 is supported in all datatypes. So if requested media bus format in xcsi2rxss_set_format()
776 * other RAW, YUV422 8/10 or RGB888, set appropriate media bus format. in xcsi2rxss_set_format()
778 dt = xcsi2rxss_get_dt(fmt->format.code); in xcsi2rxss_set_format()
779 if (dt != xcsi2rxss->datatype && dt != MIPI_CSI2_DT_RAW8) { in xcsi2rxss_set_format()
780 dev_dbg(xcsi2rxss->dev, "Unsupported media bus format"); in xcsi2rxss_set_format()
781 /* set the default format for the data type */ in xcsi2rxss_set_format()
782 fmt->format.code = xcsi2rxss_get_nth_mbus(xcsi2rxss->datatype, in xcsi2rxss_set_format()
786 *__format = fmt->format; in xcsi2rxss_set_format()
787 mutex_unlock(&xcsi2rxss->lock); in xcsi2rxss_set_format()
793 * xcsi2rxss_enum_mbus_code - Handle pixel format enumeration
798 * Return: -EINVAL or zero on success
809 if (code->index < 4) { in xcsi2rxss_enum_mbus_code()
810 n = code->index; in xcsi2rxss_enum_mbus_code()
812 } else if (state->datatype != MIPI_CSI2_DT_RAW8) { in xcsi2rxss_enum_mbus_code()
813 n = code->index - 4; in xcsi2rxss_enum_mbus_code()
814 dt = state->datatype; in xcsi2rxss_enum_mbus_code()
816 return -EINVAL; in xcsi2rxss_enum_mbus_code()
819 code->code = xcsi2rxss_get_nth_mbus(dt, n); in xcsi2rxss_enum_mbus_code()
820 if (!code->code) in xcsi2rxss_enum_mbus_code()
821 ret = -EINVAL; in xcsi2rxss_enum_mbus_code()
826 /* -----------------------------------------------------------------------------
858 struct device *dev = xcsi2rxss->dev; in xcsi2rxss_parse_of()
859 struct device_node *node = dev->of_node; in xcsi2rxss_parse_of()
868 en_csi_v20 = of_property_read_bool(node, "xlnx,en-csi-v2-0"); in xcsi2rxss_parse_of()
870 xcsi2rxss->en_vcx = of_property_read_bool(node, "xlnx,en-vcx"); in xcsi2rxss_parse_of()
872 xcsi2rxss->enable_active_lanes = in xcsi2rxss_parse_of()
873 of_property_read_bool(node, "xlnx,en-active-lanes"); in xcsi2rxss_parse_of()
875 ret = of_property_read_u32(node, "xlnx,csi-pxl-format", in xcsi2rxss_parse_of()
876 &xcsi2rxss->datatype); in xcsi2rxss_parse_of()
878 dev_err(dev, "missing xlnx,csi-pxl-format property\n"); in xcsi2rxss_parse_of()
882 switch (xcsi2rxss->datatype) { in xcsi2rxss_parse_of()
900 ret = -EINVAL; in xcsi2rxss_parse_of()
901 dev_dbg(dev, "enable csi v2 for this pixel format"); in xcsi2rxss_parse_of()
905 ret = -EINVAL; in xcsi2rxss_parse_of()
908 dev_err(dev, "invalid csi-pxl-format property!\n"); in xcsi2rxss_parse_of()
915 return -EINVAL; in xcsi2rxss_parse_of()
923 return -EINVAL; in xcsi2rxss_parse_of()
936 xcsi2rxss->max_num_lanes = vep.bus.mipi_csi2.num_data_lanes; in xcsi2rxss_parse_of()
943 return -EINVAL; in xcsi2rxss_parse_of()
949 xcsi2rxss->en_vcx ? "enabled" : "disabled", in xcsi2rxss_parse_of()
950 xcsi2rxss->max_num_lanes, in xcsi2rxss_parse_of()
951 xcsi2rxss->enable_active_lanes ? "dynamic" : "static", in xcsi2rxss_parse_of()
952 xcsi2rxss->datatype); in xcsi2rxss_parse_of()
962 struct device *dev = &pdev->dev; in xcsi2rxss_probe()
967 return -ENOMEM; in xcsi2rxss_probe()
969 xcsi2rxss->dev = dev; in xcsi2rxss_probe()
971 xcsi2rxss->clks = devm_kmemdup(dev, xcsi2rxss_clks, in xcsi2rxss_probe()
973 if (!xcsi2rxss->clks) in xcsi2rxss_probe()
974 return -ENOMEM; in xcsi2rxss_probe()
977 xcsi2rxss->rst_gpio = devm_gpiod_get_optional(dev, "video-reset", in xcsi2rxss_probe()
979 if (IS_ERR(xcsi2rxss->rst_gpio)) in xcsi2rxss_probe()
980 return dev_err_probe(dev, PTR_ERR(xcsi2rxss->rst_gpio), in xcsi2rxss_probe()
987 xcsi2rxss->iomem = devm_platform_ioremap_resource(pdev, 0); in xcsi2rxss_probe()
988 if (IS_ERR(xcsi2rxss->iomem)) in xcsi2rxss_probe()
989 return PTR_ERR(xcsi2rxss->iomem); in xcsi2rxss_probe()
1003 ret = clk_bulk_get(dev, num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1008 ret = clk_bulk_prepare_enable(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1012 mutex_init(&xcsi2rxss->lock); in xcsi2rxss_probe()
1018 xcsi2rxss->pads[XVIP_PAD_SINK].flags = MEDIA_PAD_FL_SINK; in xcsi2rxss_probe()
1019 xcsi2rxss->pads[XVIP_PAD_SOURCE].flags = MEDIA_PAD_FL_SOURCE; in xcsi2rxss_probe()
1021 /* Initialize the default format */ in xcsi2rxss_probe()
1022 xcsi2rxss->default_format.code = in xcsi2rxss_probe()
1023 xcsi2rxss_get_nth_mbus(xcsi2rxss->datatype, 0); in xcsi2rxss_probe()
1024 xcsi2rxss->default_format.field = V4L2_FIELD_NONE; in xcsi2rxss_probe()
1025 xcsi2rxss->default_format.colorspace = V4L2_COLORSPACE_SRGB; in xcsi2rxss_probe()
1026 xcsi2rxss->default_format.width = XCSI_DEFAULT_WIDTH; in xcsi2rxss_probe()
1027 xcsi2rxss->default_format.height = XCSI_DEFAULT_HEIGHT; in xcsi2rxss_probe()
1028 xcsi2rxss->format = xcsi2rxss->default_format; in xcsi2rxss_probe()
1031 subdev = &xcsi2rxss->subdev; in xcsi2rxss_probe()
1033 subdev->dev = dev; in xcsi2rxss_probe()
1034 strscpy(subdev->name, dev_name(dev), sizeof(subdev->name)); in xcsi2rxss_probe()
1035 subdev->flags |= V4L2_SUBDEV_FL_HAS_EVENTS | V4L2_SUBDEV_FL_HAS_DEVNODE; in xcsi2rxss_probe()
1036 subdev->entity.ops = &xcsi2rxss_media_ops; in xcsi2rxss_probe()
1039 ret = media_entity_pads_init(&subdev->entity, XCSI_MEDIA_PADS, in xcsi2rxss_probe()
1040 xcsi2rxss->pads); in xcsi2rxss_probe()
1054 media_entity_cleanup(&subdev->entity); in xcsi2rxss_probe()
1055 mutex_destroy(&xcsi2rxss->lock); in xcsi2rxss_probe()
1056 clk_bulk_disable_unprepare(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1058 clk_bulk_put(num_clks, xcsi2rxss->clks); in xcsi2rxss_probe()
1065 struct v4l2_subdev *subdev = &xcsi2rxss->subdev; in xcsi2rxss_remove()
1069 media_entity_cleanup(&subdev->entity); in xcsi2rxss_remove()
1070 mutex_destroy(&xcsi2rxss->lock); in xcsi2rxss_remove()
1071 clk_bulk_disable_unprepare(num_clks, xcsi2rxss->clks); in xcsi2rxss_remove()
1072 clk_bulk_put(num_clks, xcsi2rxss->clks); in xcsi2rxss_remove()
1076 { .compatible = "xlnx,mipi-csi2-rx-subsystem-5.0", },
1083 .name = "xilinx-csi2rxss",
1093 MODULE_DESCRIPTION("Xilinx MIPI CSI-2 Rx Subsystem Driver");