/*
 * Copyright (c) 2020 Actions Technology Co., Ltd
 *
 * SPDX-License-Identifier: Apache-2.0
 */

#include <sys/byteorder.h>
#include "panel_gc9c01.h"
#include "panel_device.h"

/*********************
 *      DEFINES
 *********************/

/**********************
 *      TYPEDEFS
 **********************/

/**********************
 *  STATIC PROTOTYPES
 **********************/

/**********************
 *  STATIC VARIABLES
 **********************/

/**********************
 *      MACROS
 **********************/

/**********************
 *  FUNCTIONS
 **********************/
static void _panel_transmit(const struct device *dev, uint32_t cmd,
		const uint8_t *tx_data, size_t tx_count)
{
	struct lcd_panel_data *data = dev->data;

	assert(data->transfering == 0);

	display_controller_write_config(data->lcdc_dev, (0x02 << 24) | (cmd << 8), tx_data, tx_count);
}

static inline void _panel_transmit_cmd(const struct device *dev, uint32_t cmd)
{
	_panel_transmit(dev, cmd, NULL, 0);
}

static inline void _panel_transmit_p1(const struct device *dev, uint32_t cmd, uint8_t tx_data)
{
	_panel_transmit(dev, cmd, &tx_data, 1);
}

static void _panel_exit_sleep(const struct device *dev)
{
	struct lcd_panel_data *data = dev->data;

	_panel_transmit_cmd(dev, DDIC_CMD_SLPOUT);
	k_msleep(120);

	data->in_sleep = 0;
}

static void _panel_init_te(const struct device *dev)
{
	const struct lcd_panel_config *config = dev->config;

	if (config->videomode.flags & (DISPLAY_FLAGS_TE_HIGH | DISPLAY_FLAGS_TE_LOW)) {
		uint8_t tmp[2];

		tmp[0] = 0x10;
		_panel_transmit(dev, 0x84, tmp, 1);

		tmp[0] = 0x02; /* pulse width: 3 line time */
		tmp[1] = (config->videomode.flags & DISPLAY_FLAGS_TE_LOW) ? 1 : 0;
		_panel_transmit(dev, DDIC_CMD_TECTL, tmp, 2);

		sys_put_be16(CONFIG_PANEL_TE_SCANLINE, tmp);
		_panel_transmit(dev, DDIC_CMD_STESL, tmp, 2);

		tmp[0] = 0; /* only output vsync */
		_panel_transmit(dev, DDIC_CMD_TEON, tmp, 1);
	} else {
		_panel_transmit(dev, DDIC_CMD_TEOFF, NULL, 0);
	}
}

