380 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			380 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright (c) 2021 MediaTek Inc.
 | |
|  */
 | |
| 
 | |
| #include <linux/gpio/consumer.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/i2c.h>
 | |
| #include <linux/i2c-dev.h>
 | |
| #include <linux/platform_device.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/string.h>
 | |
| #include <linux/gpio.h>
 | |
| #include <linux/of_gpio.h>
 | |
| 
 | |
| #include "gate_i2c.h"
 | |
| 
 | |
| #if IS_ENABLED(CONFIG_LEDS_MTK)
 | |
| #define LEDS_BRIGHTNESS_CHANGED
 | |
| #include <linux/leds-mtk.h>
 | |
| #endif
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Define
 | |
|  *****************************************************************************/
 | |
| #define GATE_I2C_ID_NAME "gate_ic_i2c_rt4831"
 | |
| 
 | |
| #undef pr_fmt
 | |
| #define pr_fmt(fmt) KBUILD_MODNAME " %s(%d) :[GATE][I2C] " fmt, __func__, __LINE__
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Define Register
 | |
|  *****************************************************************************/
 | |
| #define BACKLIGHT_CONFIG_1		0x02
 | |
| #define BACKLIGHT_CONFIG_2		0x03
 | |
| #define BACKLIGHT_BRIGHTNESS_LSB	0x04
 | |
| #define BACKLIGHT_BRIGHTNESS_MSB	0x05
 | |
| #define BACKLIGHT_AUTO_FREQ_LOW		0x06
 | |
| #define BACKLIGHT_AUTO_FREQ_HIGH	0x07
 | |
| #define BACKLIGHT_ENABLE		0x08
 | |
| #define FLAGS				0x0F
 | |
| #define BACKLIGHT_OPTION_1		0x10
 | |
| #define BACKLIGHT_OPTION_2		0x11
 | |
| #define BACKLIGHT_SMOOTH		0x14
 | |
| 
 | |
| #define BRIGHTNESS_HIGN_OFFSET		0x3
 | |
| #define BRIGHTNESS_HIGN_MASK		0xFF
 | |
| #define BRIGHTNESS_LOW_OFFSET		0x0
 | |
| #define BRIGHTNESS_LOW_MASK		0x7
 | |
| 
 | |
| #define DISPLAY_BIAS_CONFIGURATION_1	0x09
 | |
| #define DISPLAY_BIAS_CONFIGURATION_2	0x0A
 | |
| #define DISPLAY_BIAS_CONFIGURATION_3	0x0B
 | |
| #define LCM_BIAS			0x0C
 | |
| #define VPOS_BIAS			0x0D
 | |
| #define VNEG_BIAS			0x0E
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * GLobal Variable
 | |
|  *****************************************************************************/
 | |
| static const struct of_device_id _gate_ic_i2c_of_match[] = {
 | |
| 	{
 | |
| 		.compatible = "mediatek,gate-ic-i2c",
 | |
| 	 },
 | |
| 	{}
 | |
| };
 | |
| 
 | |
| static struct i2c_client *_gate_ic_i2c_client;
 | |
| 
 | |
| struct gate_ic_client {
 | |
| 	struct i2c_client *i2c_client;
 | |
| 	struct gpio_desc *pinctrl;
 | |
| 	struct device *dev;
 | |
| 	bool pwm_enable;
 | |
| 	atomic_t gate_ic_power_status;
 | |
| 	atomic_t backlight_status;
 | |
| };
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Driver Functions
 | |
|  *****************************************************************************/
 | |
| static void _gate_ic_backlight_enable(void)
 | |
| {
 | |
| 	/*BL enable*/
 | |
| 	struct gate_ic_client *gate_client = i2c_get_clientdata(_gate_ic_i2c_client);
 | |
| 
 | |
| 	pr_info("%s+\n", __func__);
 | |
| 
 | |
| 	if (gate_client->pwm_enable) {
 | |
| 		_gate_ic_i2c_write_bytes(BACKLIGHT_CONFIG_1, 0x6B);
 | |
| 	} else {
 | |
| 		_gate_ic_i2c_write_bytes(BACKLIGHT_CONFIG_1, 0x68);
 | |
| 		if (!atomic_read(&gate_client->backlight_status))
 | |
| 			_gate_ic_backlight_set(0);
 | |
| 	}
 | |
| 
 | |
| 	_gate_ic_i2c_write_bytes(BACKLIGHT_CONFIG_2, 0x9D);
 | |
| 	_gate_ic_i2c_write_bytes(BACKLIGHT_OPTION_1, 0x06);
 | |
| 	_gate_ic_i2c_write_bytes(BACKLIGHT_OPTION_2, 0xB7);
 | |
| 	/*bias enable*/
 | |
| 	_gate_ic_i2c_write_bytes(BACKLIGHT_ENABLE, 0x15);
 | |
| 	_gate_ic_i2c_write_bytes(BACKLIGHT_SMOOTH, 0x03);
 | |
| }
 | |
