99a0d9f5e8
Core changes: - Expose pull up/down flags for the GPIO character device to userspace. After clear input from the RaspberryPi and Beagle communities, it has been established that prototyping, industrial automation and make communities strongly need this feature, and as we want people to use the character device, we have implemented the simple pull up/down interface for GPIO lines. This means we can specify that a (chip-specific) pull up/down resistor can be enabled, but does not offer fine-grained control such as cases where the resistance of the same pull resistor can be controlled (yet). - Introduce devm_fwnode_gpiod_get_index() and start to phase out the old symbol devm_fwnode_get_index_gpiod_from_child(). - A bit of documentation clean-up work. - Introduce a define for GPIO line directions and deploy it in all GPIO drivers in the drivers/gpio directory. - Add a special callback to populate pin ranges when cooperating with the pin control subsystem and registering ranges as part of adding a gpiolib driver and a gpio_irq_chip driver at the same time. This is also deployed in the Intel Merrifield driver. New drivers: - RDA Micro GPIO controller. - XGS-iproc GPIO driver. Driver improvements: - Wake event and debounce support on the Tegra 186 driver. - Finalize the Aspeed SGPIO driver. - MPC8xxx uses a normal IRQ handler rather than a chained handler. -----BEGIN PGP SIGNATURE----- iQIzBAABCAAdFiEElDRnuGcz/wPCXQWMQRCzN7AZXXMFAl3hIDcACgkQQRCzN7AZ XXOOyw/8DcaBV6j3EZPDS+b+N74/flNf9JitdJRCtUPn8mjdm+uKNPxbtL/znZc8 zd3rlpiZOqHy8klB3gOPJsJhgXY9QX/b+F5j8fHZvu0DACugRndCqMJ7wrgwUiPn 2ni6KJmz6z5urhcfaAIgyinTWOMegvSVfqjISaUCRCAg3F9dIeQoulRvTPk2ybvv f31jAGemGbvvQPpd81SYCxuTbMg+jxBIgnOCaX+VmUaBLzh8J2W4It3Myp6M4Sbg Td6QU4b2J2Q0quk/Dc7c4saT+qRODkg2syPKV2YqmWwdLRDxZyKESMdkCXLWlWJU fP+KZ4lDxhCaOAYUrY2sEAYMw4E8MzrfeWikdIe0nk0DWqNQhWvDyzsNsB90XGFb aGgeCPH2W1MdE6usPhLidBaHbLeowzndw5BiEl0UCJUqz7tzTCd5iMIAhoSU/Sr5 ymO8J45G9rdx5pscA3cXhpR/PmqaETYQ/uNrLuxTdI4F4xY12+M0vPrV8z3oDPxB U/uL0v6HndDcFAavQQiMd9eL6Hocirnn+Z2xFut3nOznHY96ozXSnZb3lzvH/kqI Du2C8geboVcZsiZJTKVN1zxnfIA8oDauzTOEpGFbIGFhmy0zt4RRRptL4W8NxeFm KCOk/HqlGeuqJ49epta3mqhUC0MSASA9fdicCdiDqvw+puEznwU= =o13I -----END PGP SIGNATURE----- Merge tag 'gpio-v5.5-1' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio Pull GPIO updates from Linus Walleij: "This is the bulk of GPIO changes for the v5.5 kernel cycle Core changes: - Expose pull up/down flags for the GPIO character device to userspace. After clear input from the RaspberryPi and Beagle communities, it has been established that prototyping, industrial automation and make communities strongly need this feature, and as we want people to use the character device, we have implemented the simple pull up/down interface for GPIO lines. This means we can specify that a (chip-specific) pull up/down resistor can be enabled, but does not offer fine-grained control such as cases where the resistance of the same pull resistor can be controlled (yet). - Introduce devm_fwnode_gpiod_get_index() and start to phase out the old symbol devm_fwnode_get_index_gpiod_from_child(). - A bit of documentation clean-up work. - Introduce a define for GPIO line directions and deploy it in all GPIO drivers in the drivers/gpio directory. - Add a special callback to populate pin ranges when cooperating with the pin control subsystem and registering ranges as part of adding a gpiolib driver and a gpio_irq_chip driver at the same time. This is also deployed in the Intel Merrifield driver. New drivers: - RDA Micro GPIO controller. - XGS-iproc GPIO driver. Driver improvements: - Wake event and debounce support on the Tegra 186 driver. - Finalize the Aspeed SGPIO driver. - MPC8xxx uses a normal IRQ handler rather than a chained handler" * tag 'gpio-v5.5-1' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-gpio: (64 commits) gpio: Add TODO item for regmap helper Documentation: gpio: driver.rst: Fix warnings gpio: of: Fix bogus reference to gpiod_get_count() gpiolib: Grammar s/manager/managed/ gpio: lynxpoint: Setup correct IRQ handlers MAINTAINERS: Replace my email by one @kernel.org gpiolib: acpi: Make acpi_gpiochip_alloc_event always return AE_OK gpio/mpc8xxx: fix qoriq GPIO reading gpio: mpc8xxx: Don't overwrite default irq_set_type callback gpiolib: acpi: Print pin number on acpi_gpiochip_alloc_event errors gpiolib: fix coding style in gpiod_hog() drm/bridge: ti-tfp410: switch to using fwnode_gpiod_get_index() gpio: merrifield: Pass irqchip when adding gpiochip gpio: merrifield: Add GPIO <-> pin mapping ranges via callback gpiolib: Introduce ->add_pin_ranges() callback gpio: mmio: remove untrue leftover comment gpio: em: Use platform_get_irq() to obtain interrupts gpio: tegra186: Add debounce support gpio: tegra186: Program interrupt route mapping gpio: tegra186: Derive register offsets from bank/port ...
507 lines
11 KiB
C
507 lines
11 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 2016 Texas Instruments
|
|
* Author: Jyri Sarha <jsarha@ti.com>
|
|
*/
|
|
|
|
#include <linux/delay.h>
|
|
#include <linux/fwnode.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/irq.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of_graph.h>
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_bridge.h>
|
|
#include <drm/drm_crtc.h>
|
|
#include <drm/drm_print.h>
|
|
#include <drm/drm_probe_helper.h>
|
|
|
|
#define HOTPLUG_DEBOUNCE_MS 1100
|
|
|
|
struct tfp410 {
|
|
struct drm_bridge bridge;
|
|
struct drm_connector connector;
|
|
unsigned int connector_type;
|
|
|
|
u32 bus_format;
|
|
struct i2c_adapter *ddc;
|
|
struct gpio_desc *hpd;
|
|
int hpd_irq;
|
|
struct delayed_work hpd_work;
|
|
struct gpio_desc *powerdown;
|
|
|
|
struct drm_bridge_timings timings;
|
|
|
|
struct device *dev;
|
|
};
|
|
|
|
static inline struct tfp410 *
|
|
drm_bridge_to_tfp410(struct drm_bridge *bridge)
|
|
{
|
|
return container_of(bridge, struct tfp410, bridge);
|
|
}
|
|
|
|
static inline struct tfp410 *
|
|
drm_connector_to_tfp410(struct drm_connector *connector)
|
|
{
|
|
return container_of(connector, struct tfp410, connector);
|
|
}
|
|
|
|
static int tfp410_get_modes(struct drm_connector *connector)
|
|
{
|
|
struct tfp410 *dvi = drm_connector_to_tfp410(connector);
|
|
struct edid *edid;
|
|
int ret;
|
|
|
|
if (!dvi->ddc)
|
|
goto fallback;
|
|
|
|
edid = drm_get_edid(connector, dvi->ddc);
|
|
if (!edid) {
|
|
DRM_INFO("EDID read failed. Fallback to standard modes\n");
|
|
goto fallback;
|
|
}
|
|
|
|
drm_connector_update_edid_property(connector, edid);
|
|
|
|
ret = drm_add_edid_modes(connector, edid);
|
|
|
|
kfree(edid);
|
|
|
|
return ret;
|
|
|
|
fallback:
|
|
/* No EDID, fallback on the XGA standard modes */
|
|
ret = drm_add_modes_noedid(connector, 1920, 1200);
|
|
|
|
/* And prefer a mode pretty much anything can handle */
|
|
drm_set_preferred_mode(connector, 1024, 768);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static const struct drm_connector_helper_funcs tfp410_con_helper_funcs = {
|
|
.get_modes = tfp410_get_modes,
|
|
};
|
|
|
|
static enum drm_connector_status
|
|
tfp410_connector_detect(struct drm_connector *connector, bool force)
|
|
{
|
|
struct tfp410 *dvi = drm_connector_to_tfp410(connector);
|
|
|
|
if (dvi->hpd) {
|
|
if (gpiod_get_value_cansleep(dvi->hpd))
|
|
return connector_status_connected;
|
|
else
|
|
return connector_status_disconnected;
|
|
}
|
|
|
|
if (dvi->ddc) {
|
|
if (drm_probe_ddc(dvi->ddc))
|
|
return connector_status_connected;
|
|
else
|
|
return connector_status_disconnected;
|
|
}
|
|
|
|
return connector_status_unknown;
|
|
}
|
|
|
|
static const struct drm_connector_funcs tfp410_con_funcs = {
|
|
.detect = tfp410_connector_detect,
|
|
.fill_modes = drm_helper_probe_single_connector_modes,
|
|
.destroy = drm_connector_cleanup,
|
|
.reset = drm_atomic_helper_connector_reset,
|
|
.atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
|
|
.atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
|
|
};
|
|
|
|
static int tfp410_attach(struct drm_bridge *bridge)
|
|
{
|
|
struct tfp410 *dvi = drm_bridge_to_tfp410(bridge);
|
|
int ret;
|
|
|
|
if (!bridge->encoder) {
|
|
dev_err(dvi->dev, "Missing encoder\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
if (dvi->hpd_irq >= 0)
|
|
dvi->connector.polled = DRM_CONNECTOR_POLL_HPD;
|
|
else
|
|
dvi->connector.polled = DRM_CONNECTOR_POLL_CONNECT | DRM_CONNECTOR_POLL_DISCONNECT;
|
|
|
|
drm_connector_helper_add(&dvi->connector,
|
|
&tfp410_con_helper_funcs);
|
|
ret = drm_connector_init_with_ddc(bridge->dev, &dvi->connector,
|
|
&tfp410_con_funcs,
|
|
dvi->connector_type,
|
|
dvi->ddc);
|
|
if (ret) {
|
|
dev_err(dvi->dev, "drm_connector_init() failed: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
drm_display_info_set_bus_formats(&dvi->connector.display_info,
|
|
&dvi->bus_format, 1);
|
|
|
|
drm_connector_attach_encoder(&dvi->connector,
|
|
bridge->encoder);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void tfp410_enable(struct drm_bridge *bridge)
|
|
{
|
|
struct tfp410 *dvi = drm_bridge_to_tfp410(bridge);
|
|
|
|
gpiod_set_value_cansleep(dvi->powerdown, 0);
|
|
}
|
|
|
|
static void tfp410_disable(struct drm_bridge *bridge)
|
|
{
|
|
struct tfp410 *dvi = drm_bridge_to_tfp410(bridge);
|
|
|
|
gpiod_set_value_cansleep(dvi->powerdown, 1);
|
|
}
|
|
|
|
static const struct drm_bridge_funcs tfp410_bridge_funcs = {
|
|
.attach = tfp410_attach,
|
|
.enable = tfp410_enable,
|
|
.disable = tfp410_disable,
|
|
};
|
|
|
|
static void tfp410_hpd_work_func(struct work_struct *work)
|
|
{
|
|
struct tfp410 *dvi;
|
|
|
|
dvi = container_of(work, struct tfp410, hpd_work.work);
|
|
|
|
if (dvi->bridge.dev)
|
|
drm_helper_hpd_irq_event(dvi->bridge.dev);
|
|
}
|
|
|
|
static irqreturn_t tfp410_hpd_irq_thread(int irq, void *arg)
|
|
{
|
|
struct tfp410 *dvi = arg;
|
|
|
|
mod_delayed_work(system_wq, &dvi->hpd_work,
|
|
msecs_to_jiffies(HOTPLUG_DEBOUNCE_MS));
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static const struct drm_bridge_timings tfp410_default_timings = {
|
|
.input_bus_flags = DRM_BUS_FLAG_PIXDATA_SAMPLE_POSEDGE
|
|
| DRM_BUS_FLAG_DE_HIGH,
|
|
.setup_time_ps = 1200,
|
|
.hold_time_ps = 1300,
|
|
};
|
|
|
|
static int tfp410_parse_timings(struct tfp410 *dvi, bool i2c)
|
|
{
|
|
struct drm_bridge_timings *timings = &dvi->timings;
|
|
struct device_node *ep;
|
|
u32 pclk_sample = 0;
|
|
u32 bus_width = 24;
|
|
s32 deskew = 0;
|
|
|
|
/* Start with defaults. */
|
|
*timings = tfp410_default_timings;
|
|
|
|
if (i2c)
|
|
/*
|
|
* In I2C mode timings are configured through the I2C interface.
|
|
* As the driver doesn't support I2C configuration yet, we just
|
|
* go with the defaults (BSEL=1, DSEL=1, DKEN=0, EDGE=1).
|
|
*/
|
|
return 0;
|
|
|
|
/*
|
|
* In non-I2C mode, timings are configured through the BSEL, DSEL, DKEN
|
|
* and EDGE pins. They are specified in DT through endpoint properties
|
|
* and vendor-specific properties.
|
|
*/
|
|
ep = of_graph_get_endpoint_by_regs(dvi->dev->of_node, 0, 0);
|
|
if (!ep)
|
|
return -EINVAL;
|
|
|
|
/* Get the sampling edge from the endpoint. */
|
|
of_property_read_u32(ep, "pclk-sample", &pclk_sample);
|
|
of_property_read_u32(ep, "bus-width", &bus_width);
|
|
of_node_put(ep);
|
|
|
|
timings->input_bus_flags = DRM_BUS_FLAG_DE_HIGH;
|
|
|
|
switch (pclk_sample) {
|
|
case 0:
|
|
timings->input_bus_flags |= DRM_BUS_FLAG_PIXDATA_SAMPLE_NEGEDGE
|
|
| DRM_BUS_FLAG_SYNC_SAMPLE_NEGEDGE;
|
|
break;
|
|
case 1:
|
|
timings->input_bus_flags |= DRM_BUS_FLAG_PIXDATA_SAMPLE_POSEDGE
|
|
| DRM_BUS_FLAG_SYNC_SAMPLE_POSEDGE;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
switch (bus_width) {
|
|
case 12:
|
|
dvi->bus_format = MEDIA_BUS_FMT_RGB888_2X12_LE;
|
|
break;
|
|
case 24:
|
|
dvi->bus_format = MEDIA_BUS_FMT_RGB888_1X24;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Get the setup and hold time from vendor-specific properties. */
|
|
of_property_read_u32(dvi->dev->of_node, "ti,deskew", (u32 *)&deskew);
|
|
if (deskew < -4 || deskew > 3)
|
|
return -EINVAL;
|
|
|
|
timings->setup_time_ps = min(0, 1200 - 350 * deskew);
|
|
timings->hold_time_ps = min(0, 1300 + 350 * deskew);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tfp410_get_connector_properties(struct tfp410 *dvi)
|
|
{
|
|
struct device_node *connector_node, *ddc_phandle;
|
|
int ret = 0;
|
|
|
|
/* port@1 is the connector node */
|
|
connector_node = of_graph_get_remote_node(dvi->dev->of_node, 1, -1);
|
|
if (!connector_node)
|
|
return -ENODEV;
|
|
|
|
if (of_device_is_compatible(connector_node, "hdmi-connector"))
|
|
dvi->connector_type = DRM_MODE_CONNECTOR_HDMIA;
|
|
else
|
|
dvi->connector_type = DRM_MODE_CONNECTOR_DVID;
|
|
|
|
dvi->hpd = fwnode_gpiod_get_index(&connector_node->fwnode,
|
|
"hpd", 0, GPIOD_IN, "hpd");
|
|
if (IS_ERR(dvi->hpd)) {
|
|
ret = PTR_ERR(dvi->hpd);
|
|
dvi->hpd = NULL;
|
|
if (ret == -ENOENT)
|
|
ret = 0;
|
|
else
|
|
goto fail;
|
|
}
|
|
|
|
ddc_phandle = of_parse_phandle(connector_node, "ddc-i2c-bus", 0);
|
|
if (!ddc_phandle)
|
|
goto fail;
|
|
|
|
dvi->ddc = of_get_i2c_adapter_by_node(ddc_phandle);
|
|
if (dvi->ddc)
|
|
dev_info(dvi->dev, "Connector's ddc i2c bus found\n");
|
|
else
|
|
ret = -EPROBE_DEFER;
|
|
|
|
of_node_put(ddc_phandle);
|
|
|
|
fail:
|
|
of_node_put(connector_node);
|
|
return ret;
|
|
}
|
|
|
|
static int tfp410_init(struct device *dev, bool i2c)
|
|
{
|
|
struct tfp410 *dvi;
|
|
int ret;
|
|
|
|
if (!dev->of_node) {
|
|
dev_err(dev, "device-tree data is missing\n");
|
|
return -ENXIO;
|
|
}
|
|
|
|
dvi = devm_kzalloc(dev, sizeof(*dvi), GFP_KERNEL);
|
|
if (!dvi)
|
|
return -ENOMEM;
|
|
dev_set_drvdata(dev, dvi);
|
|
|
|
dvi->bridge.funcs = &tfp410_bridge_funcs;
|
|
dvi->bridge.of_node = dev->of_node;
|
|
dvi->bridge.timings = &dvi->timings;
|
|
dvi->dev = dev;
|
|
|
|
ret = tfp410_parse_timings(dvi, i2c);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
ret = tfp410_get_connector_properties(dvi);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
dvi->powerdown = devm_gpiod_get_optional(dev, "powerdown",
|
|
GPIOD_OUT_HIGH);
|
|
if (IS_ERR(dvi->powerdown)) {
|
|
dev_err(dev, "failed to parse powerdown gpio\n");
|
|
return PTR_ERR(dvi->powerdown);
|
|
}
|
|
|
|
if (dvi->hpd)
|
|
dvi->hpd_irq = gpiod_to_irq(dvi->hpd);
|
|
else
|
|
dvi->hpd_irq = -ENXIO;
|
|
|
|
if (dvi->hpd_irq >= 0) {
|
|
INIT_DELAYED_WORK(&dvi->hpd_work, tfp410_hpd_work_func);
|
|
|
|
ret = devm_request_threaded_irq(dev, dvi->hpd_irq,
|
|
NULL, tfp410_hpd_irq_thread, IRQF_TRIGGER_RISING |
|
|
IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
|
|
"hdmi-hpd", dvi);
|
|
if (ret) {
|
|
DRM_ERROR("failed to register hpd interrupt\n");
|
|
goto fail;
|
|
}
|
|
}
|
|
|
|
drm_bridge_add(&dvi->bridge);
|
|
|
|
return 0;
|
|
fail:
|
|
i2c_put_adapter(dvi->ddc);
|
|
if (dvi->hpd)
|
|
gpiod_put(dvi->hpd);
|
|
return ret;
|
|
}
|
|
|
|
static int tfp410_fini(struct device *dev)
|
|
{
|
|
struct tfp410 *dvi = dev_get_drvdata(dev);
|
|
|
|
if (dvi->hpd_irq >= 0)
|
|
cancel_delayed_work_sync(&dvi->hpd_work);
|
|
|
|
drm_bridge_remove(&dvi->bridge);
|
|
|
|
if (dvi->ddc)
|
|
i2c_put_adapter(dvi->ddc);
|
|
if (dvi->hpd)
|
|
gpiod_put(dvi->hpd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tfp410_probe(struct platform_device *pdev)
|
|
{
|
|
return tfp410_init(&pdev->dev, false);
|
|
}
|
|
|
|
static int tfp410_remove(struct platform_device *pdev)
|
|
{
|
|
return tfp410_fini(&pdev->dev);
|
|
}
|
|
|
|
static const struct of_device_id tfp410_match[] = {
|
|
{ .compatible = "ti,tfp410" },
|
|
{},
|
|
};
|
|
MODULE_DEVICE_TABLE(of, tfp410_match);
|
|
|
|
static struct platform_driver tfp410_platform_driver = {
|
|
.probe = tfp410_probe,
|
|
.remove = tfp410_remove,
|
|
.driver = {
|
|
.name = "tfp410-bridge",
|
|
.of_match_table = tfp410_match,
|
|
},
|
|
};
|
|
|
|
#if IS_ENABLED(CONFIG_I2C)
|
|
/* There is currently no i2c functionality. */
|
|
static int tfp410_i2c_probe(struct i2c_client *client,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
int reg;
|
|
|
|
if (!client->dev.of_node ||
|
|
of_property_read_u32(client->dev.of_node, "reg", ®)) {
|
|
dev_err(&client->dev,
|
|
"Can't get i2c reg property from device-tree\n");
|
|
return -ENXIO;
|
|
}
|
|
|
|
return tfp410_init(&client->dev, true);
|
|
}
|
|
|
|
static int tfp410_i2c_remove(struct i2c_client *client)
|
|
{
|
|
return tfp410_fini(&client->dev);
|
|
}
|
|
|
|
static const struct i2c_device_id tfp410_i2c_ids[] = {
|
|
{ "tfp410", 0 },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(i2c, tfp410_i2c_ids);
|
|
|
|
static struct i2c_driver tfp410_i2c_driver = {
|
|
.driver = {
|
|
.name = "tfp410",
|
|
.of_match_table = of_match_ptr(tfp410_match),
|
|
},
|
|
.id_table = tfp410_i2c_ids,
|
|
.probe = tfp410_i2c_probe,
|
|
.remove = tfp410_i2c_remove,
|
|
};
|
|
#endif /* IS_ENABLED(CONFIG_I2C) */
|
|
|
|
static struct {
|
|
uint i2c:1;
|
|
uint platform:1;
|
|
} tfp410_registered_driver;
|
|
|
|
static int __init tfp410_module_init(void)
|
|
{
|
|
int ret;
|
|
|
|
#if IS_ENABLED(CONFIG_I2C)
|
|
ret = i2c_add_driver(&tfp410_i2c_driver);
|
|
if (ret)
|
|
pr_err("%s: registering i2c driver failed: %d",
|
|
__func__, ret);
|
|
else
|
|
tfp410_registered_driver.i2c = 1;
|
|
#endif
|
|
|
|
ret = platform_driver_register(&tfp410_platform_driver);
|
|
if (ret)
|
|
pr_err("%s: registering platform driver failed: %d",
|
|
__func__, ret);
|
|
else
|
|
tfp410_registered_driver.platform = 1;
|
|
|
|
if (tfp410_registered_driver.i2c ||
|
|
tfp410_registered_driver.platform)
|
|
return 0;
|
|
|
|
return ret;
|
|
}
|
|
module_init(tfp410_module_init);
|
|
|
|
static void __exit tfp410_module_exit(void)
|
|
{
|
|
#if IS_ENABLED(CONFIG_I2C)
|
|
if (tfp410_registered_driver.i2c)
|
|
i2c_del_driver(&tfp410_i2c_driver);
|
|
#endif
|
|
if (tfp410_registered_driver.platform)
|
|
platform_driver_unregister(&tfp410_platform_driver);
|
|
}
|
|
module_exit(tfp410_module_exit);
|
|
|
|
MODULE_AUTHOR("Jyri Sarha <jsarha@ti.com>");
|
|
MODULE_DESCRIPTION("TI TFP410 DVI bridge driver");
|
|
MODULE_LICENSE("GPL");
|