static int _panel_init(const struct device *dev)
{
	_panel_exit_sleep(dev);

	// internal reg enable
	_panel_transmit(dev, DDIC_CMD_INTERREG_EN1, NULL, 0);
	_panel_transmit(dev, DDIC_CMD_INTERREG_EN2, NULL, 0);

	//reg_en for 70\74
	_panel_transmit_p1(dev, 0x80, 0x11);
	//reg_en for 7C\7D\7E
	_panel_transmit_p1(dev, 0x81, 0x70);
	//reg_en for 90\93
	_panel_transmit_p1(dev, 0x82, 0x09);
	//reg_en for 98\99
	_panel_transmit_p1(dev, 0x83, 0x03);
	//reg_en for B5
	_panel_transmit_p1(dev, 0x84, 0x20);
	//reg_en for B9\BE
	_panel_transmit_p1(dev, 0x85, 0x42);
	//reg_en for C2~7
	_panel_transmit_p1(dev, 0x86, 0xfc);
	//reg_en for C8\CB
	_panel_transmit_p1(dev, 0x87, 0x09);
	//reg_en for EC
	_panel_transmit_p1(dev, 0x89, 0x10);
	//reg_en for F0~3\F6
	_panel_transmit_p1(dev, 0x8A, 0x4f);
	//reg_en for 60\63\64\66
	_panel_transmit_p1(dev, 0x8C, 0x59);
	//reg_en for 68\6C\6E
	_panel_transmit_p1(dev, 0x8D, 0x51);
	//reg_en for A1~3\A5\A7
	_panel_transmit_p1(dev, 0x8E, 0xae);
	//reg_en for AC~F\A8\A9
	_panel_transmit_p1(dev, 0x8F, 0xf3);

	_panel_transmit_p1(dev, DDIC_CMD_MADCTL, 0x00);
	// 565 frame
	_panel_transmit_p1(dev, DDIC_CMD_COLMOD, 0x05);
	// 2COL
	_panel_transmit_p1(dev, DDIC_CMD_INVERSION, 0x77);

	// rtn 60Hz
	const uint8_t rtn_data[] = { 0x01, 0x80, 0x00, 0x00, 0x00, 0x00 };
	_panel_transmit(dev, 0x74, rtn_data, sizeof(rtn_data));

	// bvdd 3x
	_panel_transmit_p1(dev, 0x98, 0x3E);
	// bvee -2x
	_panel_transmit_p1(dev, 0x99, 0x3E);

	// VBP
	//_panel_transmit_p1(p_gc9c01, 0xC3, 0x2A);
	_panel_transmit_p1(dev, 0xC3, 0x3A + 4);
	// VBN
	//_panel_transmit_p1(p_gc9c01, 0xC4, 0x18);
	_panel_transmit_p1(dev, 0xC4, 0x16 + 4);

	const uint8_t data_0xA1A2[] = { 0x01, 0x04 };
	_panel_transmit(dev, 0xA1, data_0xA1A2, sizeof(data_0xA1A2));
	_panel_transmit(dev, 0xA2, data_0xA1A2, sizeof(data_0xA1A2));

	// IREF
	_panel_transmit_p1(dev, 0xA9, 0x1C);

	const uint8_t data_0xA5[] = { 0x11, 0x09 }; //VDDMA,VDDML
	_panel_transmit(dev, 0xA5, data_0xA5, sizeof(data_0xA5));

	// RTERM
	_panel_transmit_p1(dev, 0xB9, 0x8A);
	//VBG_BUF, DVDD
	_panel_transmit_p1(dev, 0xA8, 0x5E);
	_panel_transmit_p1(dev, 0xA7, 0x40);
	//VDDSOU ,VDDGM
	_panel_transmit_p1(dev, 0xAF, 0x73);
	//VREE,VRDD
	_panel_transmit_p1(dev, 0xAE, 0x44);
	//VRGL ,VDDSF
	_panel_transmit_p1(dev, 0xAD, 0x38);
	//VRGL ,VDDSF (adjust refresh rate about 60.1~60.6 fps)
	_panel_transmit_p1(dev, 0xA3, 0x67);
	//VREG_VREF
	_panel_transmit_p1(dev, 0xC2, 0x02);
	//VREG1A
	_panel_transmit_p1(dev, 0xC5, 0x11);
	//VREG1B
	_panel_transmit_p1(dev, 0xC6, 0x0E);
	//VREG2A
	_panel_transmit_p1(dev, 0xC7, 0x13);
	//VREG2B
	_panel_transmit_p1(dev, 0xC8, 0x0D);

	//bvdd ref_ad
	_panel_transmit_p1(dev, 0xCB, 0x02);

	const uint8_t data_0x7C[] = { 0xB6, 0x26 };
	_panel_transmit(dev, 0x7C, data_0x7C, sizeof(data_0x7C));

	//VGLO
	_panel_transmit_p1(dev, 0xAC, 0x24);
	//EPF=2
	_panel_transmit_p1(dev, 0xF6, 0x80);

	//gip start
	const uint8_t data_0xB5[] = { 0x09, 0x09 }; //VFP VBP
	_panel_transmit(dev, 0xB5, data_0xB5, sizeof(data_0xB5));

	const uint8_t data_0x60[] = { 0x38, 0x0B, 0x5B, 0x56 }; //STV1&2
	_panel_transmit(dev, 0x60, data_0x60, sizeof(data_0x60));

	const uint8_t data_0x63[] = { 0x3A, 0xE0, 0x5B, 0x56 }; //STV3&4
	_panel_transmit(dev, 0x63, data_0x63, sizeof(data_0x63));

	const uint8_t data_0x64[] = { 0x38, 0x0D, 0x72, 0xDD, 0x5B, 0x56 }; //CLK_group1
	_panel_transmit(dev, 0x64, data_0x64, sizeof(data_0x64));

	const uint8_t data_0x66[] = { 0x38, 0x11, 0x72, 0xE1, 0x5B, 0x56 }; //CLK_group1
	_panel_transmit(dev, 0x66, data_0x66, sizeof(data_0x66));

	const uint8_t data_0x68[] = { 0x3B, 0x08, 0x08, 0x00, 0x08, 0x29, 0x5B }; //FLC&FLV 1~2
	_panel_transmit(dev, 0x68, data_0x68, sizeof(data_0x68));

	const uint8_t data_0x6E[] = {
		0x00, 0x00, 0x00, 0x07, 0x01, 0x13, 0x11, 0x0B, 0x09, 0x16, 0x15, 0x1D,
		0x1E, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1E, 0x1D, 0x15, 0x16, 0x0A,
		0x0C, 0x12, 0x14, 0x02, 0x08, 0x00, 0x00, 0x00,
	};
	_panel_transmit(dev, 0x6E, data_0x6E, sizeof(data_0x6E));

	//gip end

	//SOU_BIAS_FIX
	_panel_transmit_p1(dev, 0xBE, 0x11);

	const uint8_t data_0x6C[] = { 0xCC, 0x0C, 0xCC, 0x84, 0xCC, 0x04, 0x50 }; //precharge GATE
	_panel_transmit(dev, 0x6C, data_0x6C, sizeof(data_0x6C));

	_panel_transmit_p1(dev, 0x7D, 0x72);
	_panel_transmit_p1(dev, 0x7E, 0x38);

	const uint8_t data_0x70[] = { 0x02, 0x03, 0x09, 0x05, 0x0C, 0x06, 0x09, 0x05, 0x0C, 0x06 };
	_panel_transmit(dev, 0x70, data_0x70, sizeof(data_0x70));

	const uint8_t data_0x90[] = { 0x06, 0x06, 0x05, 0x06 };
	_panel_transmit(dev, 0x90, data_0x90, sizeof(data_0x90));

	const uint8_t data_0x93[] = { 0x45, 0xFF, 0x00 };
	_panel_transmit(dev, 0x93, data_0x93, sizeof(data_0x93));

	const uint8_t data_0xF0[] = { 0x46, 0x0B, 0x0F, 0x0A, 0x10, 0x3F };
	_panel_transmit(dev, DDIC_CMD_GAMSET1, data_0xF0, sizeof(data_0xF0));

	const uint8_t data_0xF1[] = { 0x52, 0x9A, 0x4F, 0x36, 0x37, 0xFF };
	_panel_transmit(dev, DDIC_CMD_GAMSET2, data_0xF1, sizeof(data_0xF1));

	const uint8_t data_0xF2[] = { 0x46, 0x0B, 0x0F, 0x0A, 0x10, 0x3F };
	_panel_transmit(dev, DDIC_CMD_GAMSET3, data_0xF2, sizeof(data_0xF2));

	const uint8_t data_0xF3[] = { 0x52, 0x9A, 0x4F, 0x36, 0x37, 0xFF };
	_panel_transmit(dev, DDIC_CMD_GAMSET4, data_0xF3, sizeof(data_0xF3));

	_panel_init_te(dev);
	return 0;
}

