2 * Copyright 2012-15 Advanced Micro Devices, Inc.
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
26 #include <linux/version.h>
27 #include <drm/drm_atomic_helper.h>
28 #include "dm_services.h"
30 #include "amdgpu_dm_types.h"
31 #include "amdgpu_dm_mst_types.h"
34 #include "dm_helpers.h"
36 /* #define TRACE_DPCD */
39 #define SIDE_BAND_MSG(address) (address >= DP_SIDEBAND_MSG_DOWN_REQ_BASE && address < DP_SINK_COUNT_ESI)
41 static inline char *side_band_msg_type_to_str(uint32_t address
)
43 static char str
[10] = {0};
45 if (address
< DP_SIDEBAND_MSG_UP_REP_BASE
)
46 strcpy(str
, "DOWN_REQ");
47 else if (address
< DP_SIDEBAND_MSG_DOWN_REP_BASE
)
48 strcpy(str
, "UP_REP");
49 else if (address
< DP_SIDEBAND_MSG_UP_REQ_BASE
)
50 strcpy(str
, "DOWN_REP");
52 strcpy(str
, "UP_REQ");
57 void log_dpcd(uint8_t type
,
63 DRM_DEBUG_KMS("Op: %s, addr: %04x, SideBand Msg: %s, Op res: %s\n",
64 (type
== DP_AUX_NATIVE_READ
) ||
65 (type
== DP_AUX_I2C_READ
) ?
68 SIDE_BAND_MSG(address
) ?
69 side_band_msg_type_to_str(address
) : "Nop",
73 print_hex_dump(KERN_INFO
, "Body: ", DUMP_PREFIX_NONE
, 16, 1, data
, size
, false);
78 static ssize_t
dm_dp_aux_transfer(struct drm_dp_aux
*aux
, struct drm_dp_aux_msg
*msg
)
80 struct pci_dev
*pdev
= to_pci_dev(aux
->dev
);
81 struct drm_device
*drm_dev
= pci_get_drvdata(pdev
);
82 struct amdgpu_device
*adev
= drm_dev
->dev_private
;
83 struct dc
*dc
= adev
->dm
.dc
;
86 switch (msg
->request
) {
87 case DP_AUX_NATIVE_READ
:
90 TO_DM_AUX(aux
)->link_index
,
95 case DP_AUX_NATIVE_WRITE
:
98 TO_DM_AUX(aux
)->link_index
,
108 log_dpcd(msg
->request
,
118 static enum drm_connector_status
119 dm_dp_mst_detect(struct drm_connector
*connector
, bool force
)
121 struct amdgpu_connector
*aconnector
= to_amdgpu_connector(connector
);
122 struct amdgpu_connector
*master
= aconnector
->mst_port
;
124 enum drm_connector_status status
=
125 drm_dp_mst_detect_port(
134 dm_dp_mst_connector_destroy(struct drm_connector
*connector
)
136 struct amdgpu_connector
*amdgpu_connector
= to_amdgpu_connector(connector
);
137 struct amdgpu_encoder
*amdgpu_encoder
= amdgpu_connector
->mst_encoder
;
139 drm_encoder_cleanup(&amdgpu_encoder
->base
);
140 kfree(amdgpu_encoder
);
141 drm_connector_cleanup(connector
);
142 kfree(amdgpu_connector
);
145 static const struct drm_connector_funcs dm_dp_mst_connector_funcs
= {
146 .detect
= dm_dp_mst_detect
,
147 .fill_modes
= drm_helper_probe_single_connector_modes
,
148 .destroy
= dm_dp_mst_connector_destroy
,
149 .reset
= amdgpu_dm_connector_funcs_reset
,
150 .atomic_duplicate_state
= amdgpu_dm_connector_atomic_duplicate_state
,
151 .atomic_destroy_state
= drm_atomic_helper_connector_destroy_state
,
152 .atomic_set_property
= amdgpu_dm_connector_atomic_set_property
155 static int dm_dp_mst_get_modes(struct drm_connector
*connector
)
157 struct amdgpu_connector
*aconnector
= to_amdgpu_connector(connector
);
160 ret
= drm_add_edid_modes(&aconnector
->base
, aconnector
->edid
);
162 drm_edid_to_eld(&aconnector
->base
, aconnector
->edid
);
167 static struct drm_encoder
*dm_mst_best_encoder(struct drm_connector
*connector
)
169 struct amdgpu_connector
*amdgpu_connector
= to_amdgpu_connector(connector
);
171 return &amdgpu_connector
->mst_encoder
->base
;
174 static const struct drm_connector_helper_funcs dm_dp_mst_connector_helper_funcs
= {
175 .get_modes
= dm_dp_mst_get_modes
,
176 .mode_valid
= amdgpu_dm_connector_mode_valid
,
177 .best_encoder
= dm_mst_best_encoder
,
180 static struct amdgpu_encoder
*
181 dm_dp_create_fake_mst_encoder(struct amdgpu_connector
*connector
)
183 struct drm_device
*dev
= connector
->base
.dev
;
184 struct amdgpu_device
*adev
= dev
->dev_private
;
185 struct amdgpu_encoder
*amdgpu_encoder
;
186 struct drm_encoder
*encoder
;
187 const struct drm_connector_helper_funcs
*connector_funcs
=
188 connector
->base
.helper_private
;
189 struct drm_encoder
*enc_master
=
190 connector_funcs
->best_encoder(&connector
->base
);
192 DRM_DEBUG_KMS("enc master is %p\n", enc_master
);
193 amdgpu_encoder
= kzalloc(sizeof(*amdgpu_encoder
), GFP_KERNEL
);
197 encoder
= &amdgpu_encoder
->base
;
198 encoder
->possible_crtcs
= amdgpu_dm_get_encoder_crtc_mask(adev
);
202 &amdgpu_encoder
->base
,
204 DRM_MODE_ENCODER_DPMST
,
207 drm_encoder_helper_add(encoder
, &amdgpu_dm_encoder_helper_funcs
);
209 return amdgpu_encoder
;
212 static struct drm_connector
*dm_dp_add_mst_connector(struct drm_dp_mst_topology_mgr
*mgr
,
213 struct drm_dp_mst_port
*port
,
214 const char *pathprop
)
216 struct amdgpu_connector
*master
= container_of(mgr
, struct amdgpu_connector
, mst_mgr
);
217 struct drm_device
*dev
= master
->base
.dev
;
218 struct amdgpu_device
*adev
= dev
->dev_private
;
219 struct amdgpu_connector
*aconnector
;
220 struct drm_connector
*connector
;
222 drm_modeset_lock(&dev
->mode_config
.connection_mutex
, NULL
);
223 list_for_each_entry(connector
, &dev
->mode_config
.connector_list
, head
) {
224 aconnector
= to_amdgpu_connector(connector
);
225 if (aconnector
->mst_port
== master
226 && !aconnector
->port
) {
227 DRM_INFO("DM_MST: reusing connector: %p [id: %d] [master: %p]\n",
228 aconnector
, connector
->base
.id
, aconnector
->mst_port
);
230 aconnector
->port
= port
;
231 drm_mode_connector_set_path_property(connector
, pathprop
);
233 drm_modeset_unlock(&dev
->mode_config
.connection_mutex
);
234 return &aconnector
->base
;
237 drm_modeset_unlock(&dev
->mode_config
.connection_mutex
);
239 aconnector
= kzalloc(sizeof(*aconnector
), GFP_KERNEL
);
243 connector
= &aconnector
->base
;
244 aconnector
->port
= port
;
245 aconnector
->mst_port
= master
;
247 if (drm_connector_init(
250 &dm_dp_mst_connector_funcs
,
251 DRM_MODE_CONNECTOR_DisplayPort
)) {
255 drm_connector_helper_add(connector
, &dm_dp_mst_connector_helper_funcs
);
257 amdgpu_dm_connector_init_helper(
260 DRM_MODE_CONNECTOR_DisplayPort
,
262 master
->connector_id
);
264 aconnector
->mst_encoder
= dm_dp_create_fake_mst_encoder(master
);
267 * TODO: understand why this one is needed
269 drm_object_attach_property(
271 dev
->mode_config
.path_property
,
273 drm_object_attach_property(
275 dev
->mode_config
.tile_property
,
278 drm_mode_connector_set_path_property(connector
, pathprop
);
281 * Initialize connector state before adding the connectror to drm and
284 amdgpu_dm_connector_funcs_reset(connector
);
286 DRM_INFO("DM_MST: added connector: %p [id: %d] [master: %p]\n",
287 aconnector
, connector
->base
.id
, aconnector
->mst_port
);
289 DRM_DEBUG_KMS(":%d\n", connector
->base
.id
);
294 static void dm_dp_destroy_mst_connector(
295 struct drm_dp_mst_topology_mgr
*mgr
,
296 struct drm_connector
*connector
)
298 struct amdgpu_connector
*aconnector
= to_amdgpu_connector(connector
);
300 DRM_INFO("DM_MST: Disabling connector: %p [id: %d] [master: %p]\n",
301 aconnector
, connector
->base
.id
, aconnector
->mst_port
);
303 aconnector
->port
= NULL
;
304 if (aconnector
->dc_sink
) {
305 amdgpu_dm_remove_sink_from_freesync_module(connector
);
306 dc_link_remove_remote_sink(aconnector
->dc_link
, aconnector
->dc_sink
);
307 dc_sink_release(aconnector
->dc_sink
);
308 aconnector
->dc_sink
= NULL
;
310 if (aconnector
->edid
) {
311 kfree(aconnector
->edid
);
312 aconnector
->edid
= NULL
;
315 drm_mode_connector_update_edid_property(
320 static void dm_dp_mst_hotplug(struct drm_dp_mst_topology_mgr
*mgr
)
322 struct amdgpu_connector
*master
= container_of(mgr
, struct amdgpu_connector
, mst_mgr
);
323 struct drm_device
*dev
= master
->base
.dev
;
324 struct amdgpu_device
*adev
= dev
->dev_private
;
325 struct drm_connector
*connector
;
326 struct amdgpu_connector
*aconnector
;
329 drm_modeset_lock_all(dev
);
330 list_for_each_entry(connector
, &dev
->mode_config
.connector_list
, head
) {
331 aconnector
= to_amdgpu_connector(connector
);
332 if (aconnector
->port
&&
333 aconnector
->port
->pdt
!= DP_PEER_DEVICE_NONE
&&
334 aconnector
->port
->pdt
!= DP_PEER_DEVICE_MST_BRANCHING
&&
335 !aconnector
->dc_sink
) {
337 * This is plug in case, where port has been created but
338 * sink hasn't been created yet
340 if (!aconnector
->edid
) {
341 struct dc_sink_init_data init_params
= {
342 .link
= aconnector
->dc_link
,
343 .sink_signal
= SIGNAL_TYPE_DISPLAY_PORT_MST
};
344 edid
= drm_dp_mst_get_edid(connector
, &aconnector
->mst_port
->mst_mgr
, aconnector
->port
);
347 drm_mode_connector_update_edid_property(
353 aconnector
->edid
= edid
;
355 aconnector
->dc_sink
= dc_link_add_remote_sink(
358 (edid
->extensions
+ 1) * EDID_LENGTH
,
360 if (aconnector
->dc_sink
)
361 amdgpu_dm_add_sink_to_freesync_module(
365 dm_restore_drm_connector_state(connector
->dev
, connector
);
367 edid
= aconnector
->edid
;
369 DRM_DEBUG_KMS("edid retrieved %p\n", edid
);
371 drm_mode_connector_update_edid_property(
376 drm_modeset_unlock_all(dev
);
378 schedule_work(&adev
->dm
.mst_hotplug_work
);
381 static void dm_dp_mst_register_connector(struct drm_connector
*connector
)
383 struct drm_device
*dev
= connector
->dev
;
384 struct amdgpu_device
*adev
= dev
->dev_private
;
387 drm_modeset_lock_all(dev
);
388 if (adev
->mode_info
.rfbdev
) {
389 /*Do not add if already registered in past*/
390 for (i
= 0; i
< adev
->mode_info
.rfbdev
->helper
.connector_count
; i
++) {
391 if (adev
->mode_info
.rfbdev
->helper
.connector_info
[i
]->connector
393 drm_modeset_unlock_all(dev
);
398 drm_fb_helper_add_one_connector(&adev
->mode_info
.rfbdev
->helper
, connector
);
401 DRM_ERROR("adev->mode_info.rfbdev is NULL\n");
403 drm_modeset_unlock_all(dev
);
405 drm_connector_register(connector
);
409 static const struct drm_dp_mst_topology_cbs dm_mst_cbs
= {
410 .add_connector
= dm_dp_add_mst_connector
,
411 .destroy_connector
= dm_dp_destroy_mst_connector
,
412 .hotplug
= dm_dp_mst_hotplug
,
413 .register_connector
= dm_dp_mst_register_connector
416 void amdgpu_dm_initialize_mst_connector(
417 struct amdgpu_display_manager
*dm
,
418 struct amdgpu_connector
*aconnector
)
420 aconnector
->dm_dp_aux
.aux
.name
= "dmdc";
421 aconnector
->dm_dp_aux
.aux
.dev
= dm
->adev
->dev
;
422 aconnector
->dm_dp_aux
.aux
.transfer
= dm_dp_aux_transfer
;
423 aconnector
->dm_dp_aux
.link_index
= aconnector
->connector_id
;
425 drm_dp_aux_register(&aconnector
->dm_dp_aux
.aux
);
426 aconnector
->mst_mgr
.cbs
= &dm_mst_cbs
;
427 drm_dp_mst_topology_mgr_init(
428 &aconnector
->mst_mgr
,
430 &aconnector
->dm_dp_aux
.aux
,
433 aconnector
->connector_id
);