[PATCH 04/11] vga_switcheroo: Allow stashing of panel data
Jani Nikula
jani.nikula at linux.intel.com
Tue Jun 3 06:40:36 PDT 2014
On Sun, 01 Jun 2014, Matthew Garrett <matthew.garrett at nebula.com> wrote:
> Not all MUXes allow us to connect the panel data channel to a GPU without
> handing over the entire panel and triggering additional flickering during
> boot. We only need to do this in order to probe for data that the first
> GPU driver has already identified, so add some functions for stashing that
> data in vga_switcheroo where it can be retrieved by the other driver later.
>
> Signed-off-by: Matthew Garrett <matthew.garrett at nebula.com>
> ---
> drivers/gpu/vga/vga_switcheroo.c | 59 ++++++++++++++++++++++++++++++++++++++++
> include/linux/vga_switcheroo.h | 12 ++++++++
> 2 files changed, 71 insertions(+)
>
> diff --git a/drivers/gpu/vga/vga_switcheroo.c b/drivers/gpu/vga/vga_switcheroo.c
> index 6d95626..1a80b93 100644
> --- a/drivers/gpu/vga/vga_switcheroo.c
> +++ b/drivers/gpu/vga/vga_switcheroo.c
> @@ -17,6 +17,8 @@
> - switch_check - check if the device is in a position to switch now
> */
>
> +#include <drm/drm_edid.h>
> +
> #include <linux/module.h>
> #include <linux/seq_file.h>
> #include <linux/uaccess.h>
> @@ -39,6 +41,7 @@ struct vga_switcheroo_client {
> int id;
> bool active;
> bool driver_power_control;
> + bool use_panel;
> struct list_head list;
> };
>
> @@ -56,6 +59,9 @@ struct vgasr_priv {
> int registered_clients;
> struct list_head clients;
>
> + struct edid *edid;
> + u8 *dpcd;
> +
> struct vga_switcheroo_handler *handler;
> };
>
> @@ -107,7 +113,9 @@ static void vga_switcheroo_enable(void)
> VGA_SWITCHEROO_DIS : VGA_SWITCHEROO_IGD;
> if (vgasr_priv.handler->switch_ddc)
> vgasr_priv.handler->switch_ddc(client->id);
> + client->use_panel = true;
> client->ops->reprobe_connectors(client->pdev);
> + client->use_panel = false;
> if (vgasr_priv.handler->switch_ddc)
> vgasr_priv.handler->switch_ddc(old_id);
> }
> @@ -412,6 +420,9 @@ static int vga_switchto_stage2(struct vga_switcheroo_client *new_client)
> if (ret)
> goto restore_ddc;
>
> + new_client->use_panel = true;
> + active->use_panel = false;
> +
> if (new_client->ops->reprobe)
> new_client->ops->reprobe(new_client->pdev);
>
> @@ -766,6 +777,54 @@ int vga_switcheroo_init_domain_pm_optimus_hdmi_audio(struct device *dev, struct
> }
> EXPORT_SYMBOL(vga_switcheroo_init_domain_pm_optimus_hdmi_audio);
>
> +int vga_switcheroo_set_dpcd(u8 *dpcd)
Quite a few DPCD related things in drm_dp_helper.h expect
DP_RECEIVER_CAP_SIZE bytes of DPCD. Might be helpful to conform to that.
Can also be made const u8 *.
> +{
> + if (vgasr_priv.dpcd)
> + return -EEXIST;
> +
> + vgasr_priv.dpcd = kmalloc(8, GFP_KERNEL);
> + memcpy(vgasr_priv.dpcd, dpcd, 8);
kmemdup.
> + return 0;
> +}
> +EXPORT_SYMBOL(vga_switcheroo_set_dpcd);
> +
> +u8 *vga_switcheroo_get_dpcd(struct pci_dev *pdev)
> +{
> + struct vga_switcheroo_client *client;
> + client = find_client_from_pci(&vgasr_priv.clients, pdev);
> +
> + if (!client || !client->use_panel)
> + return NULL;
> +
> + return vgasr_priv.dpcd;
> +}
> +EXPORT_SYMBOL(vga_switcheroo_get_dpcd);
> +
> +int vga_switcheroo_set_edid(struct edid *edid)
> +{
> + int size = EDID_LENGTH * (1 + edid->extensions);
> +
> + if (vgasr_priv.edid)
> + return -EEXIST;
> +
> + vgasr_priv.edid = kmalloc(size, GFP_KERNEL);
> + memcpy(vgasr_priv.edid, edid, size);
kmemdup.
BR,
Jani.
> + return 0;
> +}
> +EXPORT_SYMBOL(vga_switcheroo_set_edid);
> +
> +struct edid *vga_switcheroo_get_edid(struct pci_dev *pdev)
> +{
> + struct vga_switcheroo_client *client;
> + client = find_client_from_pci(&vgasr_priv.clients, pdev);
> +
> + if (!client || !client->use_panel)
> + return NULL;
> +
> + return vgasr_priv.edid;
> +}
> +EXPORT_SYMBOL(vga_switcheroo_get_edid);
> +
> static int __init vga_switcheroo_setup(char *str)
> {
> if (!strcmp(str, "IGD"))
> diff --git a/include/linux/vga_switcheroo.h b/include/linux/vga_switcheroo.h
> index bae62fd..07b07fc 100644
> --- a/include/linux/vga_switcheroo.h
> +++ b/include/linux/vga_switcheroo.h
> @@ -10,6 +10,7 @@
> #ifndef _LINUX_VGA_SWITCHEROO_H_
> #define _LINUX_VGA_SWITCHEROO_H_
>
> +#include <drm/drm_edid.h>
> #include <linux/fb.h>
>
> struct pci_dev;
> @@ -69,6 +70,12 @@ void vga_switcheroo_set_dynamic_switch(struct pci_dev *pdev, enum vga_switcheroo
>
> int vga_switcheroo_init_domain_pm_ops(struct device *dev, struct dev_pm_domain *domain);
> int vga_switcheroo_init_domain_pm_optimus_hdmi_audio(struct device *dev, struct dev_pm_domain *domain);
> +
> +int vga_switcheroo_set_dpcd(u8 *dpcd);
> +u8 *vga_switcheroo_get_dpcd(struct pci_dev *pdev);
> +int vga_switcheroo_set_edid(struct edid *edid);
> +struct edid *vga_switcheroo_get_edid(struct pci_dev *pdev);
> +
> #else
>
> static inline void vga_switcheroo_unregister_client(struct pci_dev *dev) {}
> @@ -89,5 +96,10 @@ static inline void vga_switcheroo_set_dynamic_switch(struct pci_dev *pdev, enum
> static inline int vga_switcheroo_init_domain_pm_ops(struct device *dev, struct dev_pm_domain *domain) { return -EINVAL; }
> static inline int vga_switcheroo_init_domain_pm_optimus_hdmi_audio(struct device *dev, struct dev_pm_domain *domain) { return -EINVAL; }
>
> +static inline int vga_switcheroo_set_dpcd(u8 *dpcd) { return 0 };
> +static inline u8 *vga_switcheroo_get_dpcd(struct pci_dev *pdev) { return NULL };
> +static inline int vga_switcheroo_set_edid(struct edid *edid) { return 0 };
> +static inline struct edid *vga_switcheroo_get_edid(struct pci_dev *pdev) { return NULL };
> +
> #endif
> #endif /* _LINUX_VGA_SWITCHEROO_H_ */
> --
> 1.8.5.3
>
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/dri-devel
--
Jani Nikula, Intel Open Source Technology Center
More information about the dri-devel
mailing list