static int _panel_set_mem_area(const struct device *dev, uint16_t x, uint16_t y, uint16_t w,
				uint16_t h)
{
	uint16_t cmd_data[2];

	x += CONFIG_PANEL_MEM_OFFSET_X;
	y += CONFIG_PANEL_MEM_OFFSET_Y;

	cmd_data[0] = sys_cpu_to_be16(x);
	cmd_data[1] = sys_cpu_to_be16(x + w - 1);
	_panel_transmit(dev, DDIC_CMD_CASET, (uint8_t *)&cmd_data[0], 4);

	cmd_data[0] = sys_cpu_to_be16(y);
	cmd_data[1] = sys_cpu_to_be16(y + h - 1);
	_panel_transmit(dev, DDIC_CMD_RASET, (uint8_t *)&cmd_data[0], 4);

	return 0;
}

static int _panel_blanking_on(const struct device *dev)
{
	_panel_transmit_cmd(dev, DDIC_CMD_DISPOFF);
	_panel_transmit_cmd(dev, DDIC_CMD_SLPIN);
	return 0;
}

static int _panel_blanking_off(const struct device *dev)
{
	struct lcd_panel_data *data = dev->data;

	if (data->in_sleep)
		_panel_exit_sleep(dev);

	_panel_transmit_cmd(dev, DDIC_CMD_DISPON);
	k_msleep(120);

	return 0;
}

static const struct lcd_panel_ops lcd_panel_ops = {
	.init = _panel_init,
	.blanking_on = _panel_blanking_on,
	.blanking_off = _panel_blanking_off,
	.write_prepare = _panel_set_mem_area,
};

const struct lcd_panel_config lcd_panel_gc9c01_config = {
	.videoport = PANEL_VIDEO_PORT_INITIALIZER,
	.videomode = PANEL_VIDEO_MODE_INITIALIZER,
	.ops = &lcd_panel_ops,
	.cmd_ramwr = (0x32 << 24 | DDIC_CMD_RAMWR << 8),
	.cmd_ramwc = (0x32 << 24 | DDIC_CMD_RAMWRC << 8),
	.tw_reset = 10,
	.ts_reset = 120,
};
