68 lines
1.4 KiB
C
68 lines
1.4 KiB
C
|
// SPDX-License-Identifier: GPL-2.0-only
|
||
|
/*
|
||
|
* Copyright (c) 2012-2020, The Linux Foundation. All rights reserved.
|
||
|
*/
|
||
|
|
||
|
#define pr_fmt(fmt) "[drm-dp] %s: " fmt, __func__
|
||
|
|
||
|
#include <linux/slab.h>
|
||
|
#include <linux/device.h>
|
||
|
|
||
|
#include "dp_hpd.h"
|
||
|
|
||
|
/* DP specific VDM commands */
|
||
|
#define DP_USBPD_VDM_STATUS 0x10
|
||
|
#define DP_USBPD_VDM_CONFIGURE 0x11
|
||
|
|
||
|
/* USBPD-TypeC specific Macros */
|
||
|
#define VDM_VERSION 0x0
|
||
|
#define USB_C_DP_SID 0xFF01
|
||
|
|
||
|
struct dp_hpd_private {
|
||
|
struct device *dev;
|
||
|
struct dp_usbpd_cb *dp_cb;
|
||
|
struct dp_usbpd dp_usbpd;
|
||
|
};
|
||
|
|
||
|
int dp_hpd_connect(struct dp_usbpd *dp_usbpd, bool hpd)
|
||
|
{
|
||
|
int rc = 0;
|
||
|
struct dp_hpd_private *hpd_priv;
|
||
|
|
||
|
hpd_priv = container_of(dp_usbpd, struct dp_hpd_private,
|
||
|
dp_usbpd);
|
||
|
|
||
|
if (!hpd_priv->dp_cb || !hpd_priv->dp_cb->configure
|
||
|
|| !hpd_priv->dp_cb->disconnect) {
|
||
|
pr_err("hpd dp_cb not initialized\n");
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
if (hpd)
|
||
|
hpd_priv->dp_cb->configure(hpd_priv->dev);
|
||
|
else
|
||
|
hpd_priv->dp_cb->disconnect(hpd_priv->dev);
|
||
|
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
struct dp_usbpd *dp_hpd_get(struct device *dev, struct dp_usbpd_cb *cb)
|
||
|
{
|
||
|
struct dp_hpd_private *dp_hpd;
|
||
|
|
||
|
if (!cb) {
|
||
|
pr_err("invalid cb data\n");
|
||
|
return ERR_PTR(-EINVAL);
|
||
|
}
|
||
|
|
||
|
dp_hpd = devm_kzalloc(dev, sizeof(*dp_hpd), GFP_KERNEL);
|
||
|
if (!dp_hpd)
|
||
|
return ERR_PTR(-ENOMEM);
|
||
|
|
||
|
dp_hpd->dev = dev;
|
||
|
dp_hpd->dp_cb = cb;
|
||
|
|
||
|
dp_hpd->dp_usbpd.connect = dp_hpd_connect;
|
||
|
|
||
|
return &dp_hpd->dp_usbpd;
|
||
|
}
|