2 * Copyright (C) 2012 Avionic Design GmbH
3 * Copyright (C) 2012 NVIDIA CORPORATION. All rights reserved.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
10 #include <drm/drm_atomic_helper.h>
11 #include <drm/drm_panel.h>
14 int tegra_output_connector_get_modes(struct drm_connector
*connector
)
16 struct tegra_output
*output
= connector_to_output(connector
);
17 struct edid
*edid
= NULL
;
21 * If the panel provides one or more modes, use them exclusively and
22 * ignore any other means of obtaining a mode.
25 err
= output
->panel
->funcs
->get_modes(output
->panel
);
31 edid
= kmemdup(output
->edid
, sizeof(*edid
), GFP_KERNEL
);
33 edid
= drm_get_edid(connector
, output
->ddc
);
35 drm_mode_connector_update_edid_property(connector
, edid
);
38 err
= drm_add_edid_modes(connector
, edid
);
46 tegra_output_connector_best_encoder(struct drm_connector
*connector
)
48 struct tegra_output
*output
= connector_to_output(connector
);
50 return &output
->encoder
;
53 enum drm_connector_status
54 tegra_output_connector_detect(struct drm_connector
*connector
, bool force
)
56 struct tegra_output
*output
= connector_to_output(connector
);
57 enum drm_connector_status status
= connector_status_unknown
;
59 if (gpio_is_valid(output
->hpd_gpio
)) {
60 if (output
->hpd_gpio_flags
& OF_GPIO_ACTIVE_LOW
) {
61 if (gpio_get_value(output
->hpd_gpio
) != 0)
62 status
= connector_status_disconnected
;
64 status
= connector_status_connected
;
66 if (gpio_get_value(output
->hpd_gpio
) == 0)
67 status
= connector_status_disconnected
;
69 status
= connector_status_connected
;
73 status
= connector_status_disconnected
;
75 status
= connector_status_connected
;
81 void tegra_output_connector_destroy(struct drm_connector
*connector
)
83 drm_connector_unregister(connector
);
84 drm_connector_cleanup(connector
);
87 void tegra_output_encoder_destroy(struct drm_encoder
*encoder
)
89 drm_encoder_cleanup(encoder
);
92 static irqreturn_t
hpd_irq(int irq
, void *data
)
94 struct tegra_output
*output
= data
;
96 if (output
->connector
.dev
)
97 drm_helper_hpd_irq_event(output
->connector
.dev
);
102 int tegra_output_probe(struct tegra_output
*output
)
104 struct device_node
*ddc
, *panel
;
107 if (!output
->of_node
)
108 output
->of_node
= output
->dev
->of_node
;
110 panel
= of_parse_phandle(output
->of_node
, "nvidia,panel", 0);
112 output
->panel
= of_drm_find_panel(panel
);
114 return -EPROBE_DEFER
;
119 output
->edid
= of_get_property(output
->of_node
, "nvidia,edid", &size
);
121 ddc
= of_parse_phandle(output
->of_node
, "nvidia,ddc-i2c-bus", 0);
123 output
->ddc
= of_find_i2c_adapter_by_node(ddc
);
133 output
->hpd_gpio
= of_get_named_gpio_flags(output
->of_node
,
134 "nvidia,hpd-gpio", 0,
135 &output
->hpd_gpio_flags
);
136 if (gpio_is_valid(output
->hpd_gpio
)) {
139 err
= gpio_request_one(output
->hpd_gpio
, GPIOF_DIR_IN
,
140 "HDMI hotplug detect");
142 dev_err(output
->dev
, "gpio_request_one(): %d\n", err
);
146 err
= gpio_to_irq(output
->hpd_gpio
);
148 dev_err(output
->dev
, "gpio_to_irq(): %d\n", err
);
149 gpio_free(output
->hpd_gpio
);
153 output
->hpd_irq
= err
;
155 flags
= IRQF_TRIGGER_RISING
| IRQF_TRIGGER_FALLING
|
158 err
= request_threaded_irq(output
->hpd_irq
, NULL
, hpd_irq
,
159 flags
, "hpd", output
);
161 dev_err(output
->dev
, "failed to request IRQ#%u: %d\n",
162 output
->hpd_irq
, err
);
163 gpio_free(output
->hpd_gpio
);
167 output
->connector
.polled
= DRM_CONNECTOR_POLL_HPD
;
170 * Disable the interrupt until the connector has been
171 * initialized to avoid a race in the hotplug interrupt
174 disable_irq(output
->hpd_irq
);
180 void tegra_output_remove(struct tegra_output
*output
)
182 if (gpio_is_valid(output
->hpd_gpio
)) {
183 free_irq(output
->hpd_irq
, output
);
184 gpio_free(output
->hpd_gpio
);
188 put_device(&output
->ddc
->dev
);
191 int tegra_output_init(struct drm_device
*drm
, struct tegra_output
*output
)
196 err
= drm_panel_attach(output
->panel
, &output
->connector
);
202 * The connector is now registered and ready to receive hotplug events
203 * so the hotplug interrupt can be enabled.
205 if (gpio_is_valid(output
->hpd_gpio
))
206 enable_irq(output
->hpd_irq
);
211 void tegra_output_exit(struct tegra_output
*output
)
214 * The connector is going away, so the interrupt must be disabled to
215 * prevent the hotplug interrupt handler from potentially crashing.
217 if (gpio_is_valid(output
->hpd_gpio
))
218 disable_irq(output
->hpd_irq
);
221 drm_panel_detach(output
->panel
);