encoder-opa362.c 3.9 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * OPA362 analog video amplifier with output/power control
 *
 * Copyright (C) 2014 Golden Delicious Computers
 * Author: H. Nikolaus Schaller <hns@goldelico.com>
 *
 * based on encoder-tfp410
 *
9
 * Copyright (C) 2013 Texas Instruments Incorporated - http://www.ti.com/
10 11 12 13 14 15 16
 * Author: Tomi Valkeinen <tomi.valkeinen@ti.com>
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 as published by
 * the Free Software Foundation.
 */

T
Tomi Valkeinen 已提交
17
#include <linux/gpio/consumer.h>
18 19 20 21
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/slab.h>

22
#include "../dss/omapdss.h"
23 24 25 26 27 28 29 30 31

struct panel_drv_data {
	struct omap_dss_device dssdev;

	struct gpio_desc *enable_gpio;
};

#define to_panel_data(x) container_of(x, struct panel_drv_data, dssdev)

32 33
static int opa362_connect(struct omap_dss_device *src,
			  struct omap_dss_device *dst)
34
{
35
	return omapdss_device_connect(dst->dss, dst, dst->next);
36 37
}

38 39
static void opa362_disconnect(struct omap_dss_device *src,
			      struct omap_dss_device *dst)
40
{
41
	omapdss_device_disconnect(dst, dst->next);
42 43 44 45 46
}

static int opa362_enable(struct omap_dss_device *dssdev)
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);
47
	struct omap_dss_device *src = dssdev->src;
48 49 50 51
	int r;

	dev_dbg(dssdev->dev, "enable\n");

52
	r = src->ops->enable(src);
53 54 55 56 57 58 59 60 61 62 63 64 65 66
	if (r)
		return r;

	if (ddata->enable_gpio)
		gpiod_set_value_cansleep(ddata->enable_gpio, 1);

	dssdev->state = OMAP_DSS_DISPLAY_ACTIVE;

	return 0;
}

static void opa362_disable(struct omap_dss_device *dssdev)
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);
67
	struct omap_dss_device *src = dssdev->src;
68 69 70 71 72 73

	dev_dbg(dssdev->dev, "disable\n");

	if (ddata->enable_gpio)
		gpiod_set_value_cansleep(ddata->enable_gpio, 0);

74
	src->ops->disable(src);
75 76 77 78

	dssdev->state = OMAP_DSS_DISPLAY_DISABLED;
}

79
static const struct omap_dss_device_ops opa362_ops = {
80 81 82 83 84 85 86 87 88
	.connect	= opa362_connect,
	.disconnect	= opa362_disconnect,
	.enable		= opa362_enable,
	.disable	= opa362_disable,
};

static int opa362_probe(struct platform_device *pdev)
{
	struct panel_drv_data *ddata;
89
	struct omap_dss_device *dssdev;
90 91 92 93 94 95 96 97 98 99
	struct gpio_desc *gpio;

	dev_dbg(&pdev->dev, "probe\n");

	ddata = devm_kzalloc(&pdev->dev, sizeof(*ddata), GFP_KERNEL);
	if (!ddata)
		return -ENOMEM;

	platform_set_drvdata(pdev, ddata);

100 101 102
	gpio = devm_gpiod_get_optional(&pdev->dev, "enable", GPIOD_OUT_LOW);
	if (IS_ERR(gpio))
		return PTR_ERR(gpio);
103 104 105 106

	ddata->enable_gpio = gpio;

	dssdev = &ddata->dssdev;
107
	dssdev->ops = &opa362_ops;
108 109 110 111
	dssdev->dev = &pdev->dev;
	dssdev->type = OMAP_DISPLAY_TYPE_VENC;
	dssdev->output_type = OMAP_DISPLAY_TYPE_VENC;
	dssdev->owner = THIS_MODULE;
112
	dssdev->of_ports = BIT(1) | BIT(0);
113

114 115 116 117 118 119 120
	dssdev->next = omapdss_of_find_connected_device(pdev->dev.of_node, 1);
	if (IS_ERR(dssdev->next)) {
		if (PTR_ERR(dssdev->next) != -EPROBE_DEFER)
			dev_err(&pdev->dev, "failed to find video sink\n");
		return PTR_ERR(dssdev->next);
	}

121
	omapdss_device_register(dssdev);
122 123 124 125 126 127 128 129 130

	return 0;
}

static int __exit opa362_remove(struct platform_device *pdev)
{
	struct panel_drv_data *ddata = platform_get_drvdata(pdev);
	struct omap_dss_device *dssdev = &ddata->dssdev;

131 132
	if (dssdev->next)
		omapdss_device_put(dssdev->next);
133
	omapdss_device_unregister(&ddata->dssdev);
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162

	WARN_ON(omapdss_device_is_enabled(dssdev));
	if (omapdss_device_is_enabled(dssdev))
		opa362_disable(dssdev);

	return 0;
}

static const struct of_device_id opa362_of_match[] = {
	{ .compatible = "omapdss,ti,opa362", },
	{},
};
MODULE_DEVICE_TABLE(of, opa362_of_match);

static struct platform_driver opa362_driver = {
	.probe	= opa362_probe,
	.remove	= __exit_p(opa362_remove),
	.driver	= {
		.name	= "amplifier-opa362",
		.of_match_table = opa362_of_match,
		.suppress_bind_attrs = true,
	},
};

module_platform_driver(opa362_driver);

MODULE_AUTHOR("H. Nikolaus Schaller <hns@goldelico.com>");
MODULE_DESCRIPTION("OPA362 analog video amplifier with output/power control");
MODULE_LICENSE("GPL v2");