| 
 | |
| #ifdef LEDS_BRIGHTNESS_CHANGED
 | |
| int _backlight_changed_event(struct notifier_block *nb, unsigned long event,
 | |
| 	void *v)
 | |
| {
 | |
| 	struct led_conf_info *led_conf;
 | |
| 	struct gate_ic_client *gate_client = i2c_get_clientdata(_gate_ic_i2c_client);
 | |
| 
 | |
| 	led_conf = (struct led_conf_info *)v;
 | |
| 
 | |
| 	switch (event) {
 | |
| 	case LED_BRIGHTNESS_CHANGED:
 | |
| 		if (led_conf->cdev.brightness > 0)
 | |
| 			atomic_set(&gate_client->backlight_status, 1);
 | |
| 		else
 | |
| 			atomic_set(&gate_client->backlight_status, 0);
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	return NOTIFY_DONE;
 | |
| }
 | |
| 
 | |
| static struct notifier_block leds_init_notifier = {
 | |
| 	.notifier_call = _backlight_changed_event,
 | |
| };
 | |
| #endif
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Extern Area
 | |
|  *****************************************************************************/
 | |
| 
 | |
| int _gate_ic_backlight_set(unsigned int hw_level)
 | |
| {
 | |
| 	int level_l, level_h;
 | |
| 	struct i2c_client *client = _gate_ic_i2c_client;
 | |
| 	char cmd_buf[3] = { 0x00, 0x00, 0x00 };
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	level_h = (hw_level >> BRIGHTNESS_HIGN_OFFSET) & BRIGHTNESS_HIGN_MASK;
 | |
| 	level_l = (hw_level >> BRIGHTNESS_LOW_OFFSET) & BRIGHTNESS_LOW_MASK;
 | |
| 
 | |
| 	cmd_buf[0] = BACKLIGHT_BRIGHTNESS_LSB;
 | |
| 	cmd_buf[1] = level_l;
 | |
| 	cmd_buf[2] = level_h;
 | |
| 
 | |
| 	ret = i2c_master_send(client, cmd_buf, 3);
 | |
| 	if (ret < 0)
 | |
| 		pr_info("ERROR %d!! i2c write data fail 0x%0x, 0x%0x, 0x%0x !!\n",
 | |
| 				ret, cmd_buf[0], cmd_buf[1], cmd_buf[2]);
 | |
| 	return ret;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_backlight_set);
 | |
| 
 | |
| int _gate_ic_i2c_read_bytes(unsigned char addr, unsigned char *returnData)
 | |
| {
 | |
| 	char cmd_buf[2] = { 0x00, 0x00 };
 | |
| 	char readData = 0;
 | |
| 	int ret = 0;
 | |
| 	struct i2c_client *client = _gate_ic_i2c_client;
 | |
| 
 | |
| 	if (client == NULL) {
 | |
| 		pr_info("ERROR!! _gate_ic_i2c_client is null\n");
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	cmd_buf[0] = addr;
 | |
| 	ret = i2c_master_send(client, &cmd_buf[0], 1);
 | |
| 	ret = i2c_master_recv(client, &cmd_buf[1], 1);
 | |
| 	if (ret < 0)
 | |
| 		pr_info("ERROR %d!! i2c read data 0x%0x fail !!\n", ret, addr);
 | |
| 
 | |
| 	readData = cmd_buf[1];
 | |
| 	*returnData = readData;
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_i2c_read_bytes);
 | |
| 
 | |
| int _gate_ic_i2c_write_bytes(unsigned char addr, unsigned char value)
 | |
| {
 | |
| 	int ret = 0;
 | |
| 	struct i2c_client *client = _gate_ic_i2c_client;
 | |
| 	char write_data[2] = { 0 };
 | |
| 
 | |
| 	if (client == NULL) {
 | |
| 		pr_info("ERROR!! _gate_ic_i2c_client is null\n");
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	write_data[0] = addr;
 | |
| 	write_data[1] = value;
 | |
| 	ret = i2c_master_send(client, write_data, 2);
 | |
| 	if (ret < 0)
 | |
| 		pr_info("ERROR %d!! i2c write data fail 0x%0x, 0x%0x !!\n",
 | |
| 				ret, addr, value);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_i2c_write_bytes);
 | |
| 
 | |
| void _gate_ic_Power_on(void)
 | |
| {
 | |
| 	struct gate_ic_client *gate_client = i2c_get_clientdata(_gate_ic_i2c_client);
 | |
| 
 | |
| 	pr_info("%s+: status = (%d, %d)\n", __func__,
 | |
| 		atomic_read(&gate_client->gate_ic_power_status),
 | |
| 		atomic_read(&gate_client->backlight_status));
 | |
| 
 | |
| 	if (IS_ERR(gate_client->pinctrl)) {
 | |
| 		pr_info("ERROR!! pinctrl is error!\n");
 | |
| 	} else if (!atomic_read(&gate_client->gate_ic_power_status)) {
 | |
| 		gate_client->pinctrl = devm_gpiod_get(gate_client->dev, "gate-power",
 | |
| 				   GPIOD_OUT_HIGH);
 | |
| 		if (IS_ERR(gate_client->pinctrl)) {
 | |
| 			pr_info("ERROR!! Failed to get gpio: %d\n",
 | |
| 				PTR_ERR(gate_client->pinctrl));
 | |
| 			return;
 | |
| 		}
 | |
| 		gpiod_set_value(gate_client->pinctrl, 1);
 | |
| 		devm_gpiod_put(gate_client->dev, gate_client->pinctrl);
 | |
| 
 | |
| 		atomic_set(&gate_client->gate_ic_power_status, 1);
 | |
| 		_gate_ic_backlight_enable();
 | |
| 	}
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_Power_on);
 | |
| 
 | |
| void _gate_ic_Power_off(void)
 | |
| {
 | |
| 	struct gate_ic_client *gate_client = i2c_get_clientdata(_gate_ic_i2c_client);
 | |
| 
 | |
| 	pr_info("%s+: status = (%d, %d)\n", __func__,
 | |
| 		atomic_read(&gate_client->gate_ic_power_status),
 | |
| 		atomic_read(&gate_client->backlight_status));
 | |
| 
 | |
| 	if (IS_ERR(gate_client->pinctrl)) {
 | |
| 		pr_info("ERROR!! pinctrl is error!\n");
 | |
| 	} else if (atomic_read(&gate_client->gate_ic_power_status) &&
 | |
| 			!atomic_read(&gate_client->backlight_status)) {
 | |
| 		gate_client->pinctrl = devm_gpiod_get(gate_client->dev, "gate-power",
 | |
| 				   GPIOD_OUT_HIGH);
 | |
| 		if (IS_ERR(gate_client->pinctrl)) {
 | |
| 			pr_info("ERROR!! Failed to get gpio: %d\n",
 | |
| 				PTR_ERR(gate_client->pinctrl));
 | |
| 			return;
 | |
| 		}
 | |
| 		gpiod_set_value(gate_client->pinctrl, 0);
 | |
| 		devm_gpiod_put(gate_client->dev, gate_client->pinctrl);
 | |
| 
 | |
| 		atomic_set(&gate_client->gate_ic_power_status, 0);
 | |
| 	}
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_Power_off);
 | |
| 
 | |
| void _gate_ic_i2c_panel_bias_enable(unsigned int power_status)
 | |
| {
 | |
| 	pr_info("%s+\n", __func__);
 | |
| 
 | |
| 	if (power_status) {
 | |
| 		_gate_ic_i2c_write_bytes(DISPLAY_BIAS_CONFIGURATION_2, 0x11);
 | |
| 		_gate_ic_i2c_write_bytes(DISPLAY_BIAS_CONFIGURATION_3, 0x00);
 | |
| 		/* set bias to 5.4v */
 | |
| 		_gate_ic_i2c_write_bytes(LCM_BIAS, 0x24);
 | |
| 		_gate_ic_i2c_write_bytes(VPOS_BIAS, 0x1c);
 | |
| 		_gate_ic_i2c_write_bytes(VNEG_BIAS, 0x1c);
 | |
| 		/* set dsv FPWM mode */
 | |
| 		_gate_ic_i2c_write_bytes(0xF0, 0x69);
 | |
| 		_gate_ic_i2c_write_bytes(0xB1, 0x6c);
 | |
| 		/* bias enable */
 | |
| 		_gate_ic_i2c_write_bytes(DISPLAY_BIAS_CONFIGURATION_1, 0x9e);
 | |
| 	} else {
 | |
| 		_gate_ic_i2c_write_bytes(DISPLAY_BIAS_CONFIGURATION_1, 0x18);
 | |
| 	}
 | |
| }
 | |
| EXPORT_SYMBOL_GPL(_gate_ic_i2c_panel_bias_enable);
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Function
 | |
|  *****************************************************************************/
 | |
| 
 | |
| static int _gate_ic_i2c_probe(struct i2c_client *client,
 | |
| 	const struct i2c_device_id *id)
 | |
| {
 | |
| 	struct gate_ic_client *gate_client;
 | |
| 	struct device *dev = &client->dev;
 | |
| 	int status;
 | |
| 	int pwm_enable;
 | |
| 
 | |
| 	pr_info("%s+: client name=%s addr=0x%x\n",
 | |
| 		__func__, client->name, client->addr);
 | |
| 
 | |
| 	gate_client = devm_kzalloc(&client->dev, sizeof(struct gate_ic_client), GFP_KERNEL);
 | |
| 
 | |
| 	if (!gate_client)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	gate_client->dev = &client->dev;
 | |
| 	gate_client->i2c_client = client;
 | |
| 	gate_client->pinctrl = devm_gpiod_get(&client->dev, "gate-power",
 | |
| 				   GPIOD_OUT_HIGH);
 | |
| 	if (IS_ERR(gate_client->pinctrl)) {
 | |
| 		status = PTR_ERR(gate_client->pinctrl);
 | |
| 		pr_info("ERROR!! Failed to enable gpio: %d\n", status);
 | |
| 		return status;
 | |
| 	}
 | |
| 	devm_gpiod_put(gate_client->dev, gate_client->pinctrl);
 | |
| 	i2c_set_clientdata(client, gate_client);
 | |
| 	_gate_ic_i2c_client = client;
 | |
| 	atomic_set(&gate_client->gate_ic_power_status, 1);
 | |
| 	atomic_set(&gate_client->gate_ic_power_status, 1);
 | |
| 
 | |
| 	status = of_property_read_u32(dev->of_node, "pwm-enable", &pwm_enable);
 | |
| 	if (status) {
 | |
| 		gate_client->pwm_enable = 0;
 | |
| 		pr_info("ERROR!! Failed to get pwm-enable, use default value 0");
 | |
| 	} else {
 | |
| 		gate_client->pwm_enable = pwm_enable;
 | |
| 	}
 | |
| #ifdef LEDS_BRIGHTNESS_CHANGED
 | |
| 	mtk_leds_register_notifier(&leds_init_notifier);
 | |
| #endif
 | |
| 
 | |
| 	pr_info("%s-: pwm-enable is %d", __func__, pwm_enable);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int _gate_ic_i2c_remove(struct i2c_client *client)
 | |
| {
 | |
| 	struct gate_ic_client *gate_client;
 | |
| 
 | |
| 	pr_info("%s+\n", __func__);
 | |
| 
 | |
| 	gate_client = i2c_get_clientdata(client);
 | |
| 
 | |
| 	i2c_unregister_device(client);
 | |
| 
 | |
| 	kfree(gate_client);
 | |
| 	gate_client = NULL;
 | |
| 	_gate_ic_i2c_client = NULL;
 | |
| 	i2c_unregister_device(client);
 | |
| #ifdef LEDS_BRIGHTNESS_CHANGED
 | |
| 	mtk_leds_unregister_notifier(&leds_init_notifier);
 | |
| #endif
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*****************************************************************************
 | |
|  * Data Structure
 | |
|  *****************************************************************************/
 | |
| 
 | |
| static const struct i2c_device_id _gate_ic_i2c_id[] = {
 | |
| 	{GATE_I2C_ID_NAME, 0},
 | |
| 	{}
 | |
| };
 | |
| 
 | |
| static struct i2c_driver _gate_ic_i2c_driver = {
 | |
| 	.id_table = _gate_ic_i2c_id,
 | |
| 	.probe = _gate_ic_i2c_probe,
 | |
| 	.remove = _gate_ic_i2c_remove,
 | |
| 	.driver = {
 | |
| 		   .owner = THIS_MODULE,
 | |
| 		   .name = GATE_I2C_ID_NAME,
 | |
| 		   .of_match_table = _gate_ic_i2c_of_match,
 | |
| 		   },
 | |
| };
 | |
| 
 | |
| module_i2c_driver(_gate_ic_i2c_driver);
 | |
| 
 | |
| MODULE_AUTHOR("Mediatek Corporation");
 | |
| MODULE_DESCRIPTION("MTK RT4831 I2C Driver");
 | |
| MODULE_LICENSE("GPL");
 | |
| 
 | |
| 
 |