Angel Pons | 11ba353 | 2020-04-05 13:21:58 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 2 | |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 3 | #include <assert.h> |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 4 | #include <boardid.h> |
| 5 | #include <bootmode.h> |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 6 | #include <cbfs.h> |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 7 | #include <console/console.h> |
| 8 | #include <delay.h> |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 9 | #include <device/device.h> |
Hung-Te Lin | 7fc2281 | 2020-07-24 08:49:41 +0800 | [diff] [blame] | 10 | #include <ec/google/chromeec/ec.h> |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 11 | #include <edid.h> |
Patrick Rudolph | 7319288 | 2020-02-19 12:10:51 +0100 | [diff] [blame] | 12 | #include <framebuffer_info.h> |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 13 | #include <gpio.h> |
Kyösti Mälkki | b78e462 | 2022-12-15 22:12:29 +0200 | [diff] [blame] | 14 | #include <identity.h> |
Hung-Te Lin | a01f8bc | 2022-09-06 14:32:05 +0800 | [diff] [blame] | 15 | #include <soc/bl31.h> |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 16 | #include <soc/ddp.h> |
| 17 | #include <soc/dsi.h> |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 18 | #include <soc/mmu_operations.h> |
Jiaxin Yu | 30bc9f4 | 2019-04-23 20:45:50 +0800 | [diff] [blame] | 19 | #include <soc/mtcmos.h> |
Dawei Chien | 2422f8c | 2019-05-30 11:55:10 +0800 | [diff] [blame] | 20 | #include <soc/spm.h> |
Tristan Shieh | 71ae582 | 2018-09-26 14:33:54 +0800 | [diff] [blame] | 21 | #include <soc/usb.h> |
Elyes Haouas | bdd03c2 | 2024-05-27 11:20:07 +0200 | [diff] [blame] | 22 | #include <stdio.h> |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 23 | |
Tristan Shieh | 4d990ab | 2019-02-25 17:14:14 +0800 | [diff] [blame] | 24 | #include "gpio.h" |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 25 | #include "panel.h" |
| 26 | |
Tristan Shieh | 8db79c1 | 2018-09-07 17:26:21 +0800 | [diff] [blame] | 27 | static void configure_emmc(void) |
| 28 | { |
| 29 | const gpio_t emmc_pin[] = { |
| 30 | GPIO(MSDC0_DAT0), GPIO(MSDC0_DAT1), |
| 31 | GPIO(MSDC0_DAT2), GPIO(MSDC0_DAT3), |
| 32 | GPIO(MSDC0_DAT4), GPIO(MSDC0_DAT5), |
| 33 | GPIO(MSDC0_DAT6), GPIO(MSDC0_DAT7), |
| 34 | GPIO(MSDC0_CMD), GPIO(MSDC0_RSTB), |
| 35 | }; |
| 36 | |
| 37 | for (size_t i = 0; i < ARRAY_SIZE(emmc_pin); i++) |
| 38 | gpio_set_pull(emmc_pin[i], GPIO_PULL_ENABLE, GPIO_PULL_UP); |
| 39 | } |
| 40 | |
Tristan Shieh | 71ae582 | 2018-09-26 14:33:54 +0800 | [diff] [blame] | 41 | static void configure_usb(void) |
| 42 | { |
| 43 | setup_usb_host(); |
| 44 | } |
| 45 | |
Jiaxin Yu | 30bc9f4 | 2019-04-23 20:45:50 +0800 | [diff] [blame] | 46 | static void configure_audio(void) |
| 47 | { |
| 48 | /* Audio PWR*/ |
| 49 | mtcmos_audio_power_on(); |
| 50 | |
| 51 | /* SoC I2S */ |
| 52 | gpio_set_mode(GPIO(CAM_RST0), PAD_CAM_RST0_FUNC_I2S2_LRCK); |
| 53 | gpio_set_mode(GPIO(CAM_PDN1), PAD_CAM_PDN1_FUNC_I2S2_BCK); |
| 54 | gpio_set_mode(GPIO(CAM_PDN0), PAD_CAM_PDN0_FUNC_I2S2_MCK); |
| 55 | gpio_set_mode(GPIO(EINT3), PAD_EINT3_FUNC_I2S3_DO); |
| 56 | } |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 57 | |
Hung-Te Lin | 7fc2281 | 2020-07-24 08:49:41 +0800 | [diff] [blame] | 58 | static void configure_ec(void) |
| 59 | { |
| 60 | /* EC may need SKU ID to identify if it is clamshell or convertible. */ |
| 61 | if (CONFIG(BOARD_GOOGLE_JACUZZI_COMMON)) |
| 62 | google_chromeec_set_sku_id(sku_id()); |
| 63 | } |
| 64 | |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 65 | /* Default implementation for boards without panels defined yet. */ |
| 66 | struct panel_description __weak *get_panel_description(int panel_id) |
| 67 | { |
| 68 | printk(BIOS_ERR, "%s: ERROR: No panels defined for board: %s.\n", |
Kyösti Mälkki | b78e462 | 2022-12-15 22:12:29 +0200 | [diff] [blame] | 69 | __func__, mainboard_part_number); |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 70 | return NULL; |
| 71 | } |
| 72 | |
| 73 | /* Set up backlight control pins as output pin and power-off by default */ |
| 74 | static void configure_panel_backlight(void) |
| 75 | { |
| 76 | gpio_output(GPIO(PERIPHERAL_EN13), 0); |
| 77 | gpio_output(GPIO(DISP_PWM), 0); |
| 78 | } |
| 79 | |
| 80 | static void power_on_panel(struct panel_description *panel) |
| 81 | { |
| 82 | if (panel->power_on) { |
| 83 | panel->power_on(); |
| 84 | return; |
| 85 | } |
| 86 | |
| 87 | /* Default power sequence for most panels. */ |
| 88 | gpio_output(GPIO_LCM_RST_1V8, 0); |
| 89 | gpio_output(GPIO_PPVARP_LCD_EN, 1); |
| 90 | gpio_output(GPIO_PPVARN_LCD_EN, 1); |
| 91 | gpio_output(GPIO_PP1800_LCM_EN, 1); |
| 92 | gpio_output(GPIO_PP3300_LCM_EN, 1); |
Tao Xia | f3c0a01 | 2020-11-11 15:32:46 +0800 | [diff] [blame] | 93 | mdelay(15); |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 94 | gpio_output(GPIO_LCM_RST_1V8, 1); |
| 95 | mdelay(6); |
| 96 | } |
| 97 | |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 98 | struct panel_description *get_panel_from_cbfs(struct panel_description *desc) |
| 99 | { |
| 100 | /* The CBFS name will be panel-{MANUFACTURER}-${PANEL_NAME}, |
| 101 | * where MANUFACTURER is 3 characters and PANEL_NAME is usually |
| 102 | * 13 characters. |
| 103 | */ |
| 104 | char cbfs_name[64]; |
| 105 | static union { |
| 106 | u8 raw[4 * 1024]; /* Most panels only need < 2K. */ |
| 107 | struct panel_serializable_data s; |
| 108 | } buffer; |
| 109 | |
| 110 | if (!desc->name) |
| 111 | return NULL; |
| 112 | |
| 113 | snprintf(cbfs_name, sizeof(cbfs_name), "panel-%s", desc->name); |
Julius Werner | 834b3ec | 2020-03-04 16:52:08 -0800 | [diff] [blame] | 114 | if (cbfs_load(cbfs_name, buffer.raw, sizeof(buffer))) |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 115 | desc->s = &buffer.s; |
| 116 | else |
| 117 | printk(BIOS_ERR, "Missing %s in CBFS.\n", cbfs_name); |
| 118 | |
| 119 | return desc->s ? desc : NULL; |
| 120 | } |
| 121 | |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 122 | static struct panel_description *get_active_panel(void) |
| 123 | { |
| 124 | /* TODO(hungte) Create a dedicated panel_id() in board_id.c */ |
xuxinxiong | 3ced528 | 2021-03-26 22:53:35 +0800 | [diff] [blame] | 125 | int panel_id = sku_id() >> 4 & 0x0F; |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 126 | |
| 127 | struct panel_description *panel = get_panel_description(panel_id); |
| 128 | if (!panel) { |
| 129 | printk(BIOS_ERR, "%s: Panel %d is not supported.\n", |
| 130 | __func__, panel_id); |
| 131 | return NULL; |
| 132 | } |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 133 | assert(panel->s); |
| 134 | |
| 135 | const struct edid *edid = &panel->s->edid; |
| 136 | const char *name = edid->ascii_string; |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 137 | if (name[0] == '\0') |
| 138 | name = "unknown name"; |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 139 | printk(BIOS_INFO, "%s: Found ID %d: '%s %s' %dx%d@%dHz\n", __func__, |
| 140 | panel_id, edid->manufacturer_name, name, edid->mode.ha, |
| 141 | edid->mode.va, edid->mode.refresh); |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 142 | return panel; |
| 143 | } |
| 144 | |
| 145 | static bool configure_display(void) |
| 146 | { |
| 147 | struct panel_description *panel = get_active_panel(); |
| 148 | if (!panel) |
| 149 | return false; |
| 150 | |
| 151 | mtcmos_display_power_on(); |
| 152 | mtcmos_protect_display_bus(); |
| 153 | configure_panel_backlight(); |
| 154 | power_on_panel(panel); |
| 155 | |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 156 | struct edid *edid = &panel->s->edid; |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 157 | edid_set_framebuffer_bits_per_pixel(edid, 32, 0); |
| 158 | mtk_ddp_init(); |
| 159 | u32 mipi_dsi_flags = (MIPI_DSI_MODE_VIDEO | |
| 160 | MIPI_DSI_MODE_VIDEO_SYNC_PULSE | |
| 161 | MIPI_DSI_MODE_LPM); |
Paul Ma | 6d500a2 | 2020-05-08 13:16:04 +0800 | [diff] [blame] | 162 | if (CONFIG(DRIVER_ANALOGIX_ANX7625)) |
Jitao Shi | b923931 | 2021-03-04 21:52:24 +0800 | [diff] [blame] | 163 | mipi_dsi_flags |= MIPI_DSI_MODE_EOT_PACKET | |
| 164 | MIPI_DSI_MODE_LINE_END; |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 165 | if (mtk_dsi_init(mipi_dsi_flags, MIPI_DSI_FMT_RGB888, 4, edid, |
Hung-Te Lin | 9ede2ff | 2019-08-15 09:43:55 +0800 | [diff] [blame] | 166 | panel->s->init) < 0) { |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 167 | printk(BIOS_ERR, "%s: Failed in DSI init.\n", __func__); |
| 168 | return false; |
| 169 | } |
Jitao Shi | b32e4d6 | 2020-09-25 10:36:29 +0800 | [diff] [blame] | 170 | |
| 171 | if (panel->post_power_on) |
| 172 | panel->post_power_on(); |
| 173 | |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 174 | mtk_ddp_mode_set(edid); |
Patrick Rudolph | 8b56c8c | 2020-02-19 12:57:00 +0100 | [diff] [blame] | 175 | struct fb_info *info = fb_new_framebuffer_info_from_edid(edid, 0); |
Patrick Rudolph | 7319288 | 2020-02-19 12:10:51 +0100 | [diff] [blame] | 176 | if (info) |
Julius Werner | 1f84b2c | 2021-09-01 16:27:58 -0700 | [diff] [blame] | 177 | fb_set_orientation(info, panel->orientation); |
Patrick Rudolph | 7319288 | 2020-02-19 12:10:51 +0100 | [diff] [blame] | 178 | |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 179 | return true; |
| 180 | } |
| 181 | |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 182 | static void mainboard_init(struct device *dev) |
| 183 | { |
Hung-Te Lin | 2c307a0 | 2019-02-20 14:31:23 +0800 | [diff] [blame] | 184 | if (display_init_required()) { |
| 185 | printk(BIOS_INFO, "%s: Starting display init.\n", __func__); |
| 186 | if (!configure_display()) |
| 187 | printk(BIOS_ERR, "%s: Failed to init display.\n", |
| 188 | __func__); |
| 189 | } else { |
| 190 | printk(BIOS_INFO, "%s: Skipped display init.\n", __func__); |
| 191 | } |
| 192 | |
Tristan Shieh | 8db79c1 | 2018-09-07 17:26:21 +0800 | [diff] [blame] | 193 | configure_emmc(); |
Tristan Shieh | 71ae582 | 2018-09-26 14:33:54 +0800 | [diff] [blame] | 194 | configure_usb(); |
Jiaxin Yu | 30bc9f4 | 2019-04-23 20:45:50 +0800 | [diff] [blame] | 195 | configure_audio(); |
Hung-Te Lin | 7fc2281 | 2020-07-24 08:49:41 +0800 | [diff] [blame] | 196 | configure_ec(); |
Dawei Chien | 2422f8c | 2019-05-30 11:55:10 +0800 | [diff] [blame] | 197 | if (spm_init()) |
| 198 | printk(BIOS_ERR, |
| 199 | "SPM initialization failed, suspend/resume may fail.\n"); |
Tristan Shieh | 4d990ab | 2019-02-25 17:14:14 +0800 | [diff] [blame] | 200 | |
Hung-Te Lin | a01f8bc | 2022-09-06 14:32:05 +0800 | [diff] [blame] | 201 | if (CONFIG(ARM64_USE_ARM_TRUSTED_FIRMWARE)) |
| 202 | register_reset_to_bl31(GPIO_RESET.id, true); |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 203 | } |
| 204 | |
| 205 | static void mainboard_enable(struct device *dev) |
| 206 | { |
| 207 | dev->ops->init = &mainboard_init; |
| 208 | } |
| 209 | |
| 210 | struct chip_operations mainboard_ops = { |
Tristan Shieh | bb684e0 | 2018-06-19 16:07:54 +0800 | [diff] [blame] | 211 | .enable_dev = mainboard_enable, |
| 212 | }; |