mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-18 02:46:06 +00:00
e965a70727
After removal of the legacy EEPROM driver and I2C_CLASS_DDC support in olpc_dcon there's no i2c client driver left supporting I2C_CLASS_DDC. Class-based device auto-detection is a legacy mechanism and shouldn't be used in new code. So we can remove this class completely now. Acked-by: Alex Deucher <alexander.deucher@amd.com> Acked-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org> Acked-by: Harry Wentland <harry.wentland@amd.com> Acked-by: Heiko Stuebner <heiko@sntech.de> Acked-by: Jani Nikula <jani.nikula@intel.com> Acked-by: Jernej Skrabec <jernej.skrabec@gmail.com> Reviewed-by: Thomas Zimmermann <tzimmermann@suse.de> Reviewed-by: Andi Shyti <andi.shyti@linux.intel.com> Reviewed-by: AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com> Reviewed-by: Neil Armstrong <neil.armstrong@linaro.org> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com> Signed-off-by: Wolfram Sang <wsa@kernel.org>
179 lines
4.2 KiB
C
179 lines
4.2 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Copyright (C) 2023 Loongson Technology Corporation Limited
|
|
*/
|
|
|
|
#include <drm/drm_managed.h>
|
|
|
|
#include "lsdc_drv.h"
|
|
#include "lsdc_output.h"
|
|
|
|
/*
|
|
* __lsdc_gpio_i2c_set - set the state of a gpio pin indicated by mask
|
|
* @mask: gpio pin mask
|
|
* @state: "0" for low, "1" for high
|
|
*/
|
|
static void __lsdc_gpio_i2c_set(struct lsdc_i2c * const li2c, int mask, int state)
|
|
{
|
|
struct lsdc_device *ldev = to_lsdc(li2c->ddev);
|
|
unsigned long flags;
|
|
u8 val;
|
|
|
|
spin_lock_irqsave(&ldev->reglock, flags);
|
|
|
|
if (state) {
|
|
/*
|
|
* Setting this pin as input directly, write 1 for input.
|
|
* The external pull-up resistor will pull the level up
|
|
*/
|
|
val = readb(li2c->dir_reg);
|
|
val |= mask;
|
|
writeb(val, li2c->dir_reg);
|
|
} else {
|
|
/* First set this pin as output, write 0 for output */
|
|
val = readb(li2c->dir_reg);
|
|
val &= ~mask;
|
|
writeb(val, li2c->dir_reg);
|
|
|
|
/* Then, make this pin output 0 */
|
|
val = readb(li2c->dat_reg);
|
|
val &= ~mask;
|
|
writeb(val, li2c->dat_reg);
|
|
}
|
|
|
|
spin_unlock_irqrestore(&ldev->reglock, flags);
|
|
}
|
|
|
|
/*
|
|
* __lsdc_gpio_i2c_get - read value back from the gpio pin indicated by mask
|
|
* @mask: gpio pin mask
|
|
* return "0" for low, "1" for high
|
|
*/
|
|
static int __lsdc_gpio_i2c_get(struct lsdc_i2c * const li2c, int mask)
|
|
{
|
|
struct lsdc_device *ldev = to_lsdc(li2c->ddev);
|
|
unsigned long flags;
|
|
u8 val;
|
|
|
|
spin_lock_irqsave(&ldev->reglock, flags);
|
|
|
|
/* First set this pin as input */
|
|
val = readb(li2c->dir_reg);
|
|
val |= mask;
|
|
writeb(val, li2c->dir_reg);
|
|
|
|
/* Then get level state from this pin */
|
|
val = readb(li2c->dat_reg);
|
|
|
|
spin_unlock_irqrestore(&ldev->reglock, flags);
|
|
|
|
return (val & mask) ? 1 : 0;
|
|
}
|
|
|
|
static void lsdc_gpio_i2c_set_sda(void *i2c, int state)
|
|
{
|
|
struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
|
|
/* set state on the li2c->sda pin */
|
|
return __lsdc_gpio_i2c_set(li2c, li2c->sda, state);
|
|
}
|
|
|
|
static void lsdc_gpio_i2c_set_scl(void *i2c, int state)
|
|
{
|
|
struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
|
|
/* set state on the li2c->scl pin */
|
|
return __lsdc_gpio_i2c_set(li2c, li2c->scl, state);
|
|
}
|
|
|
|
static int lsdc_gpio_i2c_get_sda(void *i2c)
|
|
{
|
|
struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
|
|
/* read value from the li2c->sda pin */
|
|
return __lsdc_gpio_i2c_get(li2c, li2c->sda);
|
|
}
|
|
|
|
static int lsdc_gpio_i2c_get_scl(void *i2c)
|
|
{
|
|
struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
|
|
/* read the value from the li2c->scl pin */
|
|
return __lsdc_gpio_i2c_get(li2c, li2c->scl);
|
|
}
|
|
|
|
static void lsdc_destroy_i2c(struct drm_device *ddev, void *data)
|
|
{
|
|
struct lsdc_i2c *li2c = (struct lsdc_i2c *)data;
|
|
|
|
if (li2c) {
|
|
i2c_del_adapter(&li2c->adapter);
|
|
kfree(li2c);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* The DC in ls7a1000/ls7a2000/ls2k2000 has builtin gpio hardware
|
|
*
|
|
* @reg_base: gpio reg base
|
|
* @index: output channel index, 0 for PIPE0, 1 for PIPE1
|
|
*/
|
|
int lsdc_create_i2c_chan(struct drm_device *ddev,
|
|
struct lsdc_display_pipe *dispipe,
|
|
unsigned int index)
|
|
{
|
|
struct lsdc_device *ldev = to_lsdc(ddev);
|
|
struct i2c_adapter *adapter;
|
|
struct lsdc_i2c *li2c;
|
|
int ret;
|
|
|
|
li2c = kzalloc(sizeof(*li2c), GFP_KERNEL);
|
|
if (!li2c)
|
|
return -ENOMEM;
|
|
|
|
dispipe->li2c = li2c;
|
|
|
|
if (index == 0) {
|
|
li2c->sda = 0x01; /* pin 0 */
|
|
li2c->scl = 0x02; /* pin 1 */
|
|
} else if (index == 1) {
|
|
li2c->sda = 0x04; /* pin 2 */
|
|
li2c->scl = 0x08; /* pin 3 */
|
|
} else {
|
|
return -ENOENT;
|
|
}
|
|
|
|
li2c->ddev = ddev;
|
|
li2c->dir_reg = ldev->reg_base + LS7A_DC_GPIO_DIR_REG;
|
|
li2c->dat_reg = ldev->reg_base + LS7A_DC_GPIO_DAT_REG;
|
|
|
|
li2c->bit.setsda = lsdc_gpio_i2c_set_sda;
|
|
li2c->bit.setscl = lsdc_gpio_i2c_set_scl;
|
|
li2c->bit.getsda = lsdc_gpio_i2c_get_sda;
|
|
li2c->bit.getscl = lsdc_gpio_i2c_get_scl;
|
|
li2c->bit.udelay = 5;
|
|
li2c->bit.timeout = usecs_to_jiffies(2200);
|
|
li2c->bit.data = li2c;
|
|
|
|
adapter = &li2c->adapter;
|
|
adapter->algo_data = &li2c->bit;
|
|
adapter->owner = THIS_MODULE;
|
|
adapter->dev.parent = ddev->dev;
|
|
adapter->nr = -1;
|
|
|
|
snprintf(adapter->name, sizeof(adapter->name), "lsdc-i2c%u", index);
|
|
|
|
i2c_set_adapdata(adapter, li2c);
|
|
|
|
ret = i2c_bit_add_bus(adapter);
|
|
if (ret) {
|
|
kfree(li2c);
|
|
return ret;
|
|
}
|
|
|
|
ret = drmm_add_action_or_reset(ddev, lsdc_destroy_i2c, li2c);
|
|
if (ret)
|
|
return ret;
|
|
|
|
drm_info(ddev, "%s(sda pin mask=%u, scl pin mask=%u) created\n",
|
|
adapter->name, li2c->sda, li2c->scl);
|
|
|
|
return 0;
|
|
}
|