drm-misc-next for v5.20:
UAPI Changes: * media: Add various RGB666 and RGB888 format constants Cross-subsystem Changes: * media: Documentation Core Changes: * aperture: Fix segfault during hot-unplug * dp: Support waiting for HDP signal, plus driver updates; Port-validation fixes * fbcon: Improve scrolling performance; Sanitize input * Clean up <drm/drm_crtc.h> Driver Changes: * amdgpu: Cleanups * bridge: Add support for i.MX8qxp and i.MX8qm; anx7625: DPI fixes; tc358775: Fix clock settings; ti-sn65dsi83: Allow GPIO to sleep * panel: Set orientation from panel, plus driver updates * Several small cleanups -----BEGIN PGP SIGNATURE----- iQEzBAABCAAdFiEEchf7rIzpz2NEoWjlaA3BHVMLeiMFAmK0HdIACgkQaA3BHVML eiNRYwf+Iy6kDwpPzL8CWOiO6/qD1QP//AbT5v+Xu2nEqkCu6rDjaXGieZh7JHlA N/cPIrmRWv3QqhomyatAUE4kEntoEbllqamXv1QK3e0oe0QAc5s8cJXfa5k3uYS6 bRNHEEMPlodFvncsem/MoxRyG035R6ZNntSBVo+LhxTFU/vfeMMCmoEtk0plS0Aa HXoYCxBJLBa11E8zgT1fKur/l1T6N5duq5GzMo8c8R98ax2iWpY7ZPes5P4V3ULE 4kwY08r2LOthwkszWeb2IdB+6PXoRCrCCcJ/rS+PUftjPvasxvRBHfS4sFuOoGVu hqE/HcW18qiPyg+/iLLAD99W1Ft60w== =c98T -----END PGP SIGNATURE----- Merge tag 'drm-misc-next-2022-06-23' of git://anongit.freedesktop.org/drm/drm-misc into drm-next drm-misc-next for v5.20: UAPI Changes: * media: Add various RGB666 and RGB888 format constants Cross-subsystem Changes: * media: Documentation Core Changes: * aperture: Fix segfault during hot-unplug * dp: Support waiting for HDP signal, plus driver updates; Port-validation fixes * fbcon: Improve scrolling performance; Sanitize input * Clean up <drm/drm_crtc.h> Driver Changes: * amdgpu: Cleanups * bridge: Add support for i.MX8qxp and i.MX8qm; anx7625: DPI fixes; tc358775: Fix clock settings; ti-sn65dsi83: Allow GPIO to sleep * panel: Set orientation from panel, plus driver updates * Several small cleanups Signed-off-by: Dave Airlie <airlied@redhat.com> From: Thomas Zimmermann <tzimmermann@suse.de> Link: https://patchwork.freedesktop.org/patch/msgid/YrQeAAVvQ6jxu2dl@linux-uq9g
This commit is contained in:
commit
0936de1e96
|
@ -94,7 +94,22 @@ properties:
|
|||
$ref: /schemas/graph.yaml#/$defs/port-base
|
||||
unevaluatedProperties: false
|
||||
description:
|
||||
Video port for MIPI DSI input.
|
||||
MIPI DSI/DPI input.
|
||||
|
||||
properties:
|
||||
endpoint:
|
||||
$ref: /schemas/media/video-interfaces.yaml#
|
||||
type: object
|
||||
additionalProperties: false
|
||||
|
||||
properties:
|
||||
remote-endpoint: true
|
||||
|
||||
bus-type:
|
||||
enum: [7]
|
||||
default: 1
|
||||
|
||||
data-lanes: true
|
||||
|
||||
port@1:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
|
@ -143,6 +158,8 @@ examples:
|
|||
reg = <0>;
|
||||
anx7625_in: endpoint {
|
||||
remote-endpoint = <&mipi_dsi>;
|
||||
bus-type = <7>;
|
||||
data-lanes = <0 1 2 3>;
|
||||
};
|
||||
};
|
||||
|
||||
|
|
|
@ -0,0 +1,173 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/display/bridge/fsl,imx8qxp-ldb.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Freescale i.MX8qm/qxp LVDS Display Bridge
|
||||
|
||||
maintainers:
|
||||
- Liu Ying <victor.liu@nxp.com>
|
||||
|
||||
description: |
|
||||
The Freescale i.MX8qm/qxp LVDS Display Bridge(LDB) has two channels.
|
||||
|
||||
The i.MX8qm/qxp LDB is controlled by Control and Status Registers(CSR) module.
|
||||
The CSR module, as a system controller, contains the LDB's configuration
|
||||
registers.
|
||||
|
||||
For i.MX8qxp LDB, each channel supports up to 24bpp parallel input color
|
||||
format and can map the input to VESA or JEIDA standards. The two channels
|
||||
cannot be used simultaneously, that is to say, the user should pick one of
|
||||
them to use. Two LDB channels from two LDB instances can work together in
|
||||
LDB split mode to support a dual link LVDS display. The channel indexes
|
||||
have to be different. Channel0 outputs odd pixels and channel1 outputs
|
||||
even pixels.
|
||||
|
||||
For i.MX8qm LDB, each channel additionally supports up to 30bpp parallel
|
||||
input color format. The two channels can be used simultaneously, either
|
||||
in dual mode or split mode. In dual mode, the two channels output identical
|
||||
data. In split mode, channel0 outputs odd pixels and channel1 outputs even
|
||||
pixels.
|
||||
|
||||
A side note is that i.MX8qm/qxp LDB is officially called pixel mapper in
|
||||
the SoC reference manuals. The pixel mapper uses logic of LDBs embedded in
|
||||
i.MX6qdl/sx SoCs, i.e., it is essentially based on them. To keep the naming
|
||||
consistency, this binding calls it LDB.
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- fsl,imx8qm-ldb
|
||||
- fsl,imx8qxp-ldb
|
||||
|
||||
"#address-cells":
|
||||
const: 1
|
||||
|
||||
"#size-cells":
|
||||
const: 0
|
||||
|
||||
clocks:
|
||||
items:
|
||||
- description: pixel clock
|
||||
- description: bypass clock
|
||||
|
||||
clock-names:
|
||||
items:
|
||||
- const: pixel
|
||||
- const: bypass
|
||||
|
||||
power-domains:
|
||||
maxItems: 1
|
||||
|
||||
fsl,companion-ldb:
|
||||
$ref: /schemas/types.yaml#/definitions/phandle
|
||||
description: |
|
||||
A phandle which points to companion LDB which is used in LDB split mode.
|
||||
|
||||
patternProperties:
|
||||
"^channel@[0-1]$":
|
||||
type: object
|
||||
description: Represents a channel of LDB.
|
||||
|
||||
properties:
|
||||
"#address-cells":
|
||||
const: 1
|
||||
|
||||
"#size-cells":
|
||||
const: 0
|
||||
|
||||
reg:
|
||||
description: The channel index.
|
||||
enum: [ 0, 1 ]
|
||||
|
||||
phys:
|
||||
description: A phandle to the phy module representing the LVDS PHY.
|
||||
maxItems: 1
|
||||
|
||||
phy-names:
|
||||
const: lvds_phy
|
||||
|
||||
port@0:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: Input port of the channel.
|
||||
|
||||
port@1:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: Output port of the channel.
|
||||
|
||||
required:
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
- reg
|
||||
- phys
|
||||
- phy-names
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
- clocks
|
||||
- clock-names
|
||||
- power-domains
|
||||
- channel@0
|
||||
- channel@1
|
||||
|
||||
allOf:
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
const: fsl,imx8qm-ldb
|
||||
then:
|
||||
properties:
|
||||
fsl,companion-ldb: false
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
ldb {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,imx8qxp-ldb";
|
||||
clocks = <&clk IMX_SC_R_LVDS_0 IMX_SC_PM_CLK_MISC2>,
|
||||
<&clk IMX_SC_R_LVDS_0 IMX_SC_PM_CLK_BYPASS>;
|
||||
clock-names = "pixel", "bypass";
|
||||
power-domains = <&pd IMX_SC_R_LVDS_0>;
|
||||
|
||||
channel@0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0>;
|
||||
phys = <&mipi_lvds_0_phy>;
|
||||
phy-names = "lvds_phy";
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_ldb_ch0_mipi_lvds_0_pxl2dpi: endpoint {
|
||||
remote-endpoint = <&mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch0>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
channel@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
phys = <&mipi_lvds_0_phy>;
|
||||
phy-names = "lvds_phy";
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_ldb_ch1_mipi_lvds_0_pxl2dpi: endpoint {
|
||||
remote-endpoint = <&mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch1>;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
|
@ -0,0 +1,144 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/display/bridge/fsl,imx8qxp-pixel-combiner.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Freescale i.MX8qm/qxp Pixel Combiner
|
||||
|
||||
maintainers:
|
||||
- Liu Ying <victor.liu@nxp.com>
|
||||
|
||||
description: |
|
||||
The Freescale i.MX8qm/qxp Pixel Combiner takes two output streams from a
|
||||
single display controller and manipulates the two streams to support a number
|
||||
of modes(bypass, pixel combine, YUV444 to YUV422, split_RGB) configured as
|
||||
either one screen, two screens, or virtual screens. The pixel combiner is
|
||||
also responsible for generating some of the control signals for the pixel link
|
||||
output channel.
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- fsl,imx8qm-pixel-combiner
|
||||
- fsl,imx8qxp-pixel-combiner
|
||||
|
||||
"#address-cells":
|
||||
const: 1
|
||||
|
||||
"#size-cells":
|
||||
const: 0
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
||||
clock-names:
|
||||
const: apb
|
||||
|
||||
power-domains:
|
||||
maxItems: 1
|
||||
|
||||
patternProperties:
|
||||
"^channel@[0-1]$":
|
||||
type: object
|
||||
description: Represents a display stream of pixel combiner.
|
||||
|
||||
properties:
|
||||
"#address-cells":
|
||||
const: 1
|
||||
|
||||
"#size-cells":
|
||||
const: 0
|
||||
|
||||
reg:
|
||||
description: The display stream index.
|
||||
enum: [ 0, 1 ]
|
||||
|
||||
port@0:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: Input endpoint of the display stream.
|
||||
|
||||
port@1:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: Output endpoint of the display stream.
|
||||
|
||||
required:
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
- reg
|
||||
- port@0
|
||||
- port@1
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- "#address-cells"
|
||||
- "#size-cells"
|
||||
- reg
|
||||
- clocks
|
||||
- clock-names
|
||||
- power-domains
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/clock/imx8-lpcg.h>
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
pixel-combiner@56020000 {
|
||||
compatible = "fsl,imx8qxp-pixel-combiner";
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0x56020000 0x10000>;
|
||||
clocks = <&dc0_pixel_combiner_lpcg IMX_LPCG_CLK_4>;
|
||||
clock-names = "apb";
|
||||
power-domains = <&pd IMX_SC_R_DC_0>;
|
||||
|
||||
channel@0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0>;
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
dc0_pixel_combiner_ch0_dc0_dpu_disp0: endpoint {
|
||||
remote-endpoint = <&dc0_dpu_disp0_dc0_pixel_combiner_ch0>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <1>;
|
||||
|
||||
dc0_pixel_combiner_ch0_dc0_pixel_link0: endpoint {
|
||||
remote-endpoint = <&dc0_pixel_link0_dc0_pixel_combiner_ch0>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
channel@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
dc0_pixel_combiner_ch1_dc0_dpu_disp1: endpoint {
|
||||
remote-endpoint = <&dc0_dpu_disp1_dc0_pixel_combiner_ch1>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <1>;
|
||||
|
||||
dc0_pixel_combiner_ch1_dc0_pixel_link1: endpoint {
|
||||
remote-endpoint = <&dc0_pixel_link1_dc0_pixel_combiner_ch1>;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
|
@ -0,0 +1,144 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/display/bridge/fsl,imx8qxp-pixel-link.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Freescale i.MX8qm/qxp Display Pixel Link
|
||||
|
||||
maintainers:
|
||||
- Liu Ying <victor.liu@nxp.com>
|
||||
|
||||
description: |
|
||||
The Freescale i.MX8qm/qxp Display Pixel Link(DPL) forms a standard
|
||||
asynchronous linkage between pixel sources(display controller or
|
||||
camera module) and pixel consumers(imaging or displays).
|
||||
It consists of two distinct functions, a pixel transfer function and a
|
||||
control interface. Multiple pixel channels can exist per one control channel.
|
||||
This binding documentation is only for pixel links whose pixel sources are
|
||||
display controllers.
|
||||
|
||||
The i.MX8qm/qxp Display Pixel Link is accessed via System Controller Unit(SCU)
|
||||
firmware.
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- fsl,imx8qm-dc-pixel-link
|
||||
- fsl,imx8qxp-dc-pixel-link
|
||||
|
||||
fsl,dc-id:
|
||||
$ref: /schemas/types.yaml#/definitions/uint8
|
||||
description: |
|
||||
u8 value representing the display controller index that the pixel link
|
||||
connects to.
|
||||
|
||||
fsl,dc-stream-id:
|
||||
$ref: /schemas/types.yaml#/definitions/uint8
|
||||
description: |
|
||||
u8 value representing the display controller stream index that the pixel
|
||||
link connects to.
|
||||
enum: [0, 1]
|
||||
|
||||
ports:
|
||||
$ref: /schemas/graph.yaml#/properties/ports
|
||||
|
||||
properties:
|
||||
port@0:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: The pixel link input port node from upstream video source.
|
||||
|
||||
patternProperties:
|
||||
"^port@[1-4]$":
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: The pixel link output port node to downstream bridge.
|
||||
|
||||
required:
|
||||
- port@0
|
||||
- port@1
|
||||
- port@2
|
||||
- port@3
|
||||
- port@4
|
||||
|
||||
allOf:
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
const: fsl,imx8qxp-dc-pixel-link
|
||||
then:
|
||||
properties:
|
||||
fsl,dc-id:
|
||||
const: 0
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
const: fsl,imx8qm-dc-pixel-link
|
||||
then:
|
||||
properties:
|
||||
fsl,dc-id:
|
||||
enum: [0, 1]
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- fsl,dc-id
|
||||
- fsl,dc-stream-id
|
||||
- ports
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
dc0-pixel-link0 {
|
||||
compatible = "fsl,imx8qxp-dc-pixel-link";
|
||||
fsl,dc-id = /bits/ 8 <0>;
|
||||
fsl,dc-stream-id = /bits/ 8 <0>;
|
||||
|
||||
ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
/* from dc0 pixel combiner channel0 */
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
dc0_pixel_link0_dc0_pixel_combiner_ch0: endpoint {
|
||||
remote-endpoint = <&dc0_pixel_combiner_ch0_dc0_pixel_link0>;
|
||||
};
|
||||
};
|
||||
|
||||
/* to PXL2DPIs in MIPI/LVDS combo subsystems */
|
||||
port@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
|
||||
dc0_pixel_link0_mipi_lvds_0_pxl2dpi: endpoint@0 {
|
||||
reg = <0>;
|
||||
remote-endpoint = <&mipi_lvds_0_pxl2dpi_dc0_pixel_link0>;
|
||||
};
|
||||
|
||||
dc0_pixel_link0_mipi_lvds_1_pxl2dpi: endpoint@1 {
|
||||
reg = <1>;
|
||||
remote-endpoint = <&mipi_lvds_1_pxl2dpi_dc0_pixel_link0>;
|
||||
};
|
||||
};
|
||||
|
||||
/* unused */
|
||||
port@2 {
|
||||
reg = <2>;
|
||||
};
|
||||
|
||||
/* unused */
|
||||
port@3 {
|
||||
reg = <3>;
|
||||
};
|
||||
|
||||
/* to imaging subsystem */
|
||||
port@4 {
|
||||
reg = <4>;
|
||||
};
|
||||
};
|
||||
};
|
|
@ -0,0 +1,108 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/display/bridge/fsl,imx8qxp-pxl2dpi.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Freescale i.MX8qxp Pixel Link to Display Pixel Interface
|
||||
|
||||
maintainers:
|
||||
- Liu Ying <victor.liu@nxp.com>
|
||||
|
||||
description: |
|
||||
The Freescale i.MX8qxp Pixel Link to Display Pixel Interface(PXL2DPI)
|
||||
interfaces the pixel link 36-bit data output and the DSI controller’s
|
||||
MIPI-DPI 24-bit data input, and inputs of LVDS Display Bridge(LDB) module
|
||||
used in LVDS mode, to remap the pixel color codings between those modules.
|
||||
This module is purely combinatorial.
|
||||
|
||||
The i.MX8qxp PXL2DPI is controlled by Control and Status Registers(CSR) module.
|
||||
The CSR module, as a system controller, contains the PXL2DPI's configuration
|
||||
register.
|
||||
|
||||
properties:
|
||||
compatible:
|
||||
const: fsl,imx8qxp-pxl2dpi
|
||||
|
||||
fsl,sc-resource:
|
||||
$ref: /schemas/types.yaml#/definitions/uint32
|
||||
description: The SCU resource ID associated with this PXL2DPI instance.
|
||||
|
||||
power-domains:
|
||||
maxItems: 1
|
||||
|
||||
fsl,companion-pxl2dpi:
|
||||
$ref: /schemas/types.yaml#/definitions/phandle
|
||||
description: |
|
||||
A phandle which points to companion PXL2DPI which is used by downstream
|
||||
LVDS Display Bridge(LDB) in split mode.
|
||||
|
||||
ports:
|
||||
$ref: /schemas/graph.yaml#/properties/ports
|
||||
|
||||
properties:
|
||||
port@0:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: The PXL2DPI input port node from pixel link.
|
||||
|
||||
port@1:
|
||||
$ref: /schemas/graph.yaml#/properties/port
|
||||
description: The PXL2DPI output port node to downstream bridge.
|
||||
|
||||
required:
|
||||
- port@0
|
||||
- port@1
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- fsl,sc-resource
|
||||
- power-domains
|
||||
- ports
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
pxl2dpi {
|
||||
compatible = "fsl,imx8qxp-pxl2dpi";
|
||||
fsl,sc-resource = <IMX_SC_R_MIPI_0>;
|
||||
power-domains = <&pd IMX_SC_R_MIPI_0>;
|
||||
|
||||
ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
port@0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_pxl2dpi_dc_pixel_link0: endpoint@0 {
|
||||
reg = <0>;
|
||||
remote-endpoint = <&dc_pixel_link0_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
|
||||
mipi_lvds_0_pxl2dpi_dc_pixel_link1: endpoint@1 {
|
||||
reg = <1>;
|
||||
remote-endpoint = <&dc_pixel_link1_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
|
||||
mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch0: endpoint@0 {
|
||||
reg = <0>;
|
||||
remote-endpoint = <&mipi_lvds_0_ldb_ch0_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
|
||||
mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch1: endpoint@1 {
|
||||
reg = <1>;
|
||||
remote-endpoint = <&mipi_lvds_0_ldb_ch1_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
|
@ -0,0 +1,192 @@
|
|||
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
|
||||
%YAML 1.2
|
||||
---
|
||||
$id: http://devicetree.org/schemas/mfd/fsl,imx8qxp-csr.yaml#
|
||||
$schema: http://devicetree.org/meta-schemas/core.yaml#
|
||||
|
||||
title: Freescale i.MX8qm/qxp Control and Status Registers Module Bindings
|
||||
|
||||
maintainers:
|
||||
- Liu Ying <victor.liu@nxp.com>
|
||||
|
||||
description: |
|
||||
As a system controller, the Freescale i.MX8qm/qxp Control and Status
|
||||
Registers(CSR) module represents a set of miscellaneous registers of a
|
||||
specific subsystem. It may provide control and/or status report interfaces
|
||||
to a mix of standalone hardware devices within that subsystem. One typical
|
||||
use-case is for some other nodes to acquire a reference to the syscon node
|
||||
by phandle, and the other typical use-case is that the operating system
|
||||
should consider all subnodes of the CSR module as separate child devices.
|
||||
|
||||
properties:
|
||||
$nodename:
|
||||
pattern: "^syscon@[0-9a-f]+$"
|
||||
|
||||
compatible:
|
||||
items:
|
||||
- enum:
|
||||
- fsl,imx8qxp-mipi-lvds-csr
|
||||
- fsl,imx8qm-lvds-csr
|
||||
- const: syscon
|
||||
- const: simple-mfd
|
||||
|
||||
reg:
|
||||
maxItems: 1
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
||||
clock-names:
|
||||
const: ipg
|
||||
|
||||
patternProperties:
|
||||
"^(ldb|phy|pxl2dpi)$":
|
||||
type: object
|
||||
description: The possible child devices of the CSR module.
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- clocks
|
||||
- clock-names
|
||||
|
||||
allOf:
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
const: fsl,imx8qxp-mipi-lvds-csr
|
||||
then:
|
||||
required:
|
||||
- pxl2dpi
|
||||
- ldb
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
const: fsl,imx8qm-lvds-csr
|
||||
then:
|
||||
required:
|
||||
- phy
|
||||
- ldb
|
||||
|
||||
additionalProperties: false
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/clock/imx8-lpcg.h>
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
mipi_lvds_0_csr: syscon@56221000 {
|
||||
compatible = "fsl,imx8qxp-mipi-lvds-csr", "syscon", "simple-mfd";
|
||||
reg = <0x56221000 0x1000>;
|
||||
clocks = <&mipi_lvds_0_di_mipi_lvds_regs_lpcg IMX_LPCG_CLK_4>;
|
||||
clock-names = "ipg";
|
||||
|
||||
mipi_lvds_0_pxl2dpi: pxl2dpi {
|
||||
compatible = "fsl,imx8qxp-pxl2dpi";
|
||||
fsl,sc-resource = <IMX_SC_R_MIPI_0>;
|
||||
power-domains = <&pd IMX_SC_R_MIPI_0>;
|
||||
|
||||
ports {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
port@0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_pxl2dpi_dc0_pixel_link0: endpoint@0 {
|
||||
reg = <0>;
|
||||
remote-endpoint = <&dc0_pixel_link0_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
|
||||
mipi_lvds_0_pxl2dpi_dc0_pixel_link1: endpoint@1 {
|
||||
reg = <1>;
|
||||
remote-endpoint = <&dc0_pixel_link1_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
|
||||
mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch0: endpoint@0 {
|
||||
reg = <0>;
|
||||
remote-endpoint = <&mipi_lvds_0_ldb_ch0_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
|
||||
mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch1: endpoint@1 {
|
||||
reg = <1>;
|
||||
remote-endpoint = <&mipi_lvds_0_ldb_ch1_mipi_lvds_0_pxl2dpi>;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
mipi_lvds_0_ldb: ldb {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,imx8qxp-ldb";
|
||||
clocks = <&clk IMX_SC_R_LVDS_0 IMX_SC_PM_CLK_MISC2>,
|
||||
<&clk IMX_SC_R_LVDS_0 IMX_SC_PM_CLK_BYPASS>;
|
||||
clock-names = "pixel", "bypass";
|
||||
power-domains = <&pd IMX_SC_R_LVDS_0>;
|
||||
|
||||
channel@0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <0>;
|
||||
phys = <&mipi_lvds_0_phy>;
|
||||
phy-names = "lvds_phy";
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_ldb_ch0_mipi_lvds_0_pxl2dpi: endpoint {
|
||||
remote-endpoint = <&mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch0>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <1>;
|
||||
|
||||
/* ... */
|
||||
};
|
||||
};
|
||||
|
||||
channel@1 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
reg = <1>;
|
||||
phys = <&mipi_lvds_0_phy>;
|
||||
phy-names = "lvds_phy";
|
||||
|
||||
port@0 {
|
||||
reg = <0>;
|
||||
|
||||
mipi_lvds_0_ldb_ch1_mipi_lvds_0_pxl2dpi: endpoint {
|
||||
remote-endpoint = <&mipi_lvds_0_pxl2dpi_mipi_lvds_0_ldb_ch1>;
|
||||
};
|
||||
};
|
||||
|
||||
port@1 {
|
||||
reg = <1>;
|
||||
|
||||
/* ... */
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
mipi_lvds_0_phy: phy@56228300 {
|
||||
compatible = "fsl,imx8qxp-mipi-dphy";
|
||||
reg = <0x56228300 0x100>;
|
||||
clocks = <&clk IMX_SC_R_LVDS_0 IMX_SC_PM_CLK_PHY>;
|
||||
clock-names = "phy_ref";
|
||||
#phy-cells = <0>;
|
||||
fsl,syscon = <&mipi_lvds_0_csr>;
|
||||
power-domains = <&pd IMX_SC_R_MIPI_0>;
|
||||
};
|
|
@ -1492,6 +1492,80 @@ The following tables list existing packed RGB formats.
|
|||
- b\ :sub:`2`
|
||||
- b\ :sub:`1`
|
||||
- b\ :sub:`0`
|
||||
* .. _MEDIA-BUS-FMT-RGB666-1X30-CPADLO:
|
||||
|
||||
- MEDIA_BUS_FMT_RGB666_1X30-CPADLO
|
||||
- 0x101e
|
||||
-
|
||||
-
|
||||
-
|
||||
- r\ :sub:`5`
|
||||
- r\ :sub:`4`
|
||||
- r\ :sub:`3`
|
||||
- r\ :sub:`2`
|
||||
- r\ :sub:`1`
|
||||
- r\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- g\ :sub:`5`
|
||||
- g\ :sub:`4`
|
||||
- g\ :sub:`3`
|
||||
- g\ :sub:`2`
|
||||
- g\ :sub:`1`
|
||||
- g\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- b\ :sub:`5`
|
||||
- b\ :sub:`4`
|
||||
- b\ :sub:`3`
|
||||
- b\ :sub:`2`
|
||||
- b\ :sub:`1`
|
||||
- b\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
* .. _MEDIA-BUS-FMT-RGB888-1X30-CPADLO:
|
||||
|
||||
- MEDIA_BUS_FMT_RGB888_1X30-CPADLO
|
||||
- 0x101f
|
||||
-
|
||||
-
|
||||
-
|
||||
- r\ :sub:`7`
|
||||
- r\ :sub:`6`
|
||||
- r\ :sub:`5`
|
||||
- r\ :sub:`4`
|
||||
- r\ :sub:`3`
|
||||
- r\ :sub:`2`
|
||||
- r\ :sub:`1`
|
||||
- r\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- g\ :sub:`7`
|
||||
- g\ :sub:`6`
|
||||
- g\ :sub:`5`
|
||||
- g\ :sub:`4`
|
||||
- g\ :sub:`3`
|
||||
- g\ :sub:`2`
|
||||
- g\ :sub:`1`
|
||||
- g\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- b\ :sub:`7`
|
||||
- b\ :sub:`6`
|
||||
- b\ :sub:`5`
|
||||
- b\ :sub:`4`
|
||||
- b\ :sub:`3`
|
||||
- b\ :sub:`2`
|
||||
- b\ :sub:`1`
|
||||
- b\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
* .. _MEDIA-BUS-FMT-ARGB888-1X32:
|
||||
|
||||
- MEDIA_BUS_FMT_ARGB888_1X32
|
||||
|
@ -1669,6 +1743,88 @@ The following table list existing packed 36bit wide RGB formats.
|
|||
- 2
|
||||
- 1
|
||||
- 0
|
||||
* .. _MEDIA-BUS-FMT-RGB666-1X36-CPADLO:
|
||||
|
||||
- MEDIA_BUS_FMT_RGB666_1X36_CPADLO
|
||||
- 0x1020
|
||||
-
|
||||
- r\ :sub:`5`
|
||||
- r\ :sub:`4`
|
||||
- r\ :sub:`3`
|
||||
- r\ :sub:`2`
|
||||
- r\ :sub:`1`
|
||||
- r\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- g\ :sub:`5`
|
||||
- g\ :sub:`4`
|
||||
- g\ :sub:`3`
|
||||
- g\ :sub:`2`
|
||||
- g\ :sub:`1`
|
||||
- g\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- b\ :sub:`5`
|
||||
- b\ :sub:`4`
|
||||
- b\ :sub:`3`
|
||||
- b\ :sub:`2`
|
||||
- b\ :sub:`1`
|
||||
- b\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
* .. _MEDIA-BUS-FMT-RGB888-1X36-CPADLO:
|
||||
|
||||
- MEDIA_BUS_FMT_RGB888_1X36_CPADLO
|
||||
- 0x1021
|
||||
-
|
||||
- r\ :sub:`7`
|
||||
- r\ :sub:`6`
|
||||
- r\ :sub:`5`
|
||||
- r\ :sub:`4`
|
||||
- r\ :sub:`3`
|
||||
- r\ :sub:`2`
|
||||
- r\ :sub:`1`
|
||||
- r\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- g\ :sub:`7`
|
||||
- g\ :sub:`6`
|
||||
- g\ :sub:`5`
|
||||
- g\ :sub:`4`
|
||||
- g\ :sub:`3`
|
||||
- g\ :sub:`2`
|
||||
- g\ :sub:`1`
|
||||
- g\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- b\ :sub:`7`
|
||||
- b\ :sub:`6`
|
||||
- b\ :sub:`5`
|
||||
- b\ :sub:`4`
|
||||
- b\ :sub:`3`
|
||||
- b\ :sub:`2`
|
||||
- b\ :sub:`1`
|
||||
- b\ :sub:`0`
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
- 0
|
||||
* .. _MEDIA-BUS-FMT-RGB121212-1X36:
|
||||
|
||||
- MEDIA_BUS_FMT_RGB121212_1X36
|
||||
|
|
10
MAINTAINERS
10
MAINTAINERS
|
@ -6674,6 +6674,16 @@ F: Documentation/devicetree/bindings/display/imx/
|
|||
F: drivers/gpu/drm/imx/
|
||||
F: drivers/gpu/ipu-v3/
|
||||
|
||||
DRM DRIVERS FOR FREESCALE IMX BRIDGE
|
||||
M: Liu Ying <victor.liu@nxp.com>
|
||||
L: dri-devel@lists.freedesktop.org
|
||||
S: Maintained
|
||||
F: Documentation/devicetree/bindings/display/bridge/fsl,imx8qxp-ldb.yaml
|
||||
F: Documentation/devicetree/bindings/display/bridge/fsl,imx8qxp-pixel-combiner.yaml
|
||||
F: Documentation/devicetree/bindings/display/bridge/fsl,imx8qxp-pixel-link.yaml
|
||||
F: Documentation/devicetree/bindings/display/bridge/fsl,imx8qxp-pxl2dpi.yaml
|
||||
F: drivers/gpu/drm/bridge/imx/
|
||||
|
||||
DRM DRIVERS FOR GMA500 (Poulsbo, Moorestown and derivative chipsets)
|
||||
M: Patrik Jakobsson <patrik.r.jakobsson@gmail.com>
|
||||
L: dri-devel@lists.freedesktop.org
|
||||
|
|
|
@ -37,6 +37,7 @@
|
|||
#include <drm/drm_fixed.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <linux/i2c.h>
|
||||
|
|
|
@ -78,6 +78,7 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
|
@ -9138,7 +9139,6 @@ static void amdgpu_dm_commit_planes(struct drm_atomic_state *state,
|
|||
to_dm_crtc_state(drm_atomic_get_old_crtc_state(state, pcrtc));
|
||||
int planes_count = 0, vpos, hpos;
|
||||
unsigned long flags;
|
||||
struct amdgpu_bo *abo;
|
||||
uint32_t target_vblank, last_flip_vblank;
|
||||
bool vrr_active = amdgpu_dm_vrr_active(acrtc_state);
|
||||
bool pflip_present = false;
|
||||
|
@ -9210,7 +9210,6 @@ static void amdgpu_dm_commit_planes(struct drm_atomic_state *state,
|
|||
continue;
|
||||
}
|
||||
|
||||
abo = gem_to_amdgpu_bo(fb->obj[0]);
|
||||
fill_dc_plane_info_and_addr(
|
||||
dm->adev, new_plane_state,
|
||||
afb->tiling_flags,
|
||||
|
|
|
@ -34,6 +34,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_plane.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_atomic.h>
|
||||
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
*
|
||||
*/
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include "d71_dev.h"
|
||||
#include "malidp_io.h"
|
||||
|
|
|
@ -10,6 +10,7 @@
|
|||
#include <linux/list.h>
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_writeback.h>
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
*/
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include "komeda_dev.h"
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
* Author: James.Qian.Wang <james.qian.wang@arm.com>
|
||||
*
|
||||
*/
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include "komeda_dev.h"
|
||||
#include "komeda_kms.h"
|
||||
|
||||
|
|
|
@ -20,6 +20,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
|
|
|
@ -14,6 +14,7 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
|
|
@ -9,8 +9,10 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/drm_writeback.h>
|
||||
|
|
|
@ -11,9 +11,11 @@
|
|||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
|
|
|
@ -5,6 +5,8 @@
|
|||
#ifndef ARMADA_FB_H
|
||||
#define ARMADA_FB_H
|
||||
|
||||
#include <drm/drm_framebuffer.h>
|
||||
|
||||
struct armada_framebuffer {
|
||||
struct drm_framebuffer fb;
|
||||
uint8_t fmt;
|
||||
|
|
|
@ -9,6 +9,7 @@
|
|||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_atomic_helper.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_panel.h>
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
||||
#include "aspeed_gfx.h"
|
||||
|
|
|
@ -36,6 +36,7 @@
|
|||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_gem_atomic_helper.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
|
|
|
@ -11,8 +11,10 @@
|
|||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
|
||||
|
|
|
@ -385,6 +385,8 @@ source "drivers/gpu/drm/bridge/adv7511/Kconfig"
|
|||
|
||||
source "drivers/gpu/drm/bridge/cadence/Kconfig"
|
||||
|
||||
source "drivers/gpu/drm/bridge/imx/Kconfig"
|
||||
|
||||
source "drivers/gpu/drm/bridge/synopsys/Kconfig"
|
||||
|
||||
endmenu
|
||||
|
|
|
@ -36,4 +36,5 @@ obj-$(CONFIG_DRM_ITE_IT66121) += ite-it66121.o
|
|||
|
||||
obj-y += analogix/
|
||||
obj-y += cadence/
|
||||
obj-y += imx/
|
||||
obj-y += synopsys/
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_panel.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
|
|
@ -1623,14 +1623,14 @@ static int anx7625_parse_dt(struct device *dev,
|
|||
|
||||
anx7625_get_swing_setting(dev, pdata);
|
||||
|
||||
pdata->is_dpi = 1; /* default dpi mode */
|
||||
pdata->is_dpi = 0; /* default dsi mode */
|
||||
pdata->mipi_host_node = of_graph_get_remote_node(np, 0, 0);
|
||||
if (!pdata->mipi_host_node) {
|
||||
DRM_DEV_ERROR(dev, "fail to get internal panel.\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
bus_type = V4L2_FWNODE_BUS_TYPE_PARALLEL;
|
||||
bus_type = 0;
|
||||
mipi_lanes = MAX_LANES_SUPPORT;
|
||||
ep0 = of_graph_get_endpoint_by_regs(np, 0, 0);
|
||||
if (ep0) {
|
||||
|
@ -1641,8 +1641,8 @@ static int anx7625_parse_dt(struct device *dev,
|
|||
of_node_put(ep0);
|
||||
}
|
||||
|
||||
if (bus_type == V4L2_FWNODE_BUS_TYPE_PARALLEL) /* bus type is Parallel(DSI) */
|
||||
pdata->is_dpi = 0;
|
||||
if (bus_type == V4L2_FWNODE_BUS_TYPE_DPI) /* bus type is DPI */
|
||||
pdata->is_dpi = 1;
|
||||
|
||||
pdata->mipi_lanes = MAX_LANES_SUPPORT;
|
||||
if (mipi_lanes > 0)
|
||||
|
|
|
@ -43,6 +43,7 @@
|
|||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_modeset_helper_vtables.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
|
|
@ -0,0 +1,43 @@
|
|||
config DRM_IMX8QM_LDB
|
||||
tristate "Freescale i.MX8QM LVDS display bridge"
|
||||
depends on OF
|
||||
depends on COMMON_CLK
|
||||
select DRM_KMS_HELPER
|
||||
help
|
||||
Choose this to enable the internal LVDS Display Bridge(LDB) found in
|
||||
Freescale i.MX8qm processor. Official name of LDB is pixel mapper.
|
||||
|
||||
config DRM_IMX8QXP_LDB
|
||||
tristate "Freescale i.MX8QXP LVDS display bridge"
|
||||
depends on OF
|
||||
depends on COMMON_CLK
|
||||
select DRM_KMS_HELPER
|
||||
help
|
||||
Choose this to enable the internal LVDS Display Bridge(LDB) found in
|
||||
Freescale i.MX8qxp processor. Official name of LDB is pixel mapper.
|
||||
|
||||
config DRM_IMX8QXP_PIXEL_COMBINER
|
||||
tristate "Freescale i.MX8QM/QXP pixel combiner"
|
||||
depends on OF
|
||||
depends on COMMON_CLK
|
||||
select DRM_KMS_HELPER
|
||||
help
|
||||
Choose this to enable pixel combiner found in
|
||||
Freescale i.MX8qm/qxp processors.
|
||||
|
||||
config DRM_IMX8QXP_PIXEL_LINK
|
||||
tristate "Freescale i.MX8QM/QXP display pixel link"
|
||||
depends on OF
|
||||
depends on IMX_SCU
|
||||
select DRM_KMS_HELPER
|
||||
help
|
||||
Choose this to enable display pixel link found in
|
||||
Freescale i.MX8qm/qxp processors.
|
||||
|
||||
config DRM_IMX8QXP_PIXEL_LINK_TO_DPI
|
||||
tristate "Freescale i.MX8QXP pixel link to display pixel interface"
|
||||
depends on OF
|
||||
select DRM_KMS_HELPER
|
||||
help
|
||||
Choose this to enable pixel link to display pixel interface(PXL2DPI)
|
||||
found in Freescale i.MX8qxp processor.
|
|
@ -0,0 +1,9 @@
|
|||
imx8qm-ldb-objs := imx-ldb-helper.o imx8qm-ldb-drv.o
|
||||
obj-$(CONFIG_DRM_IMX8QM_LDB) += imx8qm-ldb.o
|
||||
|
||||
imx8qxp-ldb-objs := imx-ldb-helper.o imx8qxp-ldb-drv.o
|
||||
obj-$(CONFIG_DRM_IMX8QXP_LDB) += imx8qxp-ldb.o
|
||||
|
||||
obj-$(CONFIG_DRM_IMX8QXP_PIXEL_COMBINER) += imx8qxp-pixel-combiner.o
|
||||
obj-$(CONFIG_DRM_IMX8QXP_PIXEL_LINK) += imx8qxp-pixel-link.o
|
||||
obj-$(CONFIG_DRM_IMX8QXP_PIXEL_LINK_TO_DPI) += imx8qxp-pxl2dpi.o
|
|
@ -0,0 +1,220 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Copyright (C) 2012 Sascha Hauer, Pengutronix
|
||||
* Copyright 2019,2020,2022 NXP
|
||||
*/
|
||||
|
||||
#include <linux/mfd/syscon.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include "imx-ldb-helper.h"
|
||||
|
||||
bool ldb_channel_is_single_link(struct ldb_channel *ldb_ch)
|
||||
{
|
||||
return ldb_ch->link_type == LDB_CH_SINGLE_LINK;
|
||||
}
|
||||
|
||||
bool ldb_channel_is_split_link(struct ldb_channel *ldb_ch)
|
||||
{
|
||||
return ldb_ch->link_type == LDB_CH_DUAL_LINK_EVEN_ODD_PIXELS ||
|
||||
ldb_ch->link_type == LDB_CH_DUAL_LINK_ODD_EVEN_PIXELS;
|
||||
}
|
||||
|
||||
int ldb_bridge_atomic_check_helper(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
|
||||
ldb_ch->in_bus_format = bridge_state->input_bus_cfg.format;
|
||||
ldb_ch->out_bus_format = bridge_state->output_bus_cfg.format;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void ldb_bridge_mode_set_helper(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
|
||||
if (is_split)
|
||||
ldb->ldb_ctrl |= LDB_SPLIT_MODE_EN;
|
||||
|
||||
switch (ldb_ch->out_bus_format) {
|
||||
case MEDIA_BUS_FMT_RGB666_1X7X3_SPWG:
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_SPWG:
|
||||
if (ldb_ch->chno == 0 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH0_24;
|
||||
if (ldb_ch->chno == 1 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH1_24;
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA:
|
||||
if (ldb_ch->chno == 0 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH0_24 |
|
||||
LDB_BIT_MAP_CH0_JEIDA;
|
||||
if (ldb_ch->chno == 1 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DATA_WIDTH_CH1_24 |
|
||||
LDB_BIT_MAP_CH1_JEIDA;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void ldb_bridge_enable_helper(struct drm_bridge *bridge)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
|
||||
/*
|
||||
* Platform specific bridge drivers should set ldb_ctrl properly
|
||||
* for the enablement, so just write the ctrl_reg here.
|
||||
*/
|
||||
regmap_write(ldb->regmap, ldb->ctrl_reg, ldb->ldb_ctrl);
|
||||
}
|
||||
|
||||
void ldb_bridge_disable_helper(struct drm_bridge *bridge)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
|
||||
if (ldb_ch->chno == 0 || is_split)
|
||||
ldb->ldb_ctrl &= ~LDB_CH0_MODE_EN_MASK;
|
||||
if (ldb_ch->chno == 1 || is_split)
|
||||
ldb->ldb_ctrl &= ~LDB_CH1_MODE_EN_MASK;
|
||||
|
||||
regmap_write(ldb->regmap, ldb->ctrl_reg, ldb->ldb_ctrl);
|
||||
}
|
||||
|
||||
int ldb_bridge_attach_helper(struct drm_bridge *bridge,
|
||||
enum drm_bridge_attach_flags flags)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
|
||||
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
|
||||
DRM_DEV_ERROR(ldb->dev,
|
||||
"do not support creating a drm_connector\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (!bridge->encoder) {
|
||||
DRM_DEV_ERROR(ldb->dev, "missing encoder\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
return drm_bridge_attach(bridge->encoder,
|
||||
ldb_ch->next_bridge, bridge,
|
||||
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
||||
}
|
||||
|
||||
int ldb_init_helper(struct ldb *ldb)
|
||||
{
|
||||
struct device *dev = ldb->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
struct device_node *child;
|
||||
int ret;
|
||||
u32 i;
|
||||
|
||||
ldb->regmap = syscon_node_to_regmap(np->parent);
|
||||
if (IS_ERR(ldb->regmap)) {
|
||||
ret = PTR_ERR(ldb->regmap);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get regmap: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
for_each_available_child_of_node(np, child) {
|
||||
struct ldb_channel *ldb_ch;
|
||||
|
||||
ret = of_property_read_u32(child, "reg", &i);
|
||||
if (ret || i > MAX_LDB_CHAN_NUM - 1) {
|
||||
ret = -EINVAL;
|
||||
DRM_DEV_ERROR(dev,
|
||||
"invalid channel node address: %u\n", i);
|
||||
of_node_put(child);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ldb_ch = ldb->channel[i];
|
||||
ldb_ch->ldb = ldb;
|
||||
ldb_ch->chno = i;
|
||||
ldb_ch->is_available = true;
|
||||
ldb_ch->np = child;
|
||||
|
||||
ldb->available_ch_cnt++;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int ldb_find_next_bridge_helper(struct ldb *ldb)
|
||||
{
|
||||
struct device *dev = ldb->dev;
|
||||
struct ldb_channel *ldb_ch;
|
||||
int ret, i;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
ldb_ch = ldb->channel[i];
|
||||
|
||||
if (!ldb_ch->is_available)
|
||||
continue;
|
||||
|
||||
ldb_ch->next_bridge = devm_drm_of_get_bridge(dev, ldb_ch->np,
|
||||
1, 0);
|
||||
if (IS_ERR(ldb_ch->next_bridge)) {
|
||||
ret = PTR_ERR(ldb_ch->next_bridge);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get next bridge: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void ldb_add_bridge_helper(struct ldb *ldb,
|
||||
const struct drm_bridge_funcs *bridge_funcs)
|
||||
{
|
||||
struct ldb_channel *ldb_ch;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
ldb_ch = ldb->channel[i];
|
||||
|
||||
if (!ldb_ch->is_available)
|
||||
continue;
|
||||
|
||||
ldb_ch->bridge.driver_private = ldb_ch;
|
||||
ldb_ch->bridge.funcs = bridge_funcs;
|
||||
ldb_ch->bridge.of_node = ldb_ch->np;
|
||||
|
||||
drm_bridge_add(&ldb_ch->bridge);
|
||||
}
|
||||
}
|
||||
|
||||
void ldb_remove_bridge_helper(struct ldb *ldb)
|
||||
{
|
||||
struct ldb_channel *ldb_ch;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
ldb_ch = ldb->channel[i];
|
||||
|
||||
if (!ldb_ch->is_available)
|
||||
continue;
|
||||
|
||||
drm_bridge_remove(&ldb_ch->bridge);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,96 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0+ */
|
||||
|
||||
/*
|
||||
* Copyright 2019,2020,2022 NXP
|
||||
*/
|
||||
|
||||
#ifndef __IMX_LDB_HELPER__
|
||||
#define __IMX_LDB_HELPER__
|
||||
|
||||
#include <linux/device.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_modeset_helper_vtables.h>
|
||||
|
||||
#define LDB_CH0_MODE_EN_TO_DI0 BIT(0)
|
||||
#define LDB_CH0_MODE_EN_TO_DI1 (3 << 0)
|
||||
#define LDB_CH0_MODE_EN_MASK (3 << 0)
|
||||
#define LDB_CH1_MODE_EN_TO_DI0 BIT(2)
|
||||
#define LDB_CH1_MODE_EN_TO_DI1 (3 << 2)
|
||||
#define LDB_CH1_MODE_EN_MASK (3 << 2)
|
||||
#define LDB_SPLIT_MODE_EN BIT(4)
|
||||
#define LDB_DATA_WIDTH_CH0_24 BIT(5)
|
||||
#define LDB_BIT_MAP_CH0_JEIDA BIT(6)
|
||||
#define LDB_DATA_WIDTH_CH1_24 BIT(7)
|
||||
#define LDB_BIT_MAP_CH1_JEIDA BIT(8)
|
||||
#define LDB_DI0_VS_POL_ACT_LOW BIT(9)
|
||||
#define LDB_DI1_VS_POL_ACT_LOW BIT(10)
|
||||
|
||||
#define MAX_LDB_CHAN_NUM 2
|
||||
|
||||
enum ldb_channel_link_type {
|
||||
LDB_CH_SINGLE_LINK,
|
||||
LDB_CH_DUAL_LINK_EVEN_ODD_PIXELS,
|
||||
LDB_CH_DUAL_LINK_ODD_EVEN_PIXELS,
|
||||
};
|
||||
|
||||
struct ldb;
|
||||
|
||||
struct ldb_channel {
|
||||
struct ldb *ldb;
|
||||
struct drm_bridge bridge;
|
||||
struct drm_bridge *next_bridge;
|
||||
struct device_node *np;
|
||||
u32 chno;
|
||||
bool is_available;
|
||||
u32 in_bus_format;
|
||||
u32 out_bus_format;
|
||||
enum ldb_channel_link_type link_type;
|
||||
};
|
||||
|
||||
struct ldb {
|
||||
struct regmap *regmap;
|
||||
struct device *dev;
|
||||
struct ldb_channel *channel[MAX_LDB_CHAN_NUM];
|
||||
unsigned int ctrl_reg;
|
||||
u32 ldb_ctrl;
|
||||
unsigned int available_ch_cnt;
|
||||
};
|
||||
|
||||
#define bridge_to_ldb_ch(b) container_of(b, struct ldb_channel, bridge)
|
||||
|
||||
bool ldb_channel_is_single_link(struct ldb_channel *ldb_ch);
|
||||
bool ldb_channel_is_split_link(struct ldb_channel *ldb_ch);
|
||||
|
||||
int ldb_bridge_atomic_check_helper(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state);
|
||||
|
||||
void ldb_bridge_mode_set_helper(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode);
|
||||
|
||||
void ldb_bridge_enable_helper(struct drm_bridge *bridge);
|
||||
|
||||
void ldb_bridge_disable_helper(struct drm_bridge *bridge);
|
||||
|
||||
int ldb_bridge_attach_helper(struct drm_bridge *bridge,
|
||||
enum drm_bridge_attach_flags flags);
|
||||
|
||||
int ldb_init_helper(struct ldb *ldb);
|
||||
|
||||
int ldb_find_next_bridge_helper(struct ldb *ldb);
|
||||
|
||||
void ldb_add_bridge_helper(struct ldb *ldb,
|
||||
const struct drm_bridge_funcs *bridge_funcs);
|
||||
|
||||
void ldb_remove_bridge_helper(struct ldb *ldb);
|
||||
|
||||
#endif /* __IMX_LDB_HELPER__ */
|
|
@ -0,0 +1,587 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
/*
|
||||
* Copyright 2020 NXP
|
||||
*/
|
||||
|
||||
#include <linux/clk.h>
|
||||
#include <linux/mfd/syscon.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of_graph.h>
|
||||
#include <linux/phy/phy.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include "imx-ldb-helper.h"
|
||||
|
||||
#define LDB_CH0_10BIT_EN BIT(22)
|
||||
#define LDB_CH1_10BIT_EN BIT(23)
|
||||
#define LDB_CH0_DATA_WIDTH_24BIT BIT(24)
|
||||
#define LDB_CH1_DATA_WIDTH_24BIT BIT(26)
|
||||
#define LDB_CH0_DATA_WIDTH_30BIT (2 << 24)
|
||||
#define LDB_CH1_DATA_WIDTH_30BIT (2 << 26)
|
||||
|
||||
#define SS_CTRL 0x20
|
||||
#define CH_HSYNC_M(id) BIT(0 + ((id) * 2))
|
||||
#define CH_VSYNC_M(id) BIT(1 + ((id) * 2))
|
||||
#define CH_PHSYNC(id) BIT(0 + ((id) * 2))
|
||||
#define CH_PVSYNC(id) BIT(1 + ((id) * 2))
|
||||
|
||||
#define DRIVER_NAME "imx8qm-ldb"
|
||||
|
||||
struct imx8qm_ldb_channel {
|
||||
struct ldb_channel base;
|
||||
struct phy *phy;
|
||||
};
|
||||
|
||||
struct imx8qm_ldb {
|
||||
struct ldb base;
|
||||
struct device *dev;
|
||||
struct imx8qm_ldb_channel channel[MAX_LDB_CHAN_NUM];
|
||||
struct clk *clk_pixel;
|
||||
struct clk *clk_bypass;
|
||||
int active_chno;
|
||||
};
|
||||
|
||||
static inline struct imx8qm_ldb_channel *
|
||||
base_to_imx8qm_ldb_channel(struct ldb_channel *base)
|
||||
{
|
||||
return container_of(base, struct imx8qm_ldb_channel, base);
|
||||
}
|
||||
|
||||
static inline struct imx8qm_ldb *base_to_imx8qm_ldb(struct ldb *base)
|
||||
{
|
||||
return container_of(base, struct imx8qm_ldb, base);
|
||||
}
|
||||
|
||||
static void imx8qm_ldb_set_phy_cfg(struct imx8qm_ldb *imx8qm_ldb,
|
||||
unsigned long di_clk,
|
||||
bool is_split, bool is_slave,
|
||||
struct phy_configure_opts_lvds *phy_cfg)
|
||||
{
|
||||
phy_cfg->bits_per_lane_and_dclk_cycle = 7;
|
||||
phy_cfg->lanes = 4;
|
||||
phy_cfg->differential_clk_rate = is_split ? di_clk / 2 : di_clk;
|
||||
phy_cfg->is_slave = is_slave;
|
||||
}
|
||||
|
||||
static int imx8qm_ldb_bridge_atomic_check(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch =
|
||||
base_to_imx8qm_ldb_channel(ldb_ch);
|
||||
struct imx8qm_ldb *imx8qm_ldb = base_to_imx8qm_ldb(ldb);
|
||||
struct drm_display_mode *adj = &crtc_state->adjusted_mode;
|
||||
unsigned long di_clk = adj->clock * 1000;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
union phy_configure_opts opts = { };
|
||||
struct phy_configure_opts_lvds *phy_cfg = &opts.lvds;
|
||||
int ret;
|
||||
|
||||
ret = ldb_bridge_atomic_check_helper(bridge, bridge_state,
|
||||
crtc_state, conn_state);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
imx8qm_ldb_set_phy_cfg(imx8qm_ldb, di_clk, is_split, false, phy_cfg);
|
||||
ret = phy_validate(imx8qm_ldb_ch->phy, PHY_MODE_LVDS, 0, &opts);
|
||||
if (ret < 0) {
|
||||
DRM_DEV_DEBUG_DRIVER(imx8qm_ldb->dev,
|
||||
"failed to validate PHY: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (is_split) {
|
||||
imx8qm_ldb_ch =
|
||||
&imx8qm_ldb->channel[imx8qm_ldb->active_chno ^ 1];
|
||||
imx8qm_ldb_set_phy_cfg(imx8qm_ldb, di_clk, is_split, true,
|
||||
phy_cfg);
|
||||
ret = phy_validate(imx8qm_ldb_ch->phy, PHY_MODE_LVDS, 0, &opts);
|
||||
if (ret < 0) {
|
||||
DRM_DEV_DEBUG_DRIVER(imx8qm_ldb->dev,
|
||||
"failed to validate slave PHY: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qm_ldb_bridge_mode_set(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch =
|
||||
base_to_imx8qm_ldb_channel(ldb_ch);
|
||||
struct imx8qm_ldb *imx8qm_ldb = base_to_imx8qm_ldb(ldb);
|
||||
struct device *dev = imx8qm_ldb->dev;
|
||||
unsigned long di_clk = adjusted_mode->clock * 1000;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
union phy_configure_opts opts = { };
|
||||
struct phy_configure_opts_lvds *phy_cfg = &opts.lvds;
|
||||
u32 chno = ldb_ch->chno;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to get runtime PM sync: %d\n", ret);
|
||||
|
||||
ret = phy_init(imx8qm_ldb_ch->phy);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to initialize PHY: %d\n", ret);
|
||||
|
||||
clk_set_rate(imx8qm_ldb->clk_bypass, di_clk);
|
||||
clk_set_rate(imx8qm_ldb->clk_pixel, di_clk);
|
||||
|
||||
imx8qm_ldb_set_phy_cfg(imx8qm_ldb, di_clk, is_split, false, phy_cfg);
|
||||
ret = phy_configure(imx8qm_ldb_ch->phy, &opts);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to configure PHY: %d\n", ret);
|
||||
|
||||
if (is_split) {
|
||||
imx8qm_ldb_ch =
|
||||
&imx8qm_ldb->channel[imx8qm_ldb->active_chno ^ 1];
|
||||
imx8qm_ldb_set_phy_cfg(imx8qm_ldb, di_clk, is_split, true,
|
||||
phy_cfg);
|
||||
ret = phy_configure(imx8qm_ldb_ch->phy, &opts);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to configure slave PHY: %d\n",
|
||||
ret);
|
||||
}
|
||||
|
||||
/* input VSYNC signal from pixel link is active low */
|
||||
if (ldb_ch->chno == 0 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DI0_VS_POL_ACT_LOW;
|
||||
if (ldb_ch->chno == 1 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_DI1_VS_POL_ACT_LOW;
|
||||
|
||||
switch (ldb_ch->out_bus_format) {
|
||||
case MEDIA_BUS_FMT_RGB666_1X7X3_SPWG:
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA:
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_SPWG:
|
||||
if (ldb_ch->chno == 0 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_CH0_DATA_WIDTH_24BIT;
|
||||
if (ldb_ch->chno == 1 || is_split)
|
||||
ldb->ldb_ctrl |= LDB_CH1_DATA_WIDTH_24BIT;
|
||||
break;
|
||||
}
|
||||
|
||||
ldb_bridge_mode_set_helper(bridge, mode, adjusted_mode);
|
||||
|
||||
if (adjusted_mode->flags & DRM_MODE_FLAG_NVSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL, CH_VSYNC_M(chno), 0);
|
||||
else if (adjusted_mode->flags & DRM_MODE_FLAG_PVSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL,
|
||||
CH_VSYNC_M(chno), CH_PVSYNC(chno));
|
||||
|
||||
if (adjusted_mode->flags & DRM_MODE_FLAG_NHSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL, CH_HSYNC_M(chno), 0);
|
||||
else if (adjusted_mode->flags & DRM_MODE_FLAG_PHSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL,
|
||||
CH_HSYNC_M(chno), CH_PHSYNC(chno));
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qm_ldb_bridge_atomic_enable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch =
|
||||
base_to_imx8qm_ldb_channel(ldb_ch);
|
||||
struct imx8qm_ldb *imx8qm_ldb = base_to_imx8qm_ldb(ldb);
|
||||
struct device *dev = imx8qm_ldb->dev;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
int ret;
|
||||
|
||||
clk_prepare_enable(imx8qm_ldb->clk_pixel);
|
||||
clk_prepare_enable(imx8qm_ldb->clk_bypass);
|
||||
|
||||
/* both DI0 and DI1 connect with pixel link, so ok to use DI0 only */
|
||||
if (ldb_ch->chno == 0 || is_split) {
|
||||
ldb->ldb_ctrl &= ~LDB_CH0_MODE_EN_MASK;
|
||||
ldb->ldb_ctrl |= LDB_CH0_MODE_EN_TO_DI0;
|
||||
}
|
||||
if (ldb_ch->chno == 1 || is_split) {
|
||||
ldb->ldb_ctrl &= ~LDB_CH1_MODE_EN_MASK;
|
||||
ldb->ldb_ctrl |= LDB_CH1_MODE_EN_TO_DI0;
|
||||
}
|
||||
|
||||
if (is_split) {
|
||||
ret = phy_power_on(imx8qm_ldb->channel[0].phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to power on channel0 PHY: %d\n",
|
||||
ret);
|
||||
|
||||
ret = phy_power_on(imx8qm_ldb->channel[1].phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to power on channel1 PHY: %d\n",
|
||||
ret);
|
||||
} else {
|
||||
ret = phy_power_on(imx8qm_ldb_ch->phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev, "failed to power on PHY: %d\n", ret);
|
||||
}
|
||||
|
||||
ldb_bridge_enable_helper(bridge);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qm_ldb_bridge_atomic_disable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch =
|
||||
base_to_imx8qm_ldb_channel(ldb_ch);
|
||||
struct imx8qm_ldb *imx8qm_ldb = base_to_imx8qm_ldb(ldb);
|
||||
struct device *dev = imx8qm_ldb->dev;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
int ret;
|
||||
|
||||
ldb_bridge_disable_helper(bridge);
|
||||
|
||||
if (is_split) {
|
||||
ret = phy_power_off(imx8qm_ldb->channel[0].phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to power off channel0 PHY: %d\n",
|
||||
ret);
|
||||
ret = phy_power_off(imx8qm_ldb->channel[1].phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to power off channel1 PHY: %d\n",
|
||||
ret);
|
||||
} else {
|
||||
ret = phy_power_off(imx8qm_ldb_ch->phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev, "failed to power off PHY: %d\n", ret);
|
||||
}
|
||||
|
||||
clk_disable_unprepare(imx8qm_ldb->clk_bypass);
|
||||
clk_disable_unprepare(imx8qm_ldb->clk_pixel);
|
||||
|
||||
ret = pm_runtime_put(dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to put runtime PM: %d\n", ret);
|
||||
}
|
||||
|
||||
static const u32 imx8qm_ldb_bus_output_fmts[] = {
|
||||
MEDIA_BUS_FMT_RGB666_1X7X3_SPWG,
|
||||
MEDIA_BUS_FMT_RGB888_1X7X4_SPWG,
|
||||
MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA,
|
||||
MEDIA_BUS_FMT_FIXED,
|
||||
};
|
||||
|
||||
static bool imx8qm_ldb_bus_output_fmt_supported(u32 fmt)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(imx8qm_ldb_bus_output_fmts); i++) {
|
||||
if (imx8qm_ldb_bus_output_fmts[i] == fmt)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qm_ldb_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
u32 output_fmt,
|
||||
unsigned int *num_input_fmts)
|
||||
{
|
||||
struct drm_display_info *di;
|
||||
const struct drm_format_info *finfo;
|
||||
u32 *input_fmts;
|
||||
|
||||
if (!imx8qm_ldb_bus_output_fmt_supported(output_fmt))
|
||||
return NULL;
|
||||
|
||||
*num_input_fmts = 1;
|
||||
|
||||
input_fmts = kmalloc(sizeof(*input_fmts), GFP_KERNEL);
|
||||
if (!input_fmts)
|
||||
return NULL;
|
||||
|
||||
switch (output_fmt) {
|
||||
case MEDIA_BUS_FMT_FIXED:
|
||||
di = &conn_state->connector->display_info;
|
||||
|
||||
/*
|
||||
* Look at the first bus format to determine input format.
|
||||
* Default to MEDIA_BUS_FMT_RGB888_1X36_CPADLO, if no match.
|
||||
*/
|
||||
if (di->num_bus_formats) {
|
||||
finfo = drm_format_info(di->bus_formats[0]);
|
||||
|
||||
input_fmts[0] = finfo->depth == 18 ?
|
||||
MEDIA_BUS_FMT_RGB666_1X36_CPADLO :
|
||||
MEDIA_BUS_FMT_RGB888_1X36_CPADLO;
|
||||
} else {
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X36_CPADLO;
|
||||
}
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB666_1X7X3_SPWG:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB666_1X36_CPADLO;
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_SPWG:
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X36_CPADLO;
|
||||
break;
|
||||
default:
|
||||
kfree(input_fmts);
|
||||
input_fmts = NULL;
|
||||
break;
|
||||
}
|
||||
|
||||
return input_fmts;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qm_ldb_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
unsigned int *num_output_fmts)
|
||||
{
|
||||
*num_output_fmts = ARRAY_SIZE(imx8qm_ldb_bus_output_fmts);
|
||||
return kmemdup(imx8qm_ldb_bus_output_fmts,
|
||||
sizeof(imx8qm_ldb_bus_output_fmts), GFP_KERNEL);
|
||||
}
|
||||
|
||||
static enum drm_mode_status
|
||||
imx8qm_ldb_bridge_mode_valid(struct drm_bridge *bridge,
|
||||
const struct drm_display_info *info,
|
||||
const struct drm_display_mode *mode)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
bool is_single = ldb_channel_is_single_link(ldb_ch);
|
||||
|
||||
if (mode->clock > 300000)
|
||||
return MODE_CLOCK_HIGH;
|
||||
|
||||
if (mode->clock > 150000 && is_single)
|
||||
return MODE_CLOCK_HIGH;
|
||||
|
||||
return MODE_OK;
|
||||
}
|
||||
|
||||
static const struct drm_bridge_funcs imx8qm_ldb_bridge_funcs = {
|
||||
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
||||
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
||||
.atomic_reset = drm_atomic_helper_bridge_reset,
|
||||
.mode_valid = imx8qm_ldb_bridge_mode_valid,
|
||||
.attach = ldb_bridge_attach_helper,
|
||||
.atomic_check = imx8qm_ldb_bridge_atomic_check,
|
||||
.mode_set = imx8qm_ldb_bridge_mode_set,
|
||||
.atomic_enable = imx8qm_ldb_bridge_atomic_enable,
|
||||
.atomic_disable = imx8qm_ldb_bridge_atomic_disable,
|
||||
.atomic_get_input_bus_fmts =
|
||||
imx8qm_ldb_bridge_atomic_get_input_bus_fmts,
|
||||
.atomic_get_output_bus_fmts =
|
||||
imx8qm_ldb_bridge_atomic_get_output_bus_fmts,
|
||||
};
|
||||
|
||||
static int imx8qm_ldb_get_phy(struct imx8qm_ldb *imx8qm_ldb)
|
||||
{
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch;
|
||||
struct ldb_channel *ldb_ch;
|
||||
struct device *dev = imx8qm_ldb->dev;
|
||||
int i, ret;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
imx8qm_ldb_ch = &imx8qm_ldb->channel[i];
|
||||
ldb_ch = &imx8qm_ldb_ch->base;
|
||||
|
||||
if (!ldb_ch->is_available)
|
||||
continue;
|
||||
|
||||
imx8qm_ldb_ch->phy = devm_of_phy_get(dev, ldb_ch->np,
|
||||
"lvds_phy");
|
||||
if (IS_ERR(imx8qm_ldb_ch->phy)) {
|
||||
ret = PTR_ERR(imx8qm_ldb_ch->phy);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get channel%d PHY: %d\n",
|
||||
i, ret);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int imx8qm_ldb_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
struct imx8qm_ldb *imx8qm_ldb;
|
||||
struct imx8qm_ldb_channel *imx8qm_ldb_ch;
|
||||
struct ldb *ldb;
|
||||
struct ldb_channel *ldb_ch;
|
||||
struct device_node *port1, *port2;
|
||||
int pixel_order;
|
||||
int ret, i;
|
||||
|
||||
imx8qm_ldb = devm_kzalloc(dev, sizeof(*imx8qm_ldb), GFP_KERNEL);
|
||||
if (!imx8qm_ldb)
|
||||
return -ENOMEM;
|
||||
|
||||
imx8qm_ldb->clk_pixel = devm_clk_get(dev, "pixel");
|
||||
if (IS_ERR(imx8qm_ldb->clk_pixel)) {
|
||||
ret = PTR_ERR(imx8qm_ldb->clk_pixel);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get pixel clock: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
imx8qm_ldb->clk_bypass = devm_clk_get(dev, "bypass");
|
||||
if (IS_ERR(imx8qm_ldb->clk_bypass)) {
|
||||
ret = PTR_ERR(imx8qm_ldb->clk_bypass);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get bypass clock: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
imx8qm_ldb->dev = dev;
|
||||
|
||||
ldb = &imx8qm_ldb->base;
|
||||
ldb->dev = dev;
|
||||
ldb->ctrl_reg = 0xe0;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++)
|
||||
ldb->channel[i] = &imx8qm_ldb->channel[i].base;
|
||||
|
||||
ret = ldb_init_helper(ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (ldb->available_ch_cnt == 0) {
|
||||
DRM_DEV_DEBUG_DRIVER(dev, "no available channel\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (ldb->available_ch_cnt == 2) {
|
||||
port1 = of_graph_get_port_by_id(ldb->channel[0]->np, 1);
|
||||
port2 = of_graph_get_port_by_id(ldb->channel[1]->np, 1);
|
||||
pixel_order =
|
||||
drm_of_lvds_get_dual_link_pixel_order(port1, port2);
|
||||
of_node_put(port1);
|
||||
of_node_put(port2);
|
||||
|
||||
if (pixel_order != DRM_LVDS_DUAL_LINK_ODD_EVEN_PIXELS) {
|
||||
DRM_DEV_ERROR(dev, "invalid dual link pixel order: %d\n",
|
||||
pixel_order);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
imx8qm_ldb->active_chno = 0;
|
||||
imx8qm_ldb_ch = &imx8qm_ldb->channel[0];
|
||||
ldb_ch = &imx8qm_ldb_ch->base;
|
||||
ldb_ch->link_type = pixel_order;
|
||||
} else {
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
imx8qm_ldb_ch = &imx8qm_ldb->channel[i];
|
||||
ldb_ch = &imx8qm_ldb_ch->base;
|
||||
|
||||
if (ldb_ch->is_available) {
|
||||
imx8qm_ldb->active_chno = ldb_ch->chno;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ret = imx8qm_ldb_get_phy(imx8qm_ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = ldb_find_next_bridge_helper(ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
platform_set_drvdata(pdev, imx8qm_ldb);
|
||||
pm_runtime_enable(dev);
|
||||
|
||||
ldb_add_bridge_helper(ldb, &imx8qm_ldb_bridge_funcs);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qm_ldb_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qm_ldb *imx8qm_ldb = platform_get_drvdata(pdev);
|
||||
struct ldb *ldb = &imx8qm_ldb->base;
|
||||
|
||||
ldb_remove_bridge_helper(ldb);
|
||||
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qm_ldb_runtime_suspend(struct device *dev)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qm_ldb_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct imx8qm_ldb *imx8qm_ldb = dev_get_drvdata(dev);
|
||||
struct ldb *ldb = &imx8qm_ldb->base;
|
||||
|
||||
/* disable LDB by resetting the control register to POR default */
|
||||
regmap_write(ldb->regmap, ldb->ctrl_reg, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct dev_pm_ops imx8qm_ldb_pm_ops = {
|
||||
SET_RUNTIME_PM_OPS(imx8qm_ldb_runtime_suspend,
|
||||
imx8qm_ldb_runtime_resume, NULL)
|
||||
};
|
||||
|
||||
static const struct of_device_id imx8qm_ldb_dt_ids[] = {
|
||||
{ .compatible = "fsl,imx8qm-ldb" },
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx8qm_ldb_dt_ids);
|
||||
|
||||
static struct platform_driver imx8qm_ldb_driver = {
|
||||
.probe = imx8qm_ldb_probe,
|
||||
.remove = imx8qm_ldb_remove,
|
||||
.driver = {
|
||||
.pm = &imx8qm_ldb_pm_ops,
|
||||
.name = DRIVER_NAME,
|
||||
.of_match_table = imx8qm_ldb_dt_ids,
|
||||
},
|
||||
};
|
||||
module_platform_driver(imx8qm_ldb_driver);
|
||||
|
||||
MODULE_DESCRIPTION("i.MX8QM LVDS Display Bridge(LDB)/Pixel Mapper bridge driver");
|
||||
MODULE_AUTHOR("Liu Ying <victor.liu@nxp.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:" DRIVER_NAME);
|
|
@ -0,0 +1,722 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
/*
|
||||
* Copyright 2020 NXP
|
||||
*/
|
||||
|
||||
#include <linux/clk.h>
|
||||
#include <linux/mfd/syscon.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of_graph.h>
|
||||
#include <linux/phy/phy.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include "imx-ldb-helper.h"
|
||||
|
||||
#define LDB_CH_SEL BIT(28)
|
||||
|
||||
#define SS_CTRL 0x20
|
||||
#define CH_HSYNC_M(id) BIT(0 + ((id) * 2))
|
||||
#define CH_VSYNC_M(id) BIT(1 + ((id) * 2))
|
||||
#define CH_PHSYNC(id) BIT(0 + ((id) * 2))
|
||||
#define CH_PVSYNC(id) BIT(1 + ((id) * 2))
|
||||
|
||||
#define DRIVER_NAME "imx8qxp-ldb"
|
||||
|
||||
struct imx8qxp_ldb_channel {
|
||||
struct ldb_channel base;
|
||||
struct phy *phy;
|
||||
unsigned int di_id;
|
||||
};
|
||||
|
||||
struct imx8qxp_ldb {
|
||||
struct ldb base;
|
||||
struct device *dev;
|
||||
struct imx8qxp_ldb_channel channel[MAX_LDB_CHAN_NUM];
|
||||
struct clk *clk_pixel;
|
||||
struct clk *clk_bypass;
|
||||
struct drm_bridge *companion;
|
||||
int active_chno;
|
||||
};
|
||||
|
||||
static inline struct imx8qxp_ldb_channel *
|
||||
base_to_imx8qxp_ldb_channel(struct ldb_channel *base)
|
||||
{
|
||||
return container_of(base, struct imx8qxp_ldb_channel, base);
|
||||
}
|
||||
|
||||
static inline struct imx8qxp_ldb *base_to_imx8qxp_ldb(struct ldb *base)
|
||||
{
|
||||
return container_of(base, struct imx8qxp_ldb, base);
|
||||
}
|
||||
|
||||
static void imx8qxp_ldb_set_phy_cfg(struct imx8qxp_ldb *imx8qxp_ldb,
|
||||
unsigned long di_clk, bool is_split,
|
||||
struct phy_configure_opts_lvds *phy_cfg)
|
||||
{
|
||||
phy_cfg->bits_per_lane_and_dclk_cycle = 7;
|
||||
phy_cfg->lanes = 4;
|
||||
|
||||
if (is_split) {
|
||||
phy_cfg->differential_clk_rate = di_clk / 2;
|
||||
phy_cfg->is_slave = !imx8qxp_ldb->companion;
|
||||
} else {
|
||||
phy_cfg->differential_clk_rate = di_clk;
|
||||
phy_cfg->is_slave = false;
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
imx8qxp_ldb_bridge_atomic_check(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
base_to_imx8qxp_ldb_channel(ldb_ch);
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = base_to_imx8qxp_ldb(ldb);
|
||||
struct drm_bridge *companion = imx8qxp_ldb->companion;
|
||||
struct drm_display_mode *adj = &crtc_state->adjusted_mode;
|
||||
unsigned long di_clk = adj->clock * 1000;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
union phy_configure_opts opts = { };
|
||||
struct phy_configure_opts_lvds *phy_cfg = &opts.lvds;
|
||||
int ret;
|
||||
|
||||
ret = ldb_bridge_atomic_check_helper(bridge, bridge_state,
|
||||
crtc_state, conn_state);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
imx8qxp_ldb_set_phy_cfg(imx8qxp_ldb, di_clk, is_split, phy_cfg);
|
||||
ret = phy_validate(imx8qxp_ldb_ch->phy, PHY_MODE_LVDS, 0, &opts);
|
||||
if (ret < 0) {
|
||||
DRM_DEV_DEBUG_DRIVER(imx8qxp_ldb->dev,
|
||||
"failed to validate PHY: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (is_split && companion) {
|
||||
ret = companion->funcs->atomic_check(companion,
|
||||
bridge_state, crtc_state, conn_state);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_ldb_bridge_mode_set(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb_channel *companion_ldb_ch;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
base_to_imx8qxp_ldb_channel(ldb_ch);
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = base_to_imx8qxp_ldb(ldb);
|
||||
struct drm_bridge *companion = imx8qxp_ldb->companion;
|
||||
struct device *dev = imx8qxp_ldb->dev;
|
||||
unsigned long di_clk = adjusted_mode->clock * 1000;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
union phy_configure_opts opts = { };
|
||||
struct phy_configure_opts_lvds *phy_cfg = &opts.lvds;
|
||||
u32 chno = ldb_ch->chno;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to get runtime PM sync: %d\n", ret);
|
||||
|
||||
ret = phy_init(imx8qxp_ldb_ch->phy);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to initialize PHY: %d\n", ret);
|
||||
|
||||
ret = phy_set_mode(imx8qxp_ldb_ch->phy, PHY_MODE_LVDS);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to set PHY mode: %d\n", ret);
|
||||
|
||||
if (is_split && companion) {
|
||||
companion_ldb_ch = bridge_to_ldb_ch(companion);
|
||||
|
||||
companion_ldb_ch->in_bus_format = ldb_ch->in_bus_format;
|
||||
companion_ldb_ch->out_bus_format = ldb_ch->out_bus_format;
|
||||
}
|
||||
|
||||
clk_set_rate(imx8qxp_ldb->clk_bypass, di_clk);
|
||||
clk_set_rate(imx8qxp_ldb->clk_pixel, di_clk);
|
||||
|
||||
imx8qxp_ldb_set_phy_cfg(imx8qxp_ldb, di_clk, is_split, phy_cfg);
|
||||
ret = phy_configure(imx8qxp_ldb_ch->phy, &opts);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to configure PHY: %d\n", ret);
|
||||
|
||||
if (chno == 0)
|
||||
ldb->ldb_ctrl &= ~LDB_CH_SEL;
|
||||
else
|
||||
ldb->ldb_ctrl |= LDB_CH_SEL;
|
||||
|
||||
/* input VSYNC signal from pixel link is active low */
|
||||
if (imx8qxp_ldb_ch->di_id == 0)
|
||||
ldb->ldb_ctrl |= LDB_DI0_VS_POL_ACT_LOW;
|
||||
else
|
||||
ldb->ldb_ctrl |= LDB_DI1_VS_POL_ACT_LOW;
|
||||
|
||||
/*
|
||||
* For split mode, settle input VSYNC signal polarity and
|
||||
* channel selection down early.
|
||||
*/
|
||||
if (is_split)
|
||||
regmap_write(ldb->regmap, ldb->ctrl_reg, ldb->ldb_ctrl);
|
||||
|
||||
ldb_bridge_mode_set_helper(bridge, mode, adjusted_mode);
|
||||
|
||||
if (adjusted_mode->flags & DRM_MODE_FLAG_NVSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL, CH_VSYNC_M(chno), 0);
|
||||
else if (adjusted_mode->flags & DRM_MODE_FLAG_PVSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL,
|
||||
CH_VSYNC_M(chno), CH_PVSYNC(chno));
|
||||
|
||||
if (adjusted_mode->flags & DRM_MODE_FLAG_NHSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL, CH_HSYNC_M(chno), 0);
|
||||
else if (adjusted_mode->flags & DRM_MODE_FLAG_PHSYNC)
|
||||
regmap_update_bits(ldb->regmap, SS_CTRL,
|
||||
CH_HSYNC_M(chno), CH_PHSYNC(chno));
|
||||
|
||||
if (is_split && companion)
|
||||
companion->funcs->mode_set(companion, mode, adjusted_mode);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_ldb_bridge_atomic_pre_enable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = base_to_imx8qxp_ldb(ldb);
|
||||
struct drm_bridge *companion = imx8qxp_ldb->companion;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
|
||||
clk_prepare_enable(imx8qxp_ldb->clk_pixel);
|
||||
clk_prepare_enable(imx8qxp_ldb->clk_bypass);
|
||||
|
||||
if (is_split && companion)
|
||||
companion->funcs->atomic_pre_enable(companion, old_bridge_state);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_ldb_bridge_atomic_enable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
base_to_imx8qxp_ldb_channel(ldb_ch);
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = base_to_imx8qxp_ldb(ldb);
|
||||
struct drm_bridge *companion = imx8qxp_ldb->companion;
|
||||
struct device *dev = imx8qxp_ldb->dev;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
int ret;
|
||||
|
||||
if (ldb_ch->chno == 0 || is_split) {
|
||||
ldb->ldb_ctrl &= ~LDB_CH0_MODE_EN_MASK;
|
||||
ldb->ldb_ctrl |= imx8qxp_ldb_ch->di_id == 0 ?
|
||||
LDB_CH0_MODE_EN_TO_DI0 : LDB_CH0_MODE_EN_TO_DI1;
|
||||
}
|
||||
if (ldb_ch->chno == 1 || is_split) {
|
||||
ldb->ldb_ctrl &= ~LDB_CH1_MODE_EN_MASK;
|
||||
ldb->ldb_ctrl |= imx8qxp_ldb_ch->di_id == 0 ?
|
||||
LDB_CH1_MODE_EN_TO_DI0 : LDB_CH1_MODE_EN_TO_DI1;
|
||||
}
|
||||
|
||||
ldb_bridge_enable_helper(bridge);
|
||||
|
||||
ret = phy_power_on(imx8qxp_ldb_ch->phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev, "failed to power on PHY: %d\n", ret);
|
||||
|
||||
if (is_split && companion)
|
||||
companion->funcs->atomic_enable(companion, old_bridge_state);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_ldb_bridge_atomic_disable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
struct ldb *ldb = ldb_ch->ldb;
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
base_to_imx8qxp_ldb_channel(ldb_ch);
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = base_to_imx8qxp_ldb(ldb);
|
||||
struct drm_bridge *companion = imx8qxp_ldb->companion;
|
||||
struct device *dev = imx8qxp_ldb->dev;
|
||||
bool is_split = ldb_channel_is_split_link(ldb_ch);
|
||||
int ret;
|
||||
|
||||
ret = phy_power_off(imx8qxp_ldb_ch->phy);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(dev, "failed to power off PHY: %d\n", ret);
|
||||
|
||||
ret = phy_exit(imx8qxp_ldb_ch->phy);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to teardown PHY: %d\n", ret);
|
||||
|
||||
ldb_bridge_disable_helper(bridge);
|
||||
|
||||
clk_disable_unprepare(imx8qxp_ldb->clk_bypass);
|
||||
clk_disable_unprepare(imx8qxp_ldb->clk_pixel);
|
||||
|
||||
if (is_split && companion)
|
||||
companion->funcs->atomic_disable(companion, old_bridge_state);
|
||||
|
||||
ret = pm_runtime_put(dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(dev, "failed to put runtime PM: %d\n", ret);
|
||||
}
|
||||
|
||||
static const u32 imx8qxp_ldb_bus_output_fmts[] = {
|
||||
MEDIA_BUS_FMT_RGB666_1X7X3_SPWG,
|
||||
MEDIA_BUS_FMT_RGB888_1X7X4_SPWG,
|
||||
MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA,
|
||||
MEDIA_BUS_FMT_FIXED,
|
||||
};
|
||||
|
||||
static bool imx8qxp_ldb_bus_output_fmt_supported(u32 fmt)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(imx8qxp_ldb_bus_output_fmts); i++) {
|
||||
if (imx8qxp_ldb_bus_output_fmts[i] == fmt)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_ldb_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
u32 output_fmt,
|
||||
unsigned int *num_input_fmts)
|
||||
{
|
||||
struct drm_display_info *di;
|
||||
const struct drm_format_info *finfo;
|
||||
u32 *input_fmts;
|
||||
|
||||
if (!imx8qxp_ldb_bus_output_fmt_supported(output_fmt))
|
||||
return NULL;
|
||||
|
||||
*num_input_fmts = 1;
|
||||
|
||||
input_fmts = kmalloc(sizeof(*input_fmts), GFP_KERNEL);
|
||||
if (!input_fmts)
|
||||
return NULL;
|
||||
|
||||
switch (output_fmt) {
|
||||
case MEDIA_BUS_FMT_FIXED:
|
||||
di = &conn_state->connector->display_info;
|
||||
|
||||
/*
|
||||
* Look at the first bus format to determine input format.
|
||||
* Default to MEDIA_BUS_FMT_RGB888_1X24, if no match.
|
||||
*/
|
||||
if (di->num_bus_formats) {
|
||||
finfo = drm_format_info(di->bus_formats[0]);
|
||||
|
||||
input_fmts[0] = finfo->depth == 18 ?
|
||||
MEDIA_BUS_FMT_RGB666_1X24_CPADHI :
|
||||
MEDIA_BUS_FMT_RGB888_1X24;
|
||||
} else {
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;
|
||||
}
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB666_1X7X3_SPWG:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB666_1X24_CPADHI;
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_SPWG:
|
||||
case MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;
|
||||
break;
|
||||
default:
|
||||
kfree(input_fmts);
|
||||
input_fmts = NULL;
|
||||
break;
|
||||
}
|
||||
|
||||
return input_fmts;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_ldb_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
unsigned int *num_output_fmts)
|
||||
{
|
||||
*num_output_fmts = ARRAY_SIZE(imx8qxp_ldb_bus_output_fmts);
|
||||
return kmemdup(imx8qxp_ldb_bus_output_fmts,
|
||||
sizeof(imx8qxp_ldb_bus_output_fmts), GFP_KERNEL);
|
||||
}
|
||||
|
||||
static enum drm_mode_status
|
||||
imx8qxp_ldb_bridge_mode_valid(struct drm_bridge *bridge,
|
||||
const struct drm_display_info *info,
|
||||
const struct drm_display_mode *mode)
|
||||
{
|
||||
struct ldb_channel *ldb_ch = bridge->driver_private;
|
||||
bool is_single = ldb_channel_is_single_link(ldb_ch);
|
||||
|
||||
if (mode->clock > 170000)
|
||||
return MODE_CLOCK_HIGH;
|
||||
|
||||
if (mode->clock > 150000 && is_single)
|
||||
return MODE_CLOCK_HIGH;
|
||||
|
||||
return MODE_OK;
|
||||
}
|
||||
|
||||
static const struct drm_bridge_funcs imx8qxp_ldb_bridge_funcs = {
|
||||
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
||||
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
||||
.atomic_reset = drm_atomic_helper_bridge_reset,
|
||||
.mode_valid = imx8qxp_ldb_bridge_mode_valid,
|
||||
.attach = ldb_bridge_attach_helper,
|
||||
.atomic_check = imx8qxp_ldb_bridge_atomic_check,
|
||||
.mode_set = imx8qxp_ldb_bridge_mode_set,
|
||||
.atomic_pre_enable = imx8qxp_ldb_bridge_atomic_pre_enable,
|
||||
.atomic_enable = imx8qxp_ldb_bridge_atomic_enable,
|
||||
.atomic_disable = imx8qxp_ldb_bridge_atomic_disable,
|
||||
.atomic_get_input_bus_fmts =
|
||||
imx8qxp_ldb_bridge_atomic_get_input_bus_fmts,
|
||||
.atomic_get_output_bus_fmts =
|
||||
imx8qxp_ldb_bridge_atomic_get_output_bus_fmts,
|
||||
};
|
||||
|
||||
static int imx8qxp_ldb_set_di_id(struct imx8qxp_ldb *imx8qxp_ldb)
|
||||
{
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
&imx8qxp_ldb->channel[imx8qxp_ldb->active_chno];
|
||||
struct ldb_channel *ldb_ch = &imx8qxp_ldb_ch->base;
|
||||
struct device_node *ep, *remote;
|
||||
struct device *dev = imx8qxp_ldb->dev;
|
||||
struct of_endpoint endpoint;
|
||||
int ret;
|
||||
|
||||
ep = of_graph_get_endpoint_by_regs(ldb_ch->np, 0, -1);
|
||||
if (!ep) {
|
||||
DRM_DEV_ERROR(dev, "failed to get port0 endpoint\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
remote = of_graph_get_remote_endpoint(ep);
|
||||
of_node_put(ep);
|
||||
if (!remote) {
|
||||
DRM_DEV_ERROR(dev, "failed to get port0 remote endpoint\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = of_graph_parse_endpoint(remote, &endpoint);
|
||||
of_node_put(remote);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(dev, "failed to parse port0 remote endpoint: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
imx8qxp_ldb_ch->di_id = endpoint.id;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
imx8qxp_ldb_check_chno_and_dual_link(struct ldb_channel *ldb_ch, int link)
|
||||
{
|
||||
if ((link == DRM_LVDS_DUAL_LINK_ODD_EVEN_PIXELS && ldb_ch->chno != 0) ||
|
||||
(link == DRM_LVDS_DUAL_LINK_EVEN_ODD_PIXELS && ldb_ch->chno != 1))
|
||||
return -EINVAL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int imx8qxp_ldb_parse_dt_companion(struct imx8qxp_ldb *imx8qxp_ldb)
|
||||
{
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch =
|
||||
&imx8qxp_ldb->channel[imx8qxp_ldb->active_chno];
|
||||
struct ldb_channel *ldb_ch = &imx8qxp_ldb_ch->base;
|
||||
struct ldb_channel *companion_ldb_ch;
|
||||
struct device_node *companion;
|
||||
struct device_node *child;
|
||||
struct device_node *companion_port = NULL;
|
||||
struct device_node *port1, *port2;
|
||||
struct device *dev = imx8qxp_ldb->dev;
|
||||
const struct of_device_id *match;
|
||||
u32 i;
|
||||
int dual_link;
|
||||
int ret;
|
||||
|
||||
/* Locate the companion LDB for dual-link operation, if any. */
|
||||
companion = of_parse_phandle(dev->of_node, "fsl,companion-ldb", 0);
|
||||
if (!companion)
|
||||
return 0;
|
||||
|
||||
if (!of_device_is_available(companion)) {
|
||||
DRM_DEV_ERROR(dev, "companion LDB is not available\n");
|
||||
ret = -ENODEV;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* Sanity check: the companion bridge must have the same compatible
|
||||
* string.
|
||||
*/
|
||||
match = of_match_device(dev->driver->of_match_table, dev);
|
||||
if (!of_device_is_compatible(companion, match->compatible)) {
|
||||
DRM_DEV_ERROR(dev, "companion LDB is incompatible\n");
|
||||
ret = -ENXIO;
|
||||
goto out;
|
||||
}
|
||||
|
||||
for_each_available_child_of_node(companion, child) {
|
||||
ret = of_property_read_u32(child, "reg", &i);
|
||||
if (ret || i > MAX_LDB_CHAN_NUM - 1) {
|
||||
DRM_DEV_ERROR(dev,
|
||||
"invalid channel node address: %u\n", i);
|
||||
ret = -EINVAL;
|
||||
of_node_put(child);
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* Channel numbers have to be different, because channel0
|
||||
* transmits odd pixels and channel1 transmits even pixels.
|
||||
*/
|
||||
if (i == (ldb_ch->chno ^ 0x1)) {
|
||||
companion_port = child;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!companion_port) {
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to find companion LDB channel port\n");
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* We need to work out if the sink is expecting us to function in
|
||||
* dual-link mode. We do this by looking at the DT port nodes we are
|
||||
* connected to. If they are marked as expecting odd pixels and
|
||||
* even pixels than we need to enable LDB split mode.
|
||||
*/
|
||||
port1 = of_graph_get_port_by_id(ldb_ch->np, 1);
|
||||
port2 = of_graph_get_port_by_id(companion_port, 1);
|
||||
dual_link = drm_of_lvds_get_dual_link_pixel_order(port1, port2);
|
||||
of_node_put(port1);
|
||||
of_node_put(port2);
|
||||
|
||||
switch (dual_link) {
|
||||
case DRM_LVDS_DUAL_LINK_ODD_EVEN_PIXELS:
|
||||
ldb_ch->link_type = LDB_CH_DUAL_LINK_ODD_EVEN_PIXELS;
|
||||
break;
|
||||
case DRM_LVDS_DUAL_LINK_EVEN_ODD_PIXELS:
|
||||
ldb_ch->link_type = LDB_CH_DUAL_LINK_EVEN_ODD_PIXELS;
|
||||
break;
|
||||
default:
|
||||
ret = dual_link;
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get dual link pixel order: %d\n", ret);
|
||||
goto out;
|
||||
}
|
||||
|
||||
ret = imx8qxp_ldb_check_chno_and_dual_link(ldb_ch, dual_link);
|
||||
if (ret < 0) {
|
||||
DRM_DEV_ERROR(dev,
|
||||
"unmatched channel number(%u) vs dual link(%d)\n",
|
||||
ldb_ch->chno, dual_link);
|
||||
goto out;
|
||||
}
|
||||
|
||||
imx8qxp_ldb->companion = of_drm_find_bridge(companion_port);
|
||||
if (!imx8qxp_ldb->companion) {
|
||||
ret = -EPROBE_DEFER;
|
||||
DRM_DEV_DEBUG_DRIVER(dev,
|
||||
"failed to find bridge for companion bridge: %d\n",
|
||||
ret);
|
||||
goto out;
|
||||
}
|
||||
|
||||
DRM_DEV_DEBUG_DRIVER(dev,
|
||||
"dual-link configuration detected (companion bridge %pOF)\n",
|
||||
companion);
|
||||
|
||||
companion_ldb_ch = bridge_to_ldb_ch(imx8qxp_ldb->companion);
|
||||
companion_ldb_ch->link_type = ldb_ch->link_type;
|
||||
out:
|
||||
of_node_put(companion_port);
|
||||
of_node_put(companion);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_ldb_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct device *dev = &pdev->dev;
|
||||
struct imx8qxp_ldb *imx8qxp_ldb;
|
||||
struct imx8qxp_ldb_channel *imx8qxp_ldb_ch;
|
||||
struct ldb *ldb;
|
||||
struct ldb_channel *ldb_ch;
|
||||
int ret, i;
|
||||
|
||||
imx8qxp_ldb = devm_kzalloc(dev, sizeof(*imx8qxp_ldb), GFP_KERNEL);
|
||||
if (!imx8qxp_ldb)
|
||||
return -ENOMEM;
|
||||
|
||||
imx8qxp_ldb->clk_pixel = devm_clk_get(dev, "pixel");
|
||||
if (IS_ERR(imx8qxp_ldb->clk_pixel)) {
|
||||
ret = PTR_ERR(imx8qxp_ldb->clk_pixel);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get pixel clock: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
imx8qxp_ldb->clk_bypass = devm_clk_get(dev, "bypass");
|
||||
if (IS_ERR(imx8qxp_ldb->clk_bypass)) {
|
||||
ret = PTR_ERR(imx8qxp_ldb->clk_bypass);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev,
|
||||
"failed to get bypass clock: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
imx8qxp_ldb->dev = dev;
|
||||
|
||||
ldb = &imx8qxp_ldb->base;
|
||||
ldb->dev = dev;
|
||||
ldb->ctrl_reg = 0xe0;
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++)
|
||||
ldb->channel[i] = &imx8qxp_ldb->channel[i].base;
|
||||
|
||||
ret = ldb_init_helper(ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (ldb->available_ch_cnt == 0) {
|
||||
DRM_DEV_DEBUG_DRIVER(dev, "no available channel\n");
|
||||
return 0;
|
||||
} else if (ldb->available_ch_cnt > 1) {
|
||||
DRM_DEV_ERROR(dev, "invalid available channel number(%u)\n",
|
||||
ldb->available_ch_cnt);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
for (i = 0; i < MAX_LDB_CHAN_NUM; i++) {
|
||||
imx8qxp_ldb_ch = &imx8qxp_ldb->channel[i];
|
||||
ldb_ch = &imx8qxp_ldb_ch->base;
|
||||
|
||||
if (ldb_ch->is_available) {
|
||||
imx8qxp_ldb->active_chno = ldb_ch->chno;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
imx8qxp_ldb_ch->phy = devm_of_phy_get(dev, ldb_ch->np, "lvds_phy");
|
||||
if (IS_ERR(imx8qxp_ldb_ch->phy)) {
|
||||
ret = PTR_ERR(imx8qxp_ldb_ch->phy);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get channel%d PHY: %d\n",
|
||||
imx8qxp_ldb->active_chno, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = ldb_find_next_bridge_helper(ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = imx8qxp_ldb_set_di_id(imx8qxp_ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = imx8qxp_ldb_parse_dt_companion(imx8qxp_ldb);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
platform_set_drvdata(pdev, imx8qxp_ldb);
|
||||
pm_runtime_enable(dev);
|
||||
|
||||
ldb_add_bridge_helper(ldb, &imx8qxp_ldb_bridge_funcs);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_ldb_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = platform_get_drvdata(pdev);
|
||||
struct ldb *ldb = &imx8qxp_ldb->base;
|
||||
|
||||
ldb_remove_bridge_helper(ldb);
|
||||
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qxp_ldb_runtime_suspend(struct device *dev)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qxp_ldb_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct imx8qxp_ldb *imx8qxp_ldb = dev_get_drvdata(dev);
|
||||
struct ldb *ldb = &imx8qxp_ldb->base;
|
||||
|
||||
/* disable LDB by resetting the control register to POR default */
|
||||
regmap_write(ldb->regmap, ldb->ctrl_reg, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct dev_pm_ops imx8qxp_ldb_pm_ops = {
|
||||
SET_RUNTIME_PM_OPS(imx8qxp_ldb_runtime_suspend,
|
||||
imx8qxp_ldb_runtime_resume, NULL)
|
||||
};
|
||||
|
||||
static const struct of_device_id imx8qxp_ldb_dt_ids[] = {
|
||||
{ .compatible = "fsl,imx8qxp-ldb" },
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx8qxp_ldb_dt_ids);
|
||||
|
||||
static struct platform_driver imx8qxp_ldb_driver = {
|
||||
.probe = imx8qxp_ldb_probe,
|
||||
.remove = imx8qxp_ldb_remove,
|
||||
.driver = {
|
||||
.pm = &imx8qxp_ldb_pm_ops,
|
||||
.name = DRIVER_NAME,
|
||||
.of_match_table = imx8qxp_ldb_dt_ids,
|
||||
},
|
||||
};
|
||||
module_platform_driver(imx8qxp_ldb_driver);
|
||||
|
||||
MODULE_DESCRIPTION("i.MX8QXP LVDS Display Bridge(LDB)/Pixel Mapper bridge driver");
|
||||
MODULE_AUTHOR("Liu Ying <victor.liu@nxp.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:" DRIVER_NAME);
|
|
@ -0,0 +1,448 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
/*
|
||||
* Copyright 2020 NXP
|
||||
*/
|
||||
|
||||
#include <linux/bitfield.h>
|
||||
#include <linux/clk.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_graph.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#define PC_CTRL_REG 0x0
|
||||
#define PC_COMBINE_ENABLE BIT(0)
|
||||
#define PC_DISP_BYPASS(n) BIT(1 + 21 * (n))
|
||||
#define PC_DISP_HSYNC_POLARITY(n) BIT(2 + 11 * (n))
|
||||
#define PC_DISP_HSYNC_POLARITY_POS(n) DISP_HSYNC_POLARITY(n)
|
||||
#define PC_DISP_VSYNC_POLARITY(n) BIT(3 + 11 * (n))
|
||||
#define PC_DISP_VSYNC_POLARITY_POS(n) DISP_VSYNC_POLARITY(n)
|
||||
#define PC_DISP_DVALID_POLARITY(n) BIT(4 + 11 * (n))
|
||||
#define PC_DISP_DVALID_POLARITY_POS(n) DISP_DVALID_POLARITY(n)
|
||||
#define PC_VSYNC_MASK_ENABLE BIT(5)
|
||||
#define PC_SKIP_MODE BIT(6)
|
||||
#define PC_SKIP_NUMBER_MASK GENMASK(12, 7)
|
||||
#define PC_SKIP_NUMBER(n) FIELD_PREP(PC_SKIP_NUMBER_MASK, (n))
|
||||
#define PC_DISP0_PIX_DATA_FORMAT_MASK GENMASK(18, 16)
|
||||
#define PC_DISP0_PIX_DATA_FORMAT(fmt) \
|
||||
FIELD_PREP(PC_DISP0_PIX_DATA_FORMAT_MASK, (fmt))
|
||||
#define PC_DISP1_PIX_DATA_FORMAT_MASK GENMASK(21, 19)
|
||||
#define PC_DISP1_PIX_DATA_FORMAT(fmt) \
|
||||
FIELD_PREP(PC_DISP1_PIX_DATA_FORMAT_MASK, (fmt))
|
||||
|
||||
#define PC_SW_RESET_REG 0x20
|
||||
#define PC_SW_RESET_N BIT(0)
|
||||
#define PC_DISP_SW_RESET_N(n) BIT(1 + (n))
|
||||
#define PC_FULL_RESET_N (PC_SW_RESET_N | \
|
||||
PC_DISP_SW_RESET_N(0) | \
|
||||
PC_DISP_SW_RESET_N(1))
|
||||
|
||||
#define PC_REG_SET 0x4
|
||||
#define PC_REG_CLR 0x8
|
||||
|
||||
#define DRIVER_NAME "imx8qxp-pixel-combiner"
|
||||
|
||||
enum imx8qxp_pc_pix_data_format {
|
||||
RGB,
|
||||
YUV444,
|
||||
YUV422,
|
||||
SPLIT_RGB,
|
||||
};
|
||||
|
||||
struct imx8qxp_pc_channel {
|
||||
struct drm_bridge bridge;
|
||||
struct drm_bridge *next_bridge;
|
||||
struct imx8qxp_pc *pc;
|
||||
unsigned int stream_id;
|
||||
bool is_available;
|
||||
};
|
||||
|
||||
struct imx8qxp_pc {
|
||||
struct device *dev;
|
||||
struct imx8qxp_pc_channel ch[2];
|
||||
struct clk *clk_apb;
|
||||
void __iomem *base;
|
||||
};
|
||||
|
||||
static inline u32 imx8qxp_pc_read(struct imx8qxp_pc *pc, unsigned int offset)
|
||||
{
|
||||
return readl(pc->base + offset);
|
||||
}
|
||||
|
||||
static inline void
|
||||
imx8qxp_pc_write(struct imx8qxp_pc *pc, unsigned int offset, u32 value)
|
||||
{
|
||||
writel(value, pc->base + offset);
|
||||
}
|
||||
|
||||
static inline void
|
||||
imx8qxp_pc_write_set(struct imx8qxp_pc *pc, unsigned int offset, u32 value)
|
||||
{
|
||||
imx8qxp_pc_write(pc, offset + PC_REG_SET, value);
|
||||
}
|
||||
|
||||
static inline void
|
||||
imx8qxp_pc_write_clr(struct imx8qxp_pc *pc, unsigned int offset, u32 value)
|
||||
{
|
||||
imx8qxp_pc_write(pc, offset + PC_REG_CLR, value);
|
||||
}
|
||||
|
||||
static enum drm_mode_status
|
||||
imx8qxp_pc_bridge_mode_valid(struct drm_bridge *bridge,
|
||||
const struct drm_display_info *info,
|
||||
const struct drm_display_mode *mode)
|
||||
{
|
||||
if (mode->hdisplay > 2560)
|
||||
return MODE_BAD_HVALUE;
|
||||
|
||||
return MODE_OK;
|
||||
}
|
||||
|
||||
static int imx8qxp_pc_bridge_attach(struct drm_bridge *bridge,
|
||||
enum drm_bridge_attach_flags flags)
|
||||
{
|
||||
struct imx8qxp_pc_channel *ch = bridge->driver_private;
|
||||
struct imx8qxp_pc *pc = ch->pc;
|
||||
|
||||
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
|
||||
DRM_DEV_ERROR(pc->dev,
|
||||
"do not support creating a drm_connector\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (!bridge->encoder) {
|
||||
DRM_DEV_ERROR(pc->dev, "missing encoder\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
return drm_bridge_attach(bridge->encoder,
|
||||
ch->next_bridge, bridge,
|
||||
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pc_bridge_mode_set(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct imx8qxp_pc_channel *ch = bridge->driver_private;
|
||||
struct imx8qxp_pc *pc = ch->pc;
|
||||
u32 val;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(pc->dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(pc->dev,
|
||||
"failed to get runtime PM sync: %d\n", ret);
|
||||
|
||||
ret = clk_prepare_enable(pc->clk_apb);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pc->dev, "%s: failed to enable apb clock: %d\n",
|
||||
__func__, ret);
|
||||
|
||||
/* HSYNC to pixel link is active low. */
|
||||
imx8qxp_pc_write_clr(pc, PC_CTRL_REG,
|
||||
PC_DISP_HSYNC_POLARITY(ch->stream_id));
|
||||
|
||||
/* VSYNC to pixel link is active low. */
|
||||
imx8qxp_pc_write_clr(pc, PC_CTRL_REG,
|
||||
PC_DISP_VSYNC_POLARITY(ch->stream_id));
|
||||
|
||||
/* Data enable to pixel link is active high. */
|
||||
imx8qxp_pc_write_set(pc, PC_CTRL_REG,
|
||||
PC_DISP_DVALID_POLARITY(ch->stream_id));
|
||||
|
||||
/* Mask the first frame output which may be incomplete. */
|
||||
imx8qxp_pc_write_set(pc, PC_CTRL_REG, PC_VSYNC_MASK_ENABLE);
|
||||
|
||||
/* Only support RGB currently. */
|
||||
val = imx8qxp_pc_read(pc, PC_CTRL_REG);
|
||||
if (ch->stream_id == 0) {
|
||||
val &= ~PC_DISP0_PIX_DATA_FORMAT_MASK;
|
||||
val |= PC_DISP0_PIX_DATA_FORMAT(RGB);
|
||||
} else {
|
||||
val &= ~PC_DISP1_PIX_DATA_FORMAT_MASK;
|
||||
val |= PC_DISP1_PIX_DATA_FORMAT(RGB);
|
||||
}
|
||||
imx8qxp_pc_write(pc, PC_CTRL_REG, val);
|
||||
|
||||
/* Only support bypass mode currently. */
|
||||
imx8qxp_pc_write_set(pc, PC_CTRL_REG, PC_DISP_BYPASS(ch->stream_id));
|
||||
|
||||
clk_disable_unprepare(pc->clk_apb);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pc_bridge_atomic_disable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct imx8qxp_pc_channel *ch = bridge->driver_private;
|
||||
struct imx8qxp_pc *pc = ch->pc;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_put(pc->dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(pc->dev, "failed to put runtime PM: %d\n", ret);
|
||||
}
|
||||
|
||||
static const u32 imx8qxp_pc_bus_output_fmts[] = {
|
||||
MEDIA_BUS_FMT_RGB888_1X36_CPADLO,
|
||||
MEDIA_BUS_FMT_RGB666_1X36_CPADLO,
|
||||
};
|
||||
|
||||
static bool imx8qxp_pc_bus_output_fmt_supported(u32 fmt)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(imx8qxp_pc_bus_output_fmts); i++) {
|
||||
if (imx8qxp_pc_bus_output_fmts[i] == fmt)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pc_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
u32 output_fmt,
|
||||
unsigned int *num_input_fmts)
|
||||
{
|
||||
u32 *input_fmts;
|
||||
|
||||
if (!imx8qxp_pc_bus_output_fmt_supported(output_fmt))
|
||||
return NULL;
|
||||
|
||||
*num_input_fmts = 1;
|
||||
|
||||
input_fmts = kmalloc(sizeof(*input_fmts), GFP_KERNEL);
|
||||
if (!input_fmts)
|
||||
return NULL;
|
||||
|
||||
switch (output_fmt) {
|
||||
case MEDIA_BUS_FMT_RGB888_1X36_CPADLO:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X30_CPADLO;
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB666_1X36_CPADLO:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB666_1X30_CPADLO;
|
||||
break;
|
||||
default:
|
||||
kfree(input_fmts);
|
||||
input_fmts = NULL;
|
||||
break;
|
||||
}
|
||||
|
||||
return input_fmts;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pc_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
unsigned int *num_output_fmts)
|
||||
{
|
||||
*num_output_fmts = ARRAY_SIZE(imx8qxp_pc_bus_output_fmts);
|
||||
return kmemdup(imx8qxp_pc_bus_output_fmts,
|
||||
sizeof(imx8qxp_pc_bus_output_fmts), GFP_KERNEL);
|
||||
}
|
||||
|
||||
static const struct drm_bridge_funcs imx8qxp_pc_bridge_funcs = {
|
||||
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
||||
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
||||
.atomic_reset = drm_atomic_helper_bridge_reset,
|
||||
.mode_valid = imx8qxp_pc_bridge_mode_valid,
|
||||
.attach = imx8qxp_pc_bridge_attach,
|
||||
.mode_set = imx8qxp_pc_bridge_mode_set,
|
||||
.atomic_disable = imx8qxp_pc_bridge_atomic_disable,
|
||||
.atomic_get_input_bus_fmts =
|
||||
imx8qxp_pc_bridge_atomic_get_input_bus_fmts,
|
||||
.atomic_get_output_bus_fmts =
|
||||
imx8qxp_pc_bridge_atomic_get_output_bus_fmts,
|
||||
};
|
||||
|
||||
static int imx8qxp_pc_bridge_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pc *pc;
|
||||
struct imx8qxp_pc_channel *ch;
|
||||
struct device *dev = &pdev->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
struct device_node *child, *remote;
|
||||
u32 i;
|
||||
int ret;
|
||||
|
||||
pc = devm_kzalloc(dev, sizeof(*pc), GFP_KERNEL);
|
||||
if (!pc)
|
||||
return -ENOMEM;
|
||||
|
||||
pc->base = devm_platform_ioremap_resource(pdev, 0);
|
||||
if (IS_ERR(pc->base))
|
||||
return PTR_ERR(pc->base);
|
||||
|
||||
pc->dev = dev;
|
||||
|
||||
pc->clk_apb = devm_clk_get(dev, "apb");
|
||||
if (IS_ERR(pc->clk_apb)) {
|
||||
ret = PTR_ERR(pc->clk_apb);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get apb clock: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, pc);
|
||||
pm_runtime_enable(dev);
|
||||
|
||||
for_each_available_child_of_node(np, child) {
|
||||
ret = of_property_read_u32(child, "reg", &i);
|
||||
if (ret || i > 1) {
|
||||
ret = -EINVAL;
|
||||
DRM_DEV_ERROR(dev,
|
||||
"invalid channel(%u) node address\n", i);
|
||||
goto free_child;
|
||||
}
|
||||
|
||||
ch = &pc->ch[i];
|
||||
ch->pc = pc;
|
||||
ch->stream_id = i;
|
||||
|
||||
remote = of_graph_get_remote_node(child, 1, 0);
|
||||
if (!remote) {
|
||||
ret = -ENODEV;
|
||||
DRM_DEV_ERROR(dev,
|
||||
"channel%u failed to get port1's remote node: %d\n",
|
||||
i, ret);
|
||||
goto free_child;
|
||||
}
|
||||
|
||||
ch->next_bridge = of_drm_find_bridge(remote);
|
||||
if (!ch->next_bridge) {
|
||||
of_node_put(remote);
|
||||
ret = -EPROBE_DEFER;
|
||||
DRM_DEV_DEBUG_DRIVER(dev,
|
||||
"channel%u failed to find next bridge: %d\n",
|
||||
i, ret);
|
||||
goto free_child;
|
||||
}
|
||||
|
||||
of_node_put(remote);
|
||||
|
||||
ch->bridge.driver_private = ch;
|
||||
ch->bridge.funcs = &imx8qxp_pc_bridge_funcs;
|
||||
ch->bridge.of_node = child;
|
||||
ch->is_available = true;
|
||||
|
||||
drm_bridge_add(&ch->bridge);
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
free_child:
|
||||
of_node_put(child);
|
||||
|
||||
if (i == 1 && pc->ch[0].next_bridge)
|
||||
drm_bridge_remove(&pc->ch[0].bridge);
|
||||
|
||||
pm_runtime_disable(dev);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pc_bridge_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pc *pc = platform_get_drvdata(pdev);
|
||||
struct imx8qxp_pc_channel *ch;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < 2; i++) {
|
||||
ch = &pc->ch[i];
|
||||
|
||||
if (!ch->is_available)
|
||||
continue;
|
||||
|
||||
drm_bridge_remove(&ch->bridge);
|
||||
ch->is_available = false;
|
||||
}
|
||||
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qxp_pc_runtime_suspend(struct device *dev)
|
||||
{
|
||||
struct platform_device *pdev = to_platform_device(dev);
|
||||
struct imx8qxp_pc *pc = platform_get_drvdata(pdev);
|
||||
int ret;
|
||||
|
||||
ret = clk_prepare_enable(pc->clk_apb);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pc->dev, "%s: failed to enable apb clock: %d\n",
|
||||
__func__, ret);
|
||||
|
||||
/* Disable pixel combiner by full reset. */
|
||||
imx8qxp_pc_write_clr(pc, PC_SW_RESET_REG, PC_FULL_RESET_N);
|
||||
|
||||
clk_disable_unprepare(pc->clk_apb);
|
||||
|
||||
/* Ensure the reset takes effect. */
|
||||
usleep_range(10, 20);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int __maybe_unused imx8qxp_pc_runtime_resume(struct device *dev)
|
||||
{
|
||||
struct platform_device *pdev = to_platform_device(dev);
|
||||
struct imx8qxp_pc *pc = platform_get_drvdata(pdev);
|
||||
int ret;
|
||||
|
||||
ret = clk_prepare_enable(pc->clk_apb);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(pc->dev, "%s: failed to enable apb clock: %d\n",
|
||||
__func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* out of reset */
|
||||
imx8qxp_pc_write_set(pc, PC_SW_RESET_REG, PC_FULL_RESET_N);
|
||||
|
||||
clk_disable_unprepare(pc->clk_apb);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct dev_pm_ops imx8qxp_pc_pm_ops = {
|
||||
SET_RUNTIME_PM_OPS(imx8qxp_pc_runtime_suspend,
|
||||
imx8qxp_pc_runtime_resume, NULL)
|
||||
};
|
||||
|
||||
static const struct of_device_id imx8qxp_pc_dt_ids[] = {
|
||||
{ .compatible = "fsl,imx8qm-pixel-combiner", },
|
||||
{ .compatible = "fsl,imx8qxp-pixel-combiner", },
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx8qxp_pc_dt_ids);
|
||||
|
||||
static struct platform_driver imx8qxp_pc_bridge_driver = {
|
||||
.probe = imx8qxp_pc_bridge_probe,
|
||||
.remove = imx8qxp_pc_bridge_remove,
|
||||
.driver = {
|
||||
.pm = &imx8qxp_pc_pm_ops,
|
||||
.name = DRIVER_NAME,
|
||||
.of_match_table = imx8qxp_pc_dt_ids,
|
||||
},
|
||||
};
|
||||
module_platform_driver(imx8qxp_pc_bridge_driver);
|
||||
|
||||
MODULE_DESCRIPTION("i.MX8QM/QXP pixel combiner bridge driver");
|
||||
MODULE_AUTHOR("Liu Ying <victor.liu@nxp.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:" DRIVER_NAME);
|
|
@ -0,0 +1,429 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
/*
|
||||
* Copyright 2020,2022 NXP
|
||||
*/
|
||||
|
||||
#include <linux/firmware/imx/svc/misc.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_graph.h>
|
||||
#include <linux/platform_device.h>
|
||||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
|
||||
#define DRIVER_NAME "imx8qxp-display-pixel-link"
|
||||
#define PL_MAX_MST_ADDR 3
|
||||
#define PL_MAX_NEXT_BRIDGES 2
|
||||
|
||||
struct imx8qxp_pixel_link {
|
||||
struct drm_bridge bridge;
|
||||
struct drm_bridge *next_bridge;
|
||||
struct device *dev;
|
||||
struct imx_sc_ipc *ipc_handle;
|
||||
u8 stream_id;
|
||||
u8 dc_id;
|
||||
u32 sink_rsc;
|
||||
u32 mst_addr;
|
||||
u8 mst_addr_ctrl;
|
||||
u8 mst_en_ctrl;
|
||||
u8 mst_vld_ctrl;
|
||||
u8 sync_ctrl;
|
||||
};
|
||||
|
||||
static void imx8qxp_pixel_link_enable_mst_en(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->mst_en_ctrl, true);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to enable DC%u stream%u pixel link mst_en: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
}
|
||||
|
||||
static void imx8qxp_pixel_link_enable_mst_vld(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->mst_vld_ctrl, true);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to enable DC%u stream%u pixel link mst_vld: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
}
|
||||
|
||||
static void imx8qxp_pixel_link_enable_sync(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->sync_ctrl, true);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to enable DC%u stream%u pixel link sync: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_disable_mst_en(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->mst_en_ctrl, false);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to disable DC%u stream%u pixel link mst_en: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_disable_mst_vld(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->mst_vld_ctrl, false);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to disable DC%u stream%u pixel link mst_vld: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_disable_sync(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle, pl->sink_rsc,
|
||||
pl->sync_ctrl, false);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to disable DC%u stream%u pixel link sync: %d\n",
|
||||
pl->dc_id, pl->stream_id, ret);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void imx8qxp_pixel_link_set_mst_addr(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx_sc_misc_set_control(pl->ipc_handle,
|
||||
pl->sink_rsc, pl->mst_addr_ctrl,
|
||||
pl->mst_addr);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"failed to set DC%u stream%u pixel link mst addr(%u): %d\n",
|
||||
pl->dc_id, pl->stream_id, pl->mst_addr, ret);
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_bridge_attach(struct drm_bridge *bridge,
|
||||
enum drm_bridge_attach_flags flags)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl = bridge->driver_private;
|
||||
|
||||
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
|
||||
DRM_DEV_ERROR(pl->dev,
|
||||
"do not support creating a drm_connector\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (!bridge->encoder) {
|
||||
DRM_DEV_ERROR(pl->dev, "missing encoder\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
return drm_bridge_attach(bridge->encoder,
|
||||
pl->next_bridge, bridge,
|
||||
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pixel_link_bridge_mode_set(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl = bridge->driver_private;
|
||||
|
||||
imx8qxp_pixel_link_set_mst_addr(pl);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pixel_link_bridge_atomic_enable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl = bridge->driver_private;
|
||||
|
||||
imx8qxp_pixel_link_enable_mst_en(pl);
|
||||
imx8qxp_pixel_link_enable_mst_vld(pl);
|
||||
imx8qxp_pixel_link_enable_sync(pl);
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pixel_link_bridge_atomic_disable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl = bridge->driver_private;
|
||||
|
||||
imx8qxp_pixel_link_disable_mst_en(pl);
|
||||
imx8qxp_pixel_link_disable_mst_vld(pl);
|
||||
imx8qxp_pixel_link_disable_sync(pl);
|
||||
}
|
||||
|
||||
static const u32 imx8qxp_pixel_link_bus_output_fmts[] = {
|
||||
MEDIA_BUS_FMT_RGB888_1X36_CPADLO,
|
||||
MEDIA_BUS_FMT_RGB666_1X36_CPADLO,
|
||||
};
|
||||
|
||||
static bool imx8qxp_pixel_link_bus_output_fmt_supported(u32 fmt)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(imx8qxp_pixel_link_bus_output_fmts); i++) {
|
||||
if (imx8qxp_pixel_link_bus_output_fmts[i] == fmt)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pixel_link_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
u32 output_fmt,
|
||||
unsigned int *num_input_fmts)
|
||||
{
|
||||
u32 *input_fmts;
|
||||
|
||||
if (!imx8qxp_pixel_link_bus_output_fmt_supported(output_fmt))
|
||||
return NULL;
|
||||
|
||||
*num_input_fmts = 1;
|
||||
|
||||
input_fmts = kmalloc(sizeof(*input_fmts), GFP_KERNEL);
|
||||
if (!input_fmts)
|
||||
return NULL;
|
||||
|
||||
input_fmts[0] = output_fmt;
|
||||
|
||||
return input_fmts;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pixel_link_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
unsigned int *num_output_fmts)
|
||||
{
|
||||
*num_output_fmts = ARRAY_SIZE(imx8qxp_pixel_link_bus_output_fmts);
|
||||
return kmemdup(imx8qxp_pixel_link_bus_output_fmts,
|
||||
sizeof(imx8qxp_pixel_link_bus_output_fmts), GFP_KERNEL);
|
||||
}
|
||||
|
||||
static const struct drm_bridge_funcs imx8qxp_pixel_link_bridge_funcs = {
|
||||
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
||||
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
||||
.atomic_reset = drm_atomic_helper_bridge_reset,
|
||||
.attach = imx8qxp_pixel_link_bridge_attach,
|
||||
.mode_set = imx8qxp_pixel_link_bridge_mode_set,
|
||||
.atomic_enable = imx8qxp_pixel_link_bridge_atomic_enable,
|
||||
.atomic_disable = imx8qxp_pixel_link_bridge_atomic_disable,
|
||||
.atomic_get_input_bus_fmts =
|
||||
imx8qxp_pixel_link_bridge_atomic_get_input_bus_fmts,
|
||||
.atomic_get_output_bus_fmts =
|
||||
imx8qxp_pixel_link_bridge_atomic_get_output_bus_fmts,
|
||||
};
|
||||
|
||||
static int imx8qxp_pixel_link_disable_all_controls(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = imx8qxp_pixel_link_disable_mst_en(pl);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = imx8qxp_pixel_link_disable_mst_vld(pl);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return imx8qxp_pixel_link_disable_sync(pl);
|
||||
}
|
||||
|
||||
static struct drm_bridge *
|
||||
imx8qxp_pixel_link_find_next_bridge(struct imx8qxp_pixel_link *pl)
|
||||
{
|
||||
struct device_node *np = pl->dev->of_node;
|
||||
struct device_node *port, *remote;
|
||||
struct drm_bridge *next_bridge[PL_MAX_NEXT_BRIDGES];
|
||||
u32 port_id;
|
||||
bool found_port = false;
|
||||
int reg, ep_cnt = 0;
|
||||
/* select the first next bridge by default */
|
||||
int bridge_sel = 0;
|
||||
|
||||
for (port_id = 1; port_id <= PL_MAX_MST_ADDR + 1; port_id++) {
|
||||
port = of_graph_get_port_by_id(np, port_id);
|
||||
if (!port)
|
||||
continue;
|
||||
|
||||
if (of_device_is_available(port)) {
|
||||
found_port = true;
|
||||
of_node_put(port);
|
||||
break;
|
||||
}
|
||||
|
||||
of_node_put(port);
|
||||
}
|
||||
|
||||
if (!found_port) {
|
||||
DRM_DEV_ERROR(pl->dev, "no available output port\n");
|
||||
return ERR_PTR(-ENODEV);
|
||||
}
|
||||
|
||||
for (reg = 0; reg < PL_MAX_NEXT_BRIDGES; reg++) {
|
||||
remote = of_graph_get_remote_node(np, port_id, reg);
|
||||
if (!remote)
|
||||
continue;
|
||||
|
||||
if (!of_device_is_available(remote->parent)) {
|
||||
DRM_DEV_DEBUG(pl->dev,
|
||||
"port%u endpoint%u remote parent is not available\n",
|
||||
port_id, reg);
|
||||
of_node_put(remote);
|
||||
continue;
|
||||
}
|
||||
|
||||
next_bridge[ep_cnt] = of_drm_find_bridge(remote);
|
||||
if (!next_bridge[ep_cnt]) {
|
||||
of_node_put(remote);
|
||||
return ERR_PTR(-EPROBE_DEFER);
|
||||
}
|
||||
|
||||
/* specially select the next bridge with companion PXL2DPI */
|
||||
if (of_find_property(remote, "fsl,companion-pxl2dpi", NULL))
|
||||
bridge_sel = ep_cnt;
|
||||
|
||||
ep_cnt++;
|
||||
|
||||
of_node_put(remote);
|
||||
}
|
||||
|
||||
pl->mst_addr = port_id - 1;
|
||||
|
||||
return next_bridge[bridge_sel];
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_bridge_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl;
|
||||
struct device *dev = &pdev->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
int ret;
|
||||
|
||||
pl = devm_kzalloc(dev, sizeof(*pl), GFP_KERNEL);
|
||||
if (!pl)
|
||||
return -ENOMEM;
|
||||
|
||||
ret = imx_scu_get_handle(&pl->ipc_handle);
|
||||
if (ret) {
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get SCU ipc handle: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = of_property_read_u8(np, "fsl,dc-id", &pl->dc_id);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(dev, "failed to get DC index: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = of_property_read_u8(np, "fsl,dc-stream-id", &pl->stream_id);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(dev, "failed to get DC stream index: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
pl->dev = dev;
|
||||
|
||||
pl->sink_rsc = pl->dc_id ? IMX_SC_R_DC_1 : IMX_SC_R_DC_0;
|
||||
|
||||
if (pl->stream_id == 0) {
|
||||
pl->mst_addr_ctrl = IMX_SC_C_PXL_LINK_MST1_ADDR;
|
||||
pl->mst_en_ctrl = IMX_SC_C_PXL_LINK_MST1_ENB;
|
||||
pl->mst_vld_ctrl = IMX_SC_C_PXL_LINK_MST1_VLD;
|
||||
pl->sync_ctrl = IMX_SC_C_SYNC_CTRL0;
|
||||
} else {
|
||||
pl->mst_addr_ctrl = IMX_SC_C_PXL_LINK_MST2_ADDR;
|
||||
pl->mst_en_ctrl = IMX_SC_C_PXL_LINK_MST2_ENB;
|
||||
pl->mst_vld_ctrl = IMX_SC_C_PXL_LINK_MST2_VLD;
|
||||
pl->sync_ctrl = IMX_SC_C_SYNC_CTRL1;
|
||||
}
|
||||
|
||||
/* disable all controls to POR default */
|
||||
ret = imx8qxp_pixel_link_disable_all_controls(pl);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
pl->next_bridge = imx8qxp_pixel_link_find_next_bridge(pl);
|
||||
if (IS_ERR(pl->next_bridge)) {
|
||||
ret = PTR_ERR(pl->next_bridge);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to find next bridge: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
platform_set_drvdata(pdev, pl);
|
||||
|
||||
pl->bridge.driver_private = pl;
|
||||
pl->bridge.funcs = &imx8qxp_pixel_link_bridge_funcs;
|
||||
pl->bridge.of_node = np;
|
||||
|
||||
drm_bridge_add(&pl->bridge);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pixel_link_bridge_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pixel_link *pl = platform_get_drvdata(pdev);
|
||||
|
||||
drm_bridge_remove(&pl->bridge);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct of_device_id imx8qxp_pixel_link_dt_ids[] = {
|
||||
{ .compatible = "fsl,imx8qm-dc-pixel-link", },
|
||||
{ .compatible = "fsl,imx8qxp-dc-pixel-link", },
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx8qxp_pixel_link_dt_ids);
|
||||
|
||||
static struct platform_driver imx8qxp_pixel_link_bridge_driver = {
|
||||
.probe = imx8qxp_pixel_link_bridge_probe,
|
||||
.remove = imx8qxp_pixel_link_bridge_remove,
|
||||
.driver = {
|
||||
.of_match_table = imx8qxp_pixel_link_dt_ids,
|
||||
.name = DRIVER_NAME,
|
||||
},
|
||||
};
|
||||
module_platform_driver(imx8qxp_pixel_link_bridge_driver);
|
||||
|
||||
MODULE_DESCRIPTION("i.MX8QXP/QM display pixel link bridge driver");
|
||||
MODULE_AUTHOR("Liu Ying <victor.liu@nxp.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:" DRIVER_NAME);
|
|
@ -0,0 +1,487 @@
|
|||
// SPDX-License-Identifier: GPL-2.0+
|
||||
|
||||
/*
|
||||
* Copyright 2020 NXP
|
||||
*/
|
||||
|
||||
#include <linux/firmware/imx/svc/misc.h>
|
||||
#include <linux/mfd/syscon.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of_graph.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include <dt-bindings/firmware/imx/rsrc.h>
|
||||
|
||||
#define PXL2DPI_CTRL 0x40
|
||||
#define CFG1_16BIT 0x0
|
||||
#define CFG2_16BIT 0x1
|
||||
#define CFG3_16BIT 0x2
|
||||
#define CFG1_18BIT 0x3
|
||||
#define CFG2_18BIT 0x4
|
||||
#define CFG_24BIT 0x5
|
||||
|
||||
#define DRIVER_NAME "imx8qxp-pxl2dpi"
|
||||
|
||||
struct imx8qxp_pxl2dpi {
|
||||
struct regmap *regmap;
|
||||
struct drm_bridge bridge;
|
||||
struct drm_bridge *next_bridge;
|
||||
struct drm_bridge *companion;
|
||||
struct device *dev;
|
||||
struct imx_sc_ipc *ipc_handle;
|
||||
u32 sc_resource;
|
||||
u32 in_bus_format;
|
||||
u32 out_bus_format;
|
||||
u32 pl_sel;
|
||||
};
|
||||
|
||||
#define bridge_to_p2d(b) container_of(b, struct imx8qxp_pxl2dpi, bridge)
|
||||
|
||||
static int imx8qxp_pxl2dpi_bridge_attach(struct drm_bridge *bridge,
|
||||
enum drm_bridge_attach_flags flags)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d = bridge->driver_private;
|
||||
|
||||
if (!(flags & DRM_BRIDGE_ATTACH_NO_CONNECTOR)) {
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"do not support creating a drm_connector\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
if (!bridge->encoder) {
|
||||
DRM_DEV_ERROR(p2d->dev, "missing encoder\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
return drm_bridge_attach(bridge->encoder,
|
||||
p2d->next_bridge, bridge,
|
||||
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
||||
}
|
||||
|
||||
static int
|
||||
imx8qxp_pxl2dpi_bridge_atomic_check(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d = bridge->driver_private;
|
||||
|
||||
p2d->in_bus_format = bridge_state->input_bus_cfg.format;
|
||||
p2d->out_bus_format = bridge_state->output_bus_cfg.format;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pxl2dpi_bridge_mode_set(struct drm_bridge *bridge,
|
||||
const struct drm_display_mode *mode,
|
||||
const struct drm_display_mode *adjusted_mode)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d = bridge->driver_private;
|
||||
struct imx8qxp_pxl2dpi *companion_p2d;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_get_sync(p2d->dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"failed to get runtime PM sync: %d\n", ret);
|
||||
|
||||
ret = imx_sc_misc_set_control(p2d->ipc_handle, p2d->sc_resource,
|
||||
IMX_SC_C_PXL_LINK_SEL, p2d->pl_sel);
|
||||
if (ret)
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"failed to set pixel link selection(%u): %d\n",
|
||||
p2d->pl_sel, ret);
|
||||
|
||||
switch (p2d->out_bus_format) {
|
||||
case MEDIA_BUS_FMT_RGB888_1X24:
|
||||
regmap_write(p2d->regmap, PXL2DPI_CTRL, CFG_24BIT);
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB666_1X24_CPADHI:
|
||||
regmap_write(p2d->regmap, PXL2DPI_CTRL, CFG2_18BIT);
|
||||
break;
|
||||
default:
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"unsupported output bus format 0x%08x\n",
|
||||
p2d->out_bus_format);
|
||||
}
|
||||
|
||||
if (p2d->companion) {
|
||||
companion_p2d = bridge_to_p2d(p2d->companion);
|
||||
|
||||
companion_p2d->in_bus_format = p2d->in_bus_format;
|
||||
companion_p2d->out_bus_format = p2d->out_bus_format;
|
||||
|
||||
p2d->companion->funcs->mode_set(p2d->companion, mode,
|
||||
adjusted_mode);
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
imx8qxp_pxl2dpi_bridge_atomic_disable(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *old_bridge_state)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d = bridge->driver_private;
|
||||
int ret;
|
||||
|
||||
ret = pm_runtime_put(p2d->dev);
|
||||
if (ret < 0)
|
||||
DRM_DEV_ERROR(p2d->dev, "failed to put runtime PM: %d\n", ret);
|
||||
|
||||
if (p2d->companion)
|
||||
p2d->companion->funcs->atomic_disable(p2d->companion,
|
||||
old_bridge_state);
|
||||
}
|
||||
|
||||
static const u32 imx8qxp_pxl2dpi_bus_output_fmts[] = {
|
||||
MEDIA_BUS_FMT_RGB888_1X24,
|
||||
MEDIA_BUS_FMT_RGB666_1X24_CPADHI,
|
||||
};
|
||||
|
||||
static bool imx8qxp_pxl2dpi_bus_output_fmt_supported(u32 fmt)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(imx8qxp_pxl2dpi_bus_output_fmts); i++) {
|
||||
if (imx8qxp_pxl2dpi_bus_output_fmts[i] == fmt)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pxl2dpi_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
u32 output_fmt,
|
||||
unsigned int *num_input_fmts)
|
||||
{
|
||||
u32 *input_fmts;
|
||||
|
||||
if (!imx8qxp_pxl2dpi_bus_output_fmt_supported(output_fmt))
|
||||
return NULL;
|
||||
|
||||
*num_input_fmts = 1;
|
||||
|
||||
input_fmts = kmalloc(sizeof(*input_fmts), GFP_KERNEL);
|
||||
if (!input_fmts)
|
||||
return NULL;
|
||||
|
||||
switch (output_fmt) {
|
||||
case MEDIA_BUS_FMT_RGB888_1X24:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X36_CPADLO;
|
||||
break;
|
||||
case MEDIA_BUS_FMT_RGB666_1X24_CPADHI:
|
||||
input_fmts[0] = MEDIA_BUS_FMT_RGB666_1X36_CPADLO;
|
||||
break;
|
||||
default:
|
||||
kfree(input_fmts);
|
||||
input_fmts = NULL;
|
||||
break;
|
||||
}
|
||||
|
||||
return input_fmts;
|
||||
}
|
||||
|
||||
static u32 *
|
||||
imx8qxp_pxl2dpi_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge,
|
||||
struct drm_bridge_state *bridge_state,
|
||||
struct drm_crtc_state *crtc_state,
|
||||
struct drm_connector_state *conn_state,
|
||||
unsigned int *num_output_fmts)
|
||||
{
|
||||
*num_output_fmts = ARRAY_SIZE(imx8qxp_pxl2dpi_bus_output_fmts);
|
||||
return kmemdup(imx8qxp_pxl2dpi_bus_output_fmts,
|
||||
sizeof(imx8qxp_pxl2dpi_bus_output_fmts), GFP_KERNEL);
|
||||
}
|
||||
|
||||
static const struct drm_bridge_funcs imx8qxp_pxl2dpi_bridge_funcs = {
|
||||
.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
|
||||
.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
|
||||
.atomic_reset = drm_atomic_helper_bridge_reset,
|
||||
.attach = imx8qxp_pxl2dpi_bridge_attach,
|
||||
.atomic_check = imx8qxp_pxl2dpi_bridge_atomic_check,
|
||||
.mode_set = imx8qxp_pxl2dpi_bridge_mode_set,
|
||||
.atomic_disable = imx8qxp_pxl2dpi_bridge_atomic_disable,
|
||||
.atomic_get_input_bus_fmts =
|
||||
imx8qxp_pxl2dpi_bridge_atomic_get_input_bus_fmts,
|
||||
.atomic_get_output_bus_fmts =
|
||||
imx8qxp_pxl2dpi_bridge_atomic_get_output_bus_fmts,
|
||||
};
|
||||
|
||||
static struct device_node *
|
||||
imx8qxp_pxl2dpi_get_available_ep_from_port(struct imx8qxp_pxl2dpi *p2d,
|
||||
u32 port_id)
|
||||
{
|
||||
struct device_node *port, *ep;
|
||||
int ep_cnt;
|
||||
|
||||
port = of_graph_get_port_by_id(p2d->dev->of_node, port_id);
|
||||
if (!port) {
|
||||
DRM_DEV_ERROR(p2d->dev, "failed to get port@%u\n", port_id);
|
||||
return ERR_PTR(-ENODEV);
|
||||
}
|
||||
|
||||
ep_cnt = of_get_available_child_count(port);
|
||||
if (ep_cnt == 0) {
|
||||
DRM_DEV_ERROR(p2d->dev, "no available endpoints of port@%u\n",
|
||||
port_id);
|
||||
ep = ERR_PTR(-ENODEV);
|
||||
goto out;
|
||||
} else if (ep_cnt > 1) {
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"invalid available endpoints of port@%u\n",
|
||||
port_id);
|
||||
ep = ERR_PTR(-EINVAL);
|
||||
goto out;
|
||||
}
|
||||
|
||||
ep = of_get_next_available_child(port, NULL);
|
||||
if (!ep) {
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"failed to get available endpoint of port@%u\n",
|
||||
port_id);
|
||||
ep = ERR_PTR(-ENODEV);
|
||||
goto out;
|
||||
}
|
||||
out:
|
||||
of_node_put(port);
|
||||
return ep;
|
||||
}
|
||||
|
||||
static struct drm_bridge *
|
||||
imx8qxp_pxl2dpi_find_next_bridge(struct imx8qxp_pxl2dpi *p2d)
|
||||
{
|
||||
struct device_node *ep, *remote;
|
||||
struct drm_bridge *next_bridge;
|
||||
int ret;
|
||||
|
||||
ep = imx8qxp_pxl2dpi_get_available_ep_from_port(p2d, 1);
|
||||
if (IS_ERR(ep)) {
|
||||
ret = PTR_ERR(ep);
|
||||
return ERR_PTR(ret);
|
||||
}
|
||||
|
||||
remote = of_graph_get_remote_port_parent(ep);
|
||||
if (!remote || !of_device_is_available(remote)) {
|
||||
DRM_DEV_ERROR(p2d->dev, "no available remote\n");
|
||||
next_bridge = ERR_PTR(-ENODEV);
|
||||
goto out;
|
||||
} else if (!of_device_is_available(remote->parent)) {
|
||||
DRM_DEV_ERROR(p2d->dev, "remote parent is not available\n");
|
||||
next_bridge = ERR_PTR(-ENODEV);
|
||||
goto out;
|
||||
}
|
||||
|
||||
next_bridge = of_drm_find_bridge(remote);
|
||||
if (!next_bridge) {
|
||||
next_bridge = ERR_PTR(-EPROBE_DEFER);
|
||||
goto out;
|
||||
}
|
||||
out:
|
||||
of_node_put(remote);
|
||||
of_node_put(ep);
|
||||
|
||||
return next_bridge;
|
||||
}
|
||||
|
||||
static int imx8qxp_pxl2dpi_set_pixel_link_sel(struct imx8qxp_pxl2dpi *p2d)
|
||||
{
|
||||
struct device_node *ep;
|
||||
struct of_endpoint endpoint;
|
||||
int ret;
|
||||
|
||||
ep = imx8qxp_pxl2dpi_get_available_ep_from_port(p2d, 0);
|
||||
if (IS_ERR(ep))
|
||||
return PTR_ERR(ep);
|
||||
|
||||
ret = of_graph_parse_endpoint(ep, &endpoint);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(p2d->dev,
|
||||
"failed to parse endpoint of port@0: %d\n", ret);
|
||||
goto out;
|
||||
}
|
||||
|
||||
p2d->pl_sel = endpoint.id;
|
||||
out:
|
||||
of_node_put(ep);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pxl2dpi_parse_dt_companion(struct imx8qxp_pxl2dpi *p2d)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *companion_p2d;
|
||||
struct device *dev = p2d->dev;
|
||||
struct device_node *companion;
|
||||
struct device_node *port1, *port2;
|
||||
const struct of_device_id *match;
|
||||
int dual_link;
|
||||
int ret = 0;
|
||||
|
||||
/* Locate the companion PXL2DPI for dual-link operation, if any. */
|
||||
companion = of_parse_phandle(dev->of_node, "fsl,companion-pxl2dpi", 0);
|
||||
if (!companion)
|
||||
return 0;
|
||||
|
||||
if (!of_device_is_available(companion)) {
|
||||
DRM_DEV_ERROR(dev, "companion PXL2DPI is not available\n");
|
||||
ret = -ENODEV;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/*
|
||||
* Sanity check: the companion bridge must have the same compatible
|
||||
* string.
|
||||
*/
|
||||
match = of_match_device(dev->driver->of_match_table, dev);
|
||||
if (!of_device_is_compatible(companion, match->compatible)) {
|
||||
DRM_DEV_ERROR(dev, "companion PXL2DPI is incompatible\n");
|
||||
ret = -ENXIO;
|
||||
goto out;
|
||||
}
|
||||
|
||||
p2d->companion = of_drm_find_bridge(companion);
|
||||
if (!p2d->companion) {
|
||||
ret = -EPROBE_DEFER;
|
||||
DRM_DEV_DEBUG_DRIVER(p2d->dev,
|
||||
"failed to find companion bridge: %d\n",
|
||||
ret);
|
||||
goto out;
|
||||
}
|
||||
|
||||
companion_p2d = bridge_to_p2d(p2d->companion);
|
||||
|
||||
/*
|
||||
* We need to work out if the sink is expecting us to function in
|
||||
* dual-link mode. We do this by looking at the DT port nodes that
|
||||
* the next bridges are connected to. If they are marked as expecting
|
||||
* even pixels and odd pixels than we need to use the companion PXL2DPI.
|
||||
*/
|
||||
port1 = of_graph_get_port_by_id(p2d->next_bridge->of_node, 1);
|
||||
port2 = of_graph_get_port_by_id(companion_p2d->next_bridge->of_node, 1);
|
||||
dual_link = drm_of_lvds_get_dual_link_pixel_order(port1, port2);
|
||||
of_node_put(port1);
|
||||
of_node_put(port2);
|
||||
|
||||
if (dual_link < 0) {
|
||||
ret = dual_link;
|
||||
DRM_DEV_ERROR(dev, "failed to get dual link pixel order: %d\n",
|
||||
ret);
|
||||
goto out;
|
||||
}
|
||||
|
||||
DRM_DEV_DEBUG_DRIVER(dev,
|
||||
"dual-link configuration detected (companion bridge %pOF)\n",
|
||||
companion);
|
||||
out:
|
||||
of_node_put(companion);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pxl2dpi_bridge_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d;
|
||||
struct device *dev = &pdev->dev;
|
||||
struct device_node *np = dev->of_node;
|
||||
int ret;
|
||||
|
||||
p2d = devm_kzalloc(dev, sizeof(*p2d), GFP_KERNEL);
|
||||
if (!p2d)
|
||||
return -ENOMEM;
|
||||
|
||||
p2d->regmap = syscon_node_to_regmap(np->parent);
|
||||
if (IS_ERR(p2d->regmap)) {
|
||||
ret = PTR_ERR(p2d->regmap);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get regmap: %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = imx_scu_get_handle(&p2d->ipc_handle);
|
||||
if (ret) {
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to get SCU ipc handle: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
p2d->dev = dev;
|
||||
|
||||
ret = of_property_read_u32(np, "fsl,sc-resource", &p2d->sc_resource);
|
||||
if (ret) {
|
||||
DRM_DEV_ERROR(dev, "failed to get SC resource %d\n", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
p2d->next_bridge = imx8qxp_pxl2dpi_find_next_bridge(p2d);
|
||||
if (IS_ERR(p2d->next_bridge)) {
|
||||
ret = PTR_ERR(p2d->next_bridge);
|
||||
if (ret != -EPROBE_DEFER)
|
||||
DRM_DEV_ERROR(dev, "failed to find next bridge: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = imx8qxp_pxl2dpi_set_pixel_link_sel(p2d);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = imx8qxp_pxl2dpi_parse_dt_companion(p2d);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
platform_set_drvdata(pdev, p2d);
|
||||
pm_runtime_enable(dev);
|
||||
|
||||
p2d->bridge.driver_private = p2d;
|
||||
p2d->bridge.funcs = &imx8qxp_pxl2dpi_bridge_funcs;
|
||||
p2d->bridge.of_node = np;
|
||||
|
||||
drm_bridge_add(&p2d->bridge);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int imx8qxp_pxl2dpi_bridge_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct imx8qxp_pxl2dpi *p2d = platform_get_drvdata(pdev);
|
||||
|
||||
drm_bridge_remove(&p2d->bridge);
|
||||
|
||||
pm_runtime_disable(&pdev->dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct of_device_id imx8qxp_pxl2dpi_dt_ids[] = {
|
||||
{ .compatible = "fsl,imx8qxp-pxl2dpi", },
|
||||
{ /* sentinel */ }
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx8qxp_pxl2dpi_dt_ids);
|
||||
|
||||
static struct platform_driver imx8qxp_pxl2dpi_bridge_driver = {
|
||||
.probe = imx8qxp_pxl2dpi_bridge_probe,
|
||||
.remove = imx8qxp_pxl2dpi_bridge_remove,
|
||||
.driver = {
|
||||
.of_match_table = imx8qxp_pxl2dpi_dt_ids,
|
||||
.name = DRIVER_NAME,
|
||||
},
|
||||
};
|
||||
module_platform_driver(imx8qxp_pxl2dpi_bridge_driver);
|
||||
|
||||
MODULE_DESCRIPTION("i.MX8QXP pixel link to DPI bridge driver");
|
||||
MODULE_AUTHOR("Liu Ying <victor.liu@nxp.com>");
|
||||
MODULE_LICENSE("GPL v2");
|
||||
MODULE_ALIAS("platform:" DRIVER_NAME);
|
|
@ -11,6 +11,7 @@
|
|||
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_mipi_dsi.h>
|
||||
#include <drm/drm_of.h>
|
||||
|
||||
|
|
|
@ -170,6 +170,19 @@ static const struct drm_bridge_funcs panel_bridge_bridge_funcs = {
|
|||
.debugfs_init = panel_bridge_debugfs_init,
|
||||
};
|
||||
|
||||
/**
|
||||
* drm_bridge_is_panel - Checks if a drm_bridge is a panel_bridge.
|
||||
*
|
||||
* @bridge: The drm_bridge to be checked.
|
||||
*
|
||||
* Returns true if the bridge is a panel bridge, or false otherwise.
|
||||
*/
|
||||
bool drm_bridge_is_panel(const struct drm_bridge *bridge)
|
||||
{
|
||||
return bridge->funcs == &panel_bridge_bridge_funcs;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_bridge_is_panel);
|
||||
|
||||
/**
|
||||
* drm_panel_bridge_add - Creates a &drm_bridge and &drm_connector that
|
||||
* just calls the appropriate functions from &drm_panel.
|
||||
|
@ -269,6 +282,27 @@ void drm_panel_bridge_remove(struct drm_bridge *bridge)
|
|||
}
|
||||
EXPORT_SYMBOL(drm_panel_bridge_remove);
|
||||
|
||||
/**
|
||||
* drm_panel_bridge_set_orientation - Set the connector's panel orientation
|
||||
* from the bridge that can be transformed to panel bridge.
|
||||
*
|
||||
* @connector: The connector to be set panel orientation.
|
||||
* @bridge: The drm_bridge to be transformed to panel bridge.
|
||||
*
|
||||
* Returns 0 on success, negative errno on failure.
|
||||
*/
|
||||
int drm_panel_bridge_set_orientation(struct drm_connector *connector,
|
||||
struct drm_bridge *bridge)
|
||||
{
|
||||
struct panel_bridge *panel_bridge;
|
||||
|
||||
panel_bridge = drm_bridge_to_panel_bridge(bridge);
|
||||
|
||||
return drm_connector_set_orientation_from_panel(connector,
|
||||
panel_bridge->panel);
|
||||
}
|
||||
EXPORT_SYMBOL(drm_panel_bridge_set_orientation);
|
||||
|
||||
static void devm_drm_panel_bridge_release(struct device *dev, void *res)
|
||||
{
|
||||
struct drm_bridge **bridge = res;
|
||||
|
|
|
@ -16,6 +16,7 @@
|
|||
#include <drm/display/drm_dp_aux_bus.h>
|
||||
#include <drm/display/drm_dp_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_mipi_dsi.h>
|
||||
#include <drm/drm_of.h>
|
||||
#include <drm/drm_panel.h>
|
||||
|
@ -168,23 +169,35 @@ static bool ps8640_of_panel_on_aux_bus(struct device *dev)
|
|||
return true;
|
||||
}
|
||||
|
||||
static int ps8640_ensure_hpd(struct ps8640 *ps_bridge)
|
||||
static int _ps8640_wait_hpd_asserted(struct ps8640 *ps_bridge, unsigned long wait_us)
|
||||
{
|
||||
struct regmap *map = ps_bridge->regmap[PAGE2_TOP_CNTL];
|
||||
struct device *dev = &ps_bridge->page[PAGE2_TOP_CNTL]->dev;
|
||||
int status;
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* Apparently something about the firmware in the chip signals that
|
||||
* HPD goes high by reporting GPIO9 as high (even though HPD isn't
|
||||
* actually connected to GPIO9).
|
||||
*/
|
||||
ret = regmap_read_poll_timeout(map, PAGE2_GPIO_H, status,
|
||||
status & PS_GPIO9, 20 * 1000, 200 * 1000);
|
||||
return regmap_read_poll_timeout(map, PAGE2_GPIO_H, status,
|
||||
status & PS_GPIO9, wait_us / 10, wait_us);
|
||||
}
|
||||
|
||||
if (ret < 0)
|
||||
dev_warn(dev, "HPD didn't go high: %d\n", ret);
|
||||
static int ps8640_wait_hpd_asserted(struct drm_dp_aux *aux, unsigned long wait_us)
|
||||
{
|
||||
struct ps8640 *ps_bridge = aux_to_ps8640(aux);
|
||||
struct device *dev = &ps_bridge->page[PAGE0_DP_CNTL]->dev;
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* Note that this function is called by code that has already powered
|
||||
* the panel. We have to power ourselves up but we don't need to worry
|
||||
* about powering the panel.
|
||||
*/
|
||||
pm_runtime_get_sync(dev);
|
||||
ret = _ps8640_wait_hpd_asserted(ps_bridge, wait_us);
|
||||
pm_runtime_mark_last_busy(dev);
|
||||
pm_runtime_put_autosuspend(dev);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
@ -323,9 +336,7 @@ static ssize_t ps8640_aux_transfer(struct drm_dp_aux *aux,
|
|||
int ret;
|
||||
|
||||
pm_runtime_get_sync(dev);
|
||||
ret = ps8640_ensure_hpd(ps_bridge);
|
||||
if (!ret)
|
||||
ret = ps8640_aux_transfer_msg(aux, msg);
|
||||
ret = ps8640_aux_transfer_msg(aux, msg);
|
||||
pm_runtime_mark_last_busy(dev);
|
||||
pm_runtime_put_autosuspend(dev);
|
||||
|
||||
|
@ -369,8 +380,8 @@ static int __maybe_unused ps8640_resume(struct device *dev)
|
|||
* Mystery 200 ms delay for the "MCU to be ready". It's unclear if
|
||||
* this is truly necessary since the MCU will already signal that
|
||||
* things are "good to go" by signaling HPD on "gpio 9". See
|
||||
* ps8640_ensure_hpd(). For now we'll keep this mystery delay just in
|
||||
* case.
|
||||
* _ps8640_wait_hpd_asserted(). For now we'll keep this mystery delay
|
||||
* just in case.
|
||||
*/
|
||||
msleep(200);
|
||||
|
||||
|
@ -406,7 +417,9 @@ static void ps8640_pre_enable(struct drm_bridge *bridge)
|
|||
int ret;
|
||||
|
||||
pm_runtime_get_sync(dev);
|
||||
ps8640_ensure_hpd(ps_bridge);
|
||||
ret = _ps8640_wait_hpd_asserted(ps_bridge, 200 * 1000);
|
||||
if (ret < 0)
|
||||
dev_warn(dev, "HPD didn't go high: %d\n", ret);
|
||||
|
||||
/*
|
||||
* The Manufacturer Command Set (MCS) is a device dependent interface
|
||||
|
@ -682,6 +695,7 @@ static int ps8640_probe(struct i2c_client *client)
|
|||
ps_bridge->aux.name = "parade-ps8640-aux";
|
||||
ps_bridge->aux.dev = dev;
|
||||
ps_bridge->aux.transfer = ps8640_aux_transfer;
|
||||
ps_bridge->aux.wait_hpd_asserted = ps8640_wait_hpd_asserted;
|
||||
drm_dp_aux_init(&ps_bridge->aux);
|
||||
|
||||
pm_runtime_enable(dev);
|
||||
|
|
|
@ -15,6 +15,7 @@
|
|||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
||||
|
|
|
@ -339,6 +339,7 @@ static void d2l_read(struct i2c_client *i2c, u16 addr, u32 *val)
|
|||
goto fail;
|
||||
|
||||
pr_debug("d2l: I2C : addr:%04x value:%08x\n", addr, *val);
|
||||
return;
|
||||
|
||||
fail:
|
||||
dev_err(&i2c->dev, "Error %d reading from subaddress 0x%x\n",
|
||||
|
@ -429,7 +430,7 @@ static void tc_bridge_enable(struct drm_bridge *bridge)
|
|||
val = TC358775_VPCTRL_MSF(1);
|
||||
|
||||
dsiclk = mode->crtc_clock * 3 * tc->bpc / tc->num_dsi_lanes / 1000;
|
||||
clkdiv = dsiclk / DIVIDE_BY_3 * tc->lvds_link;
|
||||
clkdiv = dsiclk / (tc->lvds_link == DUAL_LINK ? DIVIDE_BY_6 : DIVIDE_BY_3);
|
||||
byteclk = dsiclk / 4;
|
||||
t1 = hactive * (tc->bpc * 3 / 8) / tc->num_dsi_lanes;
|
||||
t2 = ((100000 / clkdiv)) * (hactive + hback_porch + hsync_len + hfront_porch) / 1000;
|
||||
|
|
|
@ -344,7 +344,7 @@ static void sn65dsi83_atomic_enable(struct drm_bridge *bridge,
|
|||
}
|
||||
|
||||
/* Deassert reset */
|
||||
gpiod_set_value(ctx->enable_gpio, 1);
|
||||
gpiod_set_value_cansleep(ctx->enable_gpio, 1);
|
||||
usleep_range(1000, 1100);
|
||||
|
||||
/* Get the LVDS format from the bridge state. */
|
||||
|
@ -500,7 +500,7 @@ static void sn65dsi83_atomic_disable(struct drm_bridge *bridge,
|
|||
int ret;
|
||||
|
||||
/* Put the chip in reset, pull EN line low, and assure 10ms reset low timing. */
|
||||
gpiod_set_value(ctx->enable_gpio, 0);
|
||||
gpiod_set_value_cansleep(ctx->enable_gpio, 0);
|
||||
usleep_range(10000, 11000);
|
||||
|
||||
ret = regulator_disable(ctx->vcc);
|
||||
|
@ -677,7 +677,7 @@ static int sn65dsi83_probe(struct i2c_client *client,
|
|||
ctx->enable_gpio = devm_gpiod_get_optional(ctx->dev, "enable",
|
||||
GPIOD_OUT_LOW);
|
||||
if (IS_ERR(ctx->enable_gpio))
|
||||
return PTR_ERR(ctx->enable_gpio);
|
||||
return dev_err_probe(dev, PTR_ERR(ctx->enable_gpio), "failed to get enable GPIO\n");
|
||||
|
||||
usleep_range(10000, 11000);
|
||||
|
||||
|
@ -687,7 +687,7 @@ static int sn65dsi83_probe(struct i2c_client *client,
|
|||
|
||||
ctx->regmap = devm_regmap_init_i2c(client, &sn65dsi83_regmap_config);
|
||||
if (IS_ERR(ctx->regmap))
|
||||
return PTR_ERR(ctx->regmap);
|
||||
return dev_err_probe(dev, PTR_ERR(ctx->regmap), "failed to get regmap\n");
|
||||
|
||||
dev_set_drvdata(dev, ctx);
|
||||
i2c_set_clientdata(client, ctx);
|
||||
|
|
|
@ -14,6 +14,7 @@
|
|||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
||||
|
|
|
@ -32,6 +32,7 @@
|
|||
|
||||
#include <drm/display/drm_dp_helper.h>
|
||||
#include <drm/display/drm_dp_mst_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
#include <drm/drm_panel.h>
|
||||
|
@ -1597,7 +1598,7 @@ static int drm_dp_aux_reply_duration(const struct drm_dp_aux_msg *msg)
|
|||
* Calculate the length of the i2c transfer in usec, assuming
|
||||
* the i2c bus speed is as specified. Gives the the "worst"
|
||||
* case estimate, ie. successful while as long as possible.
|
||||
* Doesn't account the the "MOT" bit, and instead assumes each
|
||||
* Doesn't account the "MOT" bit, and instead assumes each
|
||||
* message includes a START, ADDRESS and STOP. Neither does it
|
||||
* account for additional random variables such as clock stretching.
|
||||
*/
|
||||
|
|
|
@ -42,6 +42,7 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
||||
|
@ -2666,24 +2667,14 @@ static int drm_dp_check_and_send_link_address(struct drm_dp_mst_topology_mgr *mg
|
|||
}
|
||||
|
||||
list_for_each_entry(port, &mstb->ports, next) {
|
||||
struct drm_dp_mst_branch *mstb_child = NULL;
|
||||
|
||||
if (port->input || !port->ddps)
|
||||
if (port->input || !port->ddps || !port->mstb)
|
||||
continue;
|
||||
|
||||
if (port->mstb)
|
||||
mstb_child = drm_dp_mst_topology_get_mstb_validated(
|
||||
mgr, port->mstb);
|
||||
|
||||
if (mstb_child) {
|
||||
ret = drm_dp_check_and_send_link_address(mgr,
|
||||
mstb_child);
|
||||
drm_dp_mst_topology_put_mstb(mstb_child);
|
||||
if (ret == 1)
|
||||
changed = true;
|
||||
else if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
ret = drm_dp_check_and_send_link_address(mgr, port->mstb);
|
||||
if (ret == 1)
|
||||
changed = true;
|
||||
else if (ret < 0)
|
||||
return ret;
|
||||
}
|
||||
|
||||
return changed;
|
||||
|
@ -5465,8 +5456,7 @@ EXPORT_SYMBOL(drm_dp_mst_topology_state_funcs);
|
|||
*
|
||||
* This function wraps drm_atomic_get_priv_obj_state() passing in the MST atomic
|
||||
* state vtable so that the private object state returned is that of a MST
|
||||
* topology object. Also, drm_atomic_get_private_obj_state() expects the caller
|
||||
* to care of the locking, so warn if don't hold the connection_mutex.
|
||||
* topology object.
|
||||
*
|
||||
* RETURNS:
|
||||
*
|
||||
|
|
|
@ -329,7 +329,20 @@ int drm_aperture_remove_conflicting_pci_framebuffers(struct pci_dev *pdev,
|
|||
const struct drm_driver *req_driver)
|
||||
{
|
||||
resource_size_t base, size;
|
||||
int bar, ret = 0;
|
||||
int bar, ret;
|
||||
|
||||
/*
|
||||
* WARNING: Apparently we must kick fbdev drivers before vgacon,
|
||||
* otherwise the vga fbdev driver falls over.
|
||||
*/
|
||||
#if IS_REACHABLE(CONFIG_FB)
|
||||
ret = remove_conflicting_pci_framebuffers(pdev, req_driver->name);
|
||||
if (ret)
|
||||
return ret;
|
||||
#endif
|
||||
ret = vga_remove_vgacon(pdev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
for (bar = 0; bar < PCI_STD_NUM_BARS; ++bar) {
|
||||
if (!(pci_resource_flags(pdev, bar) & IORESOURCE_MEM))
|
||||
|
@ -339,15 +352,6 @@ int drm_aperture_remove_conflicting_pci_framebuffers(struct pci_dev *pdev,
|
|||
drm_aperture_detach_drivers(base, size);
|
||||
}
|
||||
|
||||
/*
|
||||
* WARNING: Apparently we must kick fbdev drivers before vgacon,
|
||||
* otherwise the vga fbdev driver falls over.
|
||||
*/
|
||||
#if IS_REACHABLE(CONFIG_FB)
|
||||
ret = remove_conflicting_pci_framebuffers(pdev, req_driver->name);
|
||||
#endif
|
||||
if (ret == 0)
|
||||
ret = vga_remove_vgacon(pdev);
|
||||
return ret;
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_aperture_remove_conflicting_pci_framebuffers);
|
||||
|
|
|
@ -31,12 +31,14 @@
|
|||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_debugfs.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_file.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_mode.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_writeback.h>
|
||||
|
|
|
@ -31,10 +31,12 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_damage_helper.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_atomic_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
|
|
@ -26,10 +26,12 @@
|
|||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_bridge.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_plane.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
|
|
@ -29,6 +29,7 @@
|
|||
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_writeback.h>
|
||||
|
|
|
@ -331,7 +331,7 @@ struct drm_connector *drm_bridge_connector_init(struct drm_device *drm,
|
|||
struct drm_bridge_connector *bridge_connector;
|
||||
struct drm_connector *connector;
|
||||
struct i2c_adapter *ddc = NULL;
|
||||
struct drm_bridge *bridge;
|
||||
struct drm_bridge *bridge, *panel_bridge = NULL;
|
||||
int connector_type;
|
||||
|
||||
bridge_connector = kzalloc(sizeof(*bridge_connector), GFP_KERNEL);
|
||||
|
@ -373,6 +373,9 @@ struct drm_connector *drm_bridge_connector_init(struct drm_device *drm,
|
|||
|
||||
if (bridge->ddc)
|
||||
ddc = bridge->ddc;
|
||||
|
||||
if (drm_bridge_is_panel(bridge))
|
||||
panel_bridge = bridge;
|
||||
}
|
||||
|
||||
if (connector_type == DRM_MODE_CONNECTOR_Unknown) {
|
||||
|
@ -392,6 +395,9 @@ struct drm_connector *drm_bridge_connector_init(struct drm_device *drm,
|
|||
connector->polled = DRM_CONNECTOR_POLL_CONNECT
|
||||
| DRM_CONNECTOR_POLL_DISCONNECT;
|
||||
|
||||
if (panel_bridge)
|
||||
drm_panel_bridge_set_orientation(connector, panel_bridge);
|
||||
|
||||
return connector;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(drm_bridge_connector_init);
|
||||
|
|
|
@ -19,6 +19,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_panel.h>
|
||||
#include <drm/drm_utils.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_drv.h>
|
||||
|
@ -2320,6 +2321,9 @@ EXPORT_SYMBOL(drm_connector_set_vrr_capable_property);
|
|||
* It is allowed to call this function with a panel_orientation of
|
||||
* DRM_MODE_PANEL_ORIENTATION_UNKNOWN, in which case it is a no-op.
|
||||
*
|
||||
* The function shouldn't be called in panel after drm is registered (i.e.
|
||||
* drm_dev_register() is called in drm).
|
||||
*
|
||||
* Returns:
|
||||
* Zero on success, negative errno on failure.
|
||||
*/
|
||||
|
@ -2389,6 +2393,33 @@ int drm_connector_set_panel_orientation_with_quirk(
|
|||
}
|
||||
EXPORT_SYMBOL(drm_connector_set_panel_orientation_with_quirk);
|
||||
|
||||
/**
|
||||
* drm_connector_set_orientation_from_panel -
|
||||
* set the connector's panel_orientation from panel's callback.
|
||||
* @connector: connector for which to init the panel-orientation property.
|
||||
* @panel: panel that can provide orientation information.
|
||||
*
|
||||
* Drm drivers should call this function before drm_dev_register().
|
||||
* Orientation is obtained from panel's .get_orientation() callback.
|
||||
*
|
||||
* Returns:
|
||||
* Zero on success, negative errno on failure.
|
||||
*/
|
||||
int drm_connector_set_orientation_from_panel(
|
||||
struct drm_connector *connector,
|
||||
struct drm_panel *panel)
|
||||
{
|
||||
enum drm_panel_orientation orientation;
|
||||
|
||||
if (panel && panel->funcs && panel->funcs->get_orientation)
|
||||
orientation = panel->funcs->get_orientation(panel);
|
||||
else
|
||||
orientation = DRM_MODE_PANEL_ORIENTATION_UNKNOWN;
|
||||
|
||||
return drm_connector_set_panel_orientation(connector, orientation);
|
||||
}
|
||||
EXPORT_SYMBOL(drm_connector_set_orientation_from_panel);
|
||||
|
||||
static const struct drm_prop_enum_list privacy_screen_enum[] = {
|
||||
{ PRIVACY_SCREEN_DISABLED, "Disabled" },
|
||||
{ PRIVACY_SCREEN_ENABLED, "Enabled" },
|
||||
|
|
|
@ -35,9 +35,11 @@
|
|||
#include <linux/export.h>
|
||||
#include <linux/dma-fence.h>
|
||||
#include <linux/uaccess.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_managed.h>
|
||||
#include <drm/drm_modeset_lock.h>
|
||||
#include <drm/drm_atomic.h>
|
||||
|
|
|
@ -44,6 +44,7 @@
|
|||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
|
|
@ -33,6 +33,7 @@
|
|||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_damage_helper.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
|
||||
static void convert_clip_rect_to_rect(const struct drm_clip_rect *src,
|
||||
struct drm_mode_rect *dest,
|
||||
|
|
|
@ -43,6 +43,7 @@
|
|||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_print.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
|
||||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem.h>
|
||||
#include <drm/drm_gem_atomic_helper.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
|
|
|
@ -83,7 +83,7 @@
|
|||
*
|
||||
* 1. Directly call VERSION to get the version and to match against the driver
|
||||
* name returned by that ioctl. Note that SET_VERSION is not called, which
|
||||
* means the the unique name for the master node just opening is _not_ filled
|
||||
* means the unique name for the master node just opening is _not_ filled
|
||||
* out. This despite that with current drm device nodes are always bound to
|
||||
* one device, and can't be runtime assigned like with drm 1.0.
|
||||
* 2. Match driver name. If it mismatches, proceed to the next device node.
|
||||
|
|
|
@ -27,6 +27,7 @@
|
|||
|
||||
#include <linux/module.h>
|
||||
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
#include "drm_crtc_helper_internal.h"
|
||||
|
|
|
@ -18,6 +18,7 @@
|
|||
#include <drm/drm_file.h>
|
||||
#include <drm/drm_format_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
#include <drm/drm_mipi_dbi.h>
|
||||
|
|
|
@ -25,6 +25,7 @@
|
|||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_file.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_managed.h>
|
||||
#include <drm/drm_mode_config.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
|
|
@ -41,6 +41,7 @@
|
|||
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_modes.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
||||
|
|
|
@ -23,6 +23,7 @@
|
|||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_modeset_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_print.h>
|
||||
|
|
|
@ -187,7 +187,7 @@ static const struct dmi_system_id orientation_data[] = {
|
|||
},
|
||||
.driver_data = (void *)&lcd800x1280_rightside_up,
|
||||
}, { /*
|
||||
* GPD Pocket, note that the the DMI data is less generic then
|
||||
* GPD Pocket, note that the DMI data is less generic then
|
||||
* it seems, devices with a board-vendor of "AMI Corporation"
|
||||
* are quite rare, as are devices which have both board- *and*
|
||||
* product-id set to "Default String"
|
||||
|
|
|
@ -839,7 +839,7 @@ EXPORT_SYMBOL(drm_prime_pages_to_sg);
|
|||
* @sgt: sg_table describing the buffer to check
|
||||
*
|
||||
* This helper calculates the contiguous size in the DMA address space
|
||||
* of the the buffer described by the provided sg_table.
|
||||
* of the buffer described by the provided sg_table.
|
||||
*
|
||||
* This is useful for implementing
|
||||
* &drm_gem_object_funcs.gem_prime_import_sg_table.
|
||||
|
|
|
@ -14,6 +14,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_device.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_modeset_helper_vtables.h>
|
||||
#include <drm/drm_property.h>
|
||||
#include <drm/drm_writeback.h>
|
||||
|
|
|
@ -17,7 +17,9 @@
|
|||
#include <linux/pm_runtime.h>
|
||||
#include <linux/regmap.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
||||
#include "exynos_drm_crtc.h"
|
||||
|
|
|
@ -20,6 +20,7 @@
|
|||
#include <video/of_videomode.h>
|
||||
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
|
|
@ -15,6 +15,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_prime.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
|
|
@ -21,7 +21,9 @@
|
|||
#include <video/of_videomode.h>
|
||||
#include <video/samsung_fimd.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
||||
|
|
|
@ -18,6 +18,7 @@
|
|||
|
||||
#include <linux/uaccess.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_file.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_mode.h>
|
||||
|
|
|
@ -7,6 +7,8 @@
|
|||
|
||||
#include <drm/drm_atomic.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
||||
|
|
|
@ -15,6 +15,7 @@
|
|||
#include <linux/platform_device.h>
|
||||
#include <linux/pm_runtime.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
||||
|
|
|
@ -13,6 +13,7 @@
|
|||
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/drm_simple_kms_helper.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
|
|
@ -25,7 +25,10 @@
|
|||
#include <linux/spinlock.h>
|
||||
#include <linux/wait.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
#include <drm/exynos_drm.h>
|
||||
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
|
|
@ -32,6 +32,7 @@
|
|||
#include <drm/display/drm_dp_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_crtc_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_simple_kms_helper.h>
|
||||
|
||||
#include "gma_display.h"
|
||||
|
|
|
@ -21,6 +21,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
|
||||
#include "framebuffer.h"
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
||||
#include "framebuffer.h"
|
||||
|
|
|
@ -8,6 +8,7 @@
|
|||
#include <linux/pm_runtime.h>
|
||||
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
|
||||
#include "framebuffer.h"
|
||||
#include "gem.h"
|
||||
|
|
|
@ -27,6 +27,7 @@
|
|||
#include <linux/delay.h>
|
||||
|
||||
#include <drm/drm.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_simple_kms_helper.h>
|
||||
|
||||
#include "psb_drv.h"
|
||||
|
|
|
@ -13,6 +13,7 @@
|
|||
|
||||
#include <asm/intel-mid.h>
|
||||
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_simple_kms_helper.h>
|
||||
|
||||
#include "intel_bios.h"
|
||||
|
|
|
@ -7,6 +7,8 @@
|
|||
|
||||
#include <linux/i2c.h>
|
||||
|
||||
#include <drm/drm_edid.h>
|
||||
|
||||
#include "psb_intel_drv.h"
|
||||
|
||||
/**
|
||||
|
|
|
@ -10,6 +10,7 @@
|
|||
#include <drm/drm_atomic_state_helper.h>
|
||||
#include <drm/drm_connector.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_file.h>
|
||||
#include <drm/drm_modeset_helper_vtables.h>
|
||||
|
|
|
@ -14,6 +14,7 @@
|
|||
#include <linux/workqueue.h>
|
||||
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_damage_helper.h>
|
||||
#include <drm/drm_debugfs.h>
|
||||
#include <drm/drm_drv.h>
|
||||
|
|
|
@ -26,6 +26,7 @@
|
|||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_fb_cma_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_cma_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
|
|
@ -7,9 +7,11 @@
|
|||
|
||||
#include <drm/drm_damage_helper.h>
|
||||
#include <drm/drm_drv.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_fb_helper.h>
|
||||
#include <drm/drm_format_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_gem_atomic_helper.h>
|
||||
#include <drm/drm_gem_framebuffer_helper.h>
|
||||
#include <drm/drm_gem_shmem_helper.h>
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
#include <linux/kernel.h>
|
||||
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
|
||||
|
|
|
@ -25,6 +25,7 @@
|
|||
*
|
||||
*/
|
||||
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/display/drm_dp_helper.h>
|
||||
#include <drm/display/drm_dsc_helper.h>
|
||||
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_atomic_uapi.h>
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_damage_helper.h>
|
||||
#include <drm/drm_plane_helper.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
|
|
|
@ -38,6 +38,7 @@
|
|||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_encoder.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
#include <drm/drm_rect.h>
|
||||
#include <drm/drm_vblank.h>
|
||||
|
|
|
@ -40,6 +40,7 @@
|
|||
#include <drm/display/drm_hdmi_helper.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_crtc.h>
|
||||
#include <drm/drm_edid.h>
|
||||
#include <drm/drm_probe_helper.h>
|
||||
|
||||
#include "g4x_dp.h"
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
* Copyright © 2021 Intel Corporation
|
||||
*/
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_framebuffer.h>
|
||||
#include <drm/drm_modeset_helper.h>
|
||||
|
||||
|
|
|
@ -40,6 +40,7 @@
|
|||
|
||||
#include <linux/string_helpers.h>
|
||||
|
||||
#include <drm/drm_blend.h>
|
||||
#include <drm/drm_fourcc.h>
|
||||
|
||||
#include "i915_drv.h"
|
||||
|
|
|
@ -26,6 +26,7 @@
|
|||
#include <drm/display/drm_dp_dual_mode_helper.h>
|
||||
#include <drm/display/drm_hdmi_helper.h>
|
||||
#include <drm/drm_atomic_helper.h>
|
||||
#include <drm/drm_edid.h>
|
||||
|
||||
#include "intel_de.h"
|
||||
#include "intel_display_types.h"
|
||||
|
|
|
@ -30,6 +30,8 @@
|
|||
#include <linux/firmware.h>
|
||||
#include <acpi/video.h>
|
||||
|
||||
#include <drm/drm_edid.h>
|
||||
|
||||
#include "i915_drv.h"
|
||||
#include "intel_acpi.h"
|
||||
#include "intel_backlight.h"
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue