1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2016, Zodiac Inflight Innovations
- * Copyright (c) 2007-2016, Synaptics Incorporated
- * Copyright (C) 2012 Alexandra Chin <[email protected]>
- * Copyright (C) 2012 Scott Lin <[email protected]>
- */
- #include <linux/bitops.h>
- #include <linux/kernel.h>
- #include <linux/rmi.h>
- #include <linux/firmware.h>
- #include <linux/delay.h>
- #include <linux/slab.h>
- #include <linux/jiffies.h>
- #include <asm/unaligned.h>
- #include "rmi_driver.h"
- #include "rmi_f34.h"
- static int rmi_f34v7_read_flash_status(struct f34_data *f34)
- {
- u8 status;
- u8 command;
- int ret;
- ret = rmi_read_block(f34->fn->rmi_dev,
- f34->fn->fd.data_base_addr + V7_FLASH_STATUS_OFFSET,
- &status,
- sizeof(status));
- if (ret < 0) {
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Error %d reading flash status\n", __func__, ret);
- return ret;
- }
- f34->v7.in_bl_mode = status >> 7;
- f34->v7.flash_status = status & 0x1f;
- if (f34->v7.flash_status != 0x00) {
- dev_err(&f34->fn->dev, "%s: status=%d, command=0x%02x\n",
- __func__, f34->v7.flash_status, f34->v7.command);
- }
- ret = rmi_read_block(f34->fn->rmi_dev,
- f34->fn->fd.data_base_addr + V7_COMMAND_OFFSET,
- &command,
- sizeof(command));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read flash command\n",
- __func__);
- return ret;
- }
- f34->v7.command = command;
- return 0;
- }
- static int rmi_f34v7_wait_for_idle(struct f34_data *f34, int timeout_ms)
- {
- unsigned long timeout;
- timeout = msecs_to_jiffies(timeout_ms);
- if (!wait_for_completion_timeout(&f34->v7.cmd_done, timeout)) {
- dev_warn(&f34->fn->dev, "%s: Timed out waiting for idle status\n",
- __func__);
- return -ETIMEDOUT;
- }
- return 0;
- }
- static int rmi_f34v7_check_command_status(struct f34_data *f34, int timeout_ms)
- {
- int ret;
- ret = rmi_f34v7_wait_for_idle(f34, timeout_ms);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_read_flash_status(f34);
- if (ret < 0)
- return ret;
- if (f34->v7.flash_status != 0x00)
- return -EIO;
- return 0;
- }
- static int rmi_f34v7_write_command_single_transaction(struct f34_data *f34,
- u8 cmd)
- {
- int ret;
- u8 base;
- struct f34v7_data_1_5 data_1_5;
- base = f34->fn->fd.data_base_addr;
- memset(&data_1_5, 0, sizeof(data_1_5));
- switch (cmd) {
- case v7_CMD_ERASE_ALL:
- data_1_5.partition_id = CORE_CODE_PARTITION;
- data_1_5.command = CMD_V7_ERASE_AP;
- break;
- case v7_CMD_ERASE_UI_FIRMWARE:
- data_1_5.partition_id = CORE_CODE_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ERASE_BL_CONFIG:
- data_1_5.partition_id = GLOBAL_PARAMETERS_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ERASE_UI_CONFIG:
- data_1_5.partition_id = CORE_CONFIG_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ERASE_DISP_CONFIG:
- data_1_5.partition_id = DISPLAY_CONFIG_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ERASE_FLASH_CONFIG:
- data_1_5.partition_id = FLASH_CONFIG_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ERASE_GUEST_CODE:
- data_1_5.partition_id = GUEST_CODE_PARTITION;
- data_1_5.command = CMD_V7_ERASE;
- break;
- case v7_CMD_ENABLE_FLASH_PROG:
- data_1_5.partition_id = BOOTLOADER_PARTITION;
- data_1_5.command = CMD_V7_ENTER_BL;
- break;
- }
- data_1_5.payload[0] = f34->bootloader_id[0];
- data_1_5.payload[1] = f34->bootloader_id[1];
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_PARTITION_ID_OFFSET,
- &data_1_5, sizeof(data_1_5));
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Failed to write single transaction command\n",
- __func__);
- return ret;
- }
- return 0;
- }
- static int rmi_f34v7_write_command(struct f34_data *f34, u8 cmd)
- {
- int ret;
- u8 base;
- u8 command;
- base = f34->fn->fd.data_base_addr;
- switch (cmd) {
- case v7_CMD_WRITE_FW:
- case v7_CMD_WRITE_CONFIG:
- case v7_CMD_WRITE_GUEST_CODE:
- command = CMD_V7_WRITE;
- break;
- case v7_CMD_READ_CONFIG:
- command = CMD_V7_READ;
- break;
- case v7_CMD_ERASE_ALL:
- command = CMD_V7_ERASE_AP;
- break;
- case v7_CMD_ERASE_UI_FIRMWARE:
- case v7_CMD_ERASE_BL_CONFIG:
- case v7_CMD_ERASE_UI_CONFIG:
- case v7_CMD_ERASE_DISP_CONFIG:
- case v7_CMD_ERASE_FLASH_CONFIG:
- case v7_CMD_ERASE_GUEST_CODE:
- command = CMD_V7_ERASE;
- break;
- case v7_CMD_ENABLE_FLASH_PROG:
- command = CMD_V7_ENTER_BL;
- break;
- default:
- dev_err(&f34->fn->dev, "%s: Invalid command 0x%02x\n",
- __func__, cmd);
- return -EINVAL;
- }
- f34->v7.command = command;
- switch (cmd) {
- case v7_CMD_ERASE_ALL:
- case v7_CMD_ERASE_UI_FIRMWARE:
- case v7_CMD_ERASE_BL_CONFIG:
- case v7_CMD_ERASE_UI_CONFIG:
- case v7_CMD_ERASE_DISP_CONFIG:
- case v7_CMD_ERASE_FLASH_CONFIG:
- case v7_CMD_ERASE_GUEST_CODE:
- case v7_CMD_ENABLE_FLASH_PROG:
- ret = rmi_f34v7_write_command_single_transaction(f34, cmd);
- if (ret < 0)
- return ret;
- else
- return 0;
- default:
- break;
- }
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: writing cmd %02X\n",
- __func__, command);
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_COMMAND_OFFSET,
- &command, sizeof(command));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write flash command\n",
- __func__);
- return ret;
- }
- return 0;
- }
- static int rmi_f34v7_write_partition_id(struct f34_data *f34, u8 cmd)
- {
- int ret;
- u8 base;
- u8 partition;
- base = f34->fn->fd.data_base_addr;
- switch (cmd) {
- case v7_CMD_WRITE_FW:
- partition = CORE_CODE_PARTITION;
- break;
- case v7_CMD_WRITE_CONFIG:
- case v7_CMD_READ_CONFIG:
- if (f34->v7.config_area == v7_UI_CONFIG_AREA)
- partition = CORE_CONFIG_PARTITION;
- else if (f34->v7.config_area == v7_DP_CONFIG_AREA)
- partition = DISPLAY_CONFIG_PARTITION;
- else if (f34->v7.config_area == v7_PM_CONFIG_AREA)
- partition = GUEST_SERIALIZATION_PARTITION;
- else if (f34->v7.config_area == v7_BL_CONFIG_AREA)
- partition = GLOBAL_PARAMETERS_PARTITION;
- else if (f34->v7.config_area == v7_FLASH_CONFIG_AREA)
- partition = FLASH_CONFIG_PARTITION;
- break;
- case v7_CMD_WRITE_GUEST_CODE:
- partition = GUEST_CODE_PARTITION;
- break;
- case v7_CMD_ERASE_ALL:
- partition = CORE_CODE_PARTITION;
- break;
- case v7_CMD_ERASE_BL_CONFIG:
- partition = GLOBAL_PARAMETERS_PARTITION;
- break;
- case v7_CMD_ERASE_UI_CONFIG:
- partition = CORE_CONFIG_PARTITION;
- break;
- case v7_CMD_ERASE_DISP_CONFIG:
- partition = DISPLAY_CONFIG_PARTITION;
- break;
- case v7_CMD_ERASE_FLASH_CONFIG:
- partition = FLASH_CONFIG_PARTITION;
- break;
- case v7_CMD_ERASE_GUEST_CODE:
- partition = GUEST_CODE_PARTITION;
- break;
- case v7_CMD_ENABLE_FLASH_PROG:
- partition = BOOTLOADER_PARTITION;
- break;
- default:
- dev_err(&f34->fn->dev, "%s: Invalid command 0x%02x\n",
- __func__, cmd);
- return -EINVAL;
- }
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_PARTITION_ID_OFFSET,
- &partition, sizeof(partition));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write partition ID\n",
- __func__);
- return ret;
- }
- return 0;
- }
- static int rmi_f34v7_read_partition_table(struct f34_data *f34)
- {
- int ret;
- unsigned long timeout;
- u8 base;
- __le16 length;
- u16 block_number = 0;
- base = f34->fn->fd.data_base_addr;
- f34->v7.config_area = v7_FLASH_CONFIG_AREA;
- ret = rmi_f34v7_write_partition_id(f34, v7_CMD_READ_CONFIG);
- if (ret < 0)
- return ret;
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_BLOCK_NUMBER_OFFSET,
- &block_number, sizeof(block_number));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
- __func__);
- return ret;
- }
- put_unaligned_le16(f34->v7.flash_config_length, &length);
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_TRANSFER_LENGTH_OFFSET,
- &length, sizeof(length));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write transfer length\n",
- __func__);
- return ret;
- }
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_write_command(f34, v7_CMD_READ_CONFIG);
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write command\n",
- __func__);
- return ret;
- }
- /*
- * rmi_f34v7_check_command_status() can't be used here, as this
- * function is called before IRQs are available
- */
- timeout = msecs_to_jiffies(F34_WRITE_WAIT_MS);
- while (time_before(jiffies, timeout)) {
- usleep_range(5000, 6000);
- rmi_f34v7_read_flash_status(f34);
- if (f34->v7.command == v7_CMD_IDLE &&
- f34->v7.flash_status == 0x00) {
- break;
- }
- }
- ret = rmi_read_block(f34->fn->rmi_dev,
- base + V7_PAYLOAD_OFFSET,
- f34->v7.read_config_buf,
- f34->v7.partition_table_bytes);
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read block data\n",
- __func__);
- return ret;
- }
- return 0;
- }
- static void rmi_f34v7_parse_partition_table(struct f34_data *f34,
- const void *partition_table,
- struct block_count *blkcount,
- struct physical_address *phyaddr)
- {
- int i;
- int index;
- u16 partition_length;
- u16 physical_address;
- const struct partition_table *ptable;
- for (i = 0; i < f34->v7.partitions; i++) {
- index = i * 8 + 2;
- ptable = partition_table + index;
- partition_length = le16_to_cpu(ptable->partition_length);
- physical_address = le16_to_cpu(ptable->start_physical_address);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Partition entry %d: %*ph\n",
- __func__, i, sizeof(struct partition_table), ptable);
- switch (ptable->partition_id & 0x1f) {
- case CORE_CODE_PARTITION:
- blkcount->ui_firmware = partition_length;
- phyaddr->ui_firmware = physical_address;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Core code block count: %d\n",
- __func__, blkcount->ui_firmware);
- break;
- case CORE_CONFIG_PARTITION:
- blkcount->ui_config = partition_length;
- phyaddr->ui_config = physical_address;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Core config block count: %d\n",
- __func__, blkcount->ui_config);
- break;
- case DISPLAY_CONFIG_PARTITION:
- blkcount->dp_config = partition_length;
- phyaddr->dp_config = physical_address;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Display config block count: %d\n",
- __func__, blkcount->dp_config);
- break;
- case FLASH_CONFIG_PARTITION:
- blkcount->fl_config = partition_length;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Flash config block count: %d\n",
- __func__, blkcount->fl_config);
- break;
- case GUEST_CODE_PARTITION:
- blkcount->guest_code = partition_length;
- phyaddr->guest_code = physical_address;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Guest code block count: %d\n",
- __func__, blkcount->guest_code);
- break;
- case GUEST_SERIALIZATION_PARTITION:
- blkcount->pm_config = partition_length;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Guest serialization block count: %d\n",
- __func__, blkcount->pm_config);
- break;
- case GLOBAL_PARAMETERS_PARTITION:
- blkcount->bl_config = partition_length;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Global parameters block count: %d\n",
- __func__, blkcount->bl_config);
- break;
- case DEVICE_CONFIG_PARTITION:
- blkcount->lockdown = partition_length;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: Device config block count: %d\n",
- __func__, blkcount->lockdown);
- break;
- }
- }
- }
- static int rmi_f34v7_read_queries_bl_version(struct f34_data *f34)
- {
- int ret;
- u8 base;
- int offset;
- u8 query_0;
- struct f34v7_query_1_7 query_1_7;
- base = f34->fn->fd.query_base_addr;
- ret = rmi_read_block(f34->fn->rmi_dev,
- base,
- &query_0,
- sizeof(query_0));
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Failed to read query 0\n", __func__);
- return ret;
- }
- offset = (query_0 & 0x7) + 1;
- ret = rmi_read_block(f34->fn->rmi_dev,
- base + offset,
- &query_1_7,
- sizeof(query_1_7));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read queries 1 to 7\n",
- __func__);
- return ret;
- }
- f34->bootloader_id[0] = query_1_7.bl_minor_revision;
- f34->bootloader_id[1] = query_1_7.bl_major_revision;
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "Bootloader V%d.%d\n",
- f34->bootloader_id[1], f34->bootloader_id[0]);
- return 0;
- }
- static int rmi_f34v7_read_queries(struct f34_data *f34)
- {
- int ret;
- int i;
- u8 base;
- int offset;
- u8 *ptable;
- u8 query_0;
- struct f34v7_query_1_7 query_1_7;
- base = f34->fn->fd.query_base_addr;
- ret = rmi_read_block(f34->fn->rmi_dev,
- base,
- &query_0,
- sizeof(query_0));
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Failed to read query 0\n", __func__);
- return ret;
- }
- offset = (query_0 & 0x07) + 1;
- ret = rmi_read_block(f34->fn->rmi_dev,
- base + offset,
- &query_1_7,
- sizeof(query_1_7));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read queries 1 to 7\n",
- __func__);
- return ret;
- }
- f34->bootloader_id[0] = query_1_7.bl_minor_revision;
- f34->bootloader_id[1] = query_1_7.bl_major_revision;
- f34->v7.block_size = le16_to_cpu(query_1_7.block_size);
- f34->v7.flash_config_length =
- le16_to_cpu(query_1_7.flash_config_length);
- f34->v7.payload_length = le16_to_cpu(query_1_7.payload_length);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: f34->v7.block_size = %d\n",
- __func__, f34->v7.block_size);
- f34->v7.has_display_cfg = query_1_7.partition_support[1] & HAS_DISP_CFG;
- f34->v7.has_guest_code =
- query_1_7.partition_support[1] & HAS_GUEST_CODE;
- if (query_0 & HAS_CONFIG_ID) {
- u8 f34_ctrl[CONFIG_ID_SIZE];
- ret = rmi_read_block(f34->fn->rmi_dev,
- f34->fn->fd.control_base_addr,
- f34_ctrl,
- sizeof(f34_ctrl));
- if (ret)
- return ret;
- /* Eat leading zeros */
- for (i = 0; i < sizeof(f34_ctrl) - 1 && !f34_ctrl[i]; i++)
- /* Empty */;
- snprintf(f34->configuration_id, sizeof(f34->configuration_id),
- "%*phN", (int)sizeof(f34_ctrl) - i, f34_ctrl + i);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "Configuration ID: %s\n",
- f34->configuration_id);
- }
- f34->v7.partitions = 0;
- for (i = 0; i < sizeof(query_1_7.partition_support); i++)
- f34->v7.partitions += hweight8(query_1_7.partition_support[i]);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: Supported partitions: %*ph\n",
- __func__, sizeof(query_1_7.partition_support),
- query_1_7.partition_support);
- f34->v7.partition_table_bytes = f34->v7.partitions * 8 + 2;
- f34->v7.read_config_buf = devm_kzalloc(&f34->fn->dev,
- f34->v7.partition_table_bytes,
- GFP_KERNEL);
- if (!f34->v7.read_config_buf) {
- f34->v7.read_config_buf_size = 0;
- return -ENOMEM;
- }
- f34->v7.read_config_buf_size = f34->v7.partition_table_bytes;
- ptable = f34->v7.read_config_buf;
- ret = rmi_f34v7_read_partition_table(f34);
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read partition table\n",
- __func__);
- return ret;
- }
- rmi_f34v7_parse_partition_table(f34, ptable,
- &f34->v7.blkcount, &f34->v7.phyaddr);
- return 0;
- }
- static int rmi_f34v7_check_bl_config_size(struct f34_data *f34)
- {
- u16 block_count;
- block_count = f34->v7.img.bl_config.size / f34->v7.block_size;
- f34->update_size += block_count;
- if (block_count != f34->v7.blkcount.bl_config) {
- dev_err(&f34->fn->dev, "Bootloader config size mismatch\n");
- return -EINVAL;
- }
- return 0;
- }
- static int rmi_f34v7_erase_all(struct f34_data *f34)
- {
- int ret;
- dev_info(&f34->fn->dev, "Erasing firmware...\n");
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_write_command(f34, v7_CMD_ERASE_ALL);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_check_command_status(f34, F34_ERASE_WAIT_MS);
- if (ret < 0)
- return ret;
- return 0;
- }
- static int rmi_f34v7_read_blocks(struct f34_data *f34,
- u16 block_cnt, u8 command)
- {
- int ret;
- u8 base;
- __le16 length;
- u16 transfer;
- u16 max_transfer;
- u16 remaining = block_cnt;
- u16 block_number = 0;
- u16 index = 0;
- base = f34->fn->fd.data_base_addr;
- ret = rmi_f34v7_write_partition_id(f34, command);
- if (ret < 0)
- return ret;
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_BLOCK_NUMBER_OFFSET,
- &block_number, sizeof(block_number));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
- __func__);
- return ret;
- }
- max_transfer = min(f34->v7.payload_length,
- (u16)(PAGE_SIZE / f34->v7.block_size));
- do {
- transfer = min(remaining, max_transfer);
- put_unaligned_le16(transfer, &length);
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_TRANSFER_LENGTH_OFFSET,
- &length, sizeof(length));
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Write transfer length fail (%d remaining)\n",
- __func__, remaining);
- return ret;
- }
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_write_command(f34, command);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
- if (ret < 0)
- return ret;
- ret = rmi_read_block(f34->fn->rmi_dev,
- base + V7_PAYLOAD_OFFSET,
- &f34->v7.read_config_buf[index],
- transfer * f34->v7.block_size);
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Read block failed (%d blks remaining)\n",
- __func__, remaining);
- return ret;
- }
- index += (transfer * f34->v7.block_size);
- remaining -= transfer;
- } while (remaining);
- return 0;
- }
- static int rmi_f34v7_write_f34v7_blocks(struct f34_data *f34,
- const void *block_ptr, u16 block_cnt,
- u8 command)
- {
- int ret;
- u8 base;
- __le16 length;
- u16 transfer;
- u16 max_transfer;
- u16 remaining = block_cnt;
- u16 block_number = 0;
- base = f34->fn->fd.data_base_addr;
- ret = rmi_f34v7_write_partition_id(f34, command);
- if (ret < 0)
- return ret;
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_BLOCK_NUMBER_OFFSET,
- &block_number, sizeof(block_number));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to write block number\n",
- __func__);
- return ret;
- }
- if (f34->v7.payload_length > (PAGE_SIZE / f34->v7.block_size))
- max_transfer = PAGE_SIZE / f34->v7.block_size;
- else
- max_transfer = f34->v7.payload_length;
- do {
- transfer = min(remaining, max_transfer);
- put_unaligned_le16(transfer, &length);
- init_completion(&f34->v7.cmd_done);
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_TRANSFER_LENGTH_OFFSET,
- &length, sizeof(length));
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Write transfer length fail (%d remaining)\n",
- __func__, remaining);
- return ret;
- }
- ret = rmi_f34v7_write_command(f34, command);
- if (ret < 0)
- return ret;
- ret = rmi_write_block(f34->fn->rmi_dev,
- base + V7_PAYLOAD_OFFSET,
- block_ptr, transfer * f34->v7.block_size);
- if (ret < 0) {
- dev_err(&f34->fn->dev,
- "%s: Failed writing data (%d blks remaining)\n",
- __func__, remaining);
- return ret;
- }
- ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
- if (ret < 0)
- return ret;
- block_ptr += (transfer * f34->v7.block_size);
- remaining -= transfer;
- f34->update_progress += transfer;
- f34->update_status = (f34->update_progress * 100) /
- f34->update_size;
- } while (remaining);
- return 0;
- }
- static int rmi_f34v7_write_config(struct f34_data *f34)
- {
- return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.config_data,
- f34->v7.config_block_count,
- v7_CMD_WRITE_CONFIG);
- }
- static int rmi_f34v7_write_ui_config(struct f34_data *f34)
- {
- f34->v7.config_area = v7_UI_CONFIG_AREA;
- f34->v7.config_data = f34->v7.img.ui_config.data;
- f34->v7.config_size = f34->v7.img.ui_config.size;
- f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
- return rmi_f34v7_write_config(f34);
- }
- static int rmi_f34v7_write_dp_config(struct f34_data *f34)
- {
- f34->v7.config_area = v7_DP_CONFIG_AREA;
- f34->v7.config_data = f34->v7.img.dp_config.data;
- f34->v7.config_size = f34->v7.img.dp_config.size;
- f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
- return rmi_f34v7_write_config(f34);
- }
- static int rmi_f34v7_write_guest_code(struct f34_data *f34)
- {
- return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.img.guest_code.data,
- f34->v7.img.guest_code.size /
- f34->v7.block_size,
- v7_CMD_WRITE_GUEST_CODE);
- }
- static int rmi_f34v7_write_flash_config(struct f34_data *f34)
- {
- int ret;
- f34->v7.config_area = v7_FLASH_CONFIG_AREA;
- f34->v7.config_data = f34->v7.img.fl_config.data;
- f34->v7.config_size = f34->v7.img.fl_config.size;
- f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
- if (f34->v7.config_block_count != f34->v7.blkcount.fl_config) {
- dev_err(&f34->fn->dev, "%s: Flash config size mismatch\n",
- __func__);
- return -EINVAL;
- }
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_write_config(f34);
- if (ret < 0)
- return ret;
- return 0;
- }
- static int rmi_f34v7_write_partition_table(struct f34_data *f34)
- {
- u16 block_count;
- int ret;
- block_count = f34->v7.blkcount.bl_config;
- f34->v7.config_area = v7_BL_CONFIG_AREA;
- f34->v7.config_size = f34->v7.block_size * block_count;
- devm_kfree(&f34->fn->dev, f34->v7.read_config_buf);
- f34->v7.read_config_buf = devm_kzalloc(&f34->fn->dev,
- f34->v7.config_size, GFP_KERNEL);
- if (!f34->v7.read_config_buf) {
- f34->v7.read_config_buf_size = 0;
- return -ENOMEM;
- }
- f34->v7.read_config_buf_size = f34->v7.config_size;
- ret = rmi_f34v7_read_blocks(f34, block_count, v7_CMD_READ_CONFIG);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_write_flash_config(f34);
- if (ret < 0)
- return ret;
- f34->v7.config_area = v7_BL_CONFIG_AREA;
- f34->v7.config_data = f34->v7.read_config_buf;
- f34->v7.config_size = f34->v7.img.bl_config.size;
- f34->v7.config_block_count = f34->v7.config_size / f34->v7.block_size;
- ret = rmi_f34v7_write_config(f34);
- if (ret < 0)
- return ret;
- return 0;
- }
- static int rmi_f34v7_write_firmware(struct f34_data *f34)
- {
- u16 blk_count;
- blk_count = f34->v7.img.ui_firmware.size / f34->v7.block_size;
- return rmi_f34v7_write_f34v7_blocks(f34, f34->v7.img.ui_firmware.data,
- blk_count, v7_CMD_WRITE_FW);
- }
- static void rmi_f34v7_parse_img_header_10_bl_container(struct f34_data *f34,
- const void *image)
- {
- int i;
- int num_of_containers;
- unsigned int addr;
- unsigned int container_id;
- unsigned int length;
- const void *content;
- const struct container_descriptor *descriptor;
- num_of_containers = f34->v7.img.bootloader.size / 4 - 1;
- for (i = 1; i <= num_of_containers; i++) {
- addr = get_unaligned_le32(f34->v7.img.bootloader.data + i * 4);
- descriptor = image + addr;
- container_id = le16_to_cpu(descriptor->container_id);
- content = image + le32_to_cpu(descriptor->content_address);
- length = le32_to_cpu(descriptor->content_length);
- switch (container_id) {
- case BL_CONFIG_CONTAINER:
- case GLOBAL_PARAMETERS_CONTAINER:
- f34->v7.img.bl_config.data = content;
- f34->v7.img.bl_config.size = length;
- break;
- case BL_LOCKDOWN_INFO_CONTAINER:
- case DEVICE_CONFIG_CONTAINER:
- f34->v7.img.lockdown.data = content;
- f34->v7.img.lockdown.size = length;
- break;
- default:
- break;
- }
- }
- }
- static void rmi_f34v7_parse_image_header_10(struct f34_data *f34)
- {
- unsigned int i;
- unsigned int num_of_containers;
- unsigned int addr;
- unsigned int offset;
- unsigned int container_id;
- unsigned int length;
- const void *image = f34->v7.image;
- const u8 *content;
- const struct container_descriptor *descriptor;
- const struct image_header_10 *header = image;
- f34->v7.img.checksum = le32_to_cpu(header->checksum);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev, "%s: f34->v7.img.checksum=%X\n",
- __func__, f34->v7.img.checksum);
- /* address of top level container */
- offset = le32_to_cpu(header->top_level_container_start_addr);
- descriptor = image + offset;
- /* address of top level container content */
- offset = le32_to_cpu(descriptor->content_address);
- num_of_containers = le32_to_cpu(descriptor->content_length) / 4;
- for (i = 0; i < num_of_containers; i++) {
- addr = get_unaligned_le32(image + offset);
- offset += 4;
- descriptor = image + addr;
- container_id = le16_to_cpu(descriptor->container_id);
- content = image + le32_to_cpu(descriptor->content_address);
- length = le32_to_cpu(descriptor->content_length);
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: container_id=%d, length=%d\n", __func__,
- container_id, length);
- switch (container_id) {
- case UI_CONTAINER:
- case CORE_CODE_CONTAINER:
- f34->v7.img.ui_firmware.data = content;
- f34->v7.img.ui_firmware.size = length;
- break;
- case UI_CONFIG_CONTAINER:
- case CORE_CONFIG_CONTAINER:
- f34->v7.img.ui_config.data = content;
- f34->v7.img.ui_config.size = length;
- break;
- case BL_CONTAINER:
- f34->v7.img.bl_version = *content;
- f34->v7.img.bootloader.data = content;
- f34->v7.img.bootloader.size = length;
- rmi_f34v7_parse_img_header_10_bl_container(f34, image);
- break;
- case GUEST_CODE_CONTAINER:
- f34->v7.img.contains_guest_code = true;
- f34->v7.img.guest_code.data = content;
- f34->v7.img.guest_code.size = length;
- break;
- case DISPLAY_CONFIG_CONTAINER:
- f34->v7.img.contains_display_cfg = true;
- f34->v7.img.dp_config.data = content;
- f34->v7.img.dp_config.size = length;
- break;
- case FLASH_CONFIG_CONTAINER:
- f34->v7.img.contains_flash_config = true;
- f34->v7.img.fl_config.data = content;
- f34->v7.img.fl_config.size = length;
- break;
- case GENERAL_INFORMATION_CONTAINER:
- f34->v7.img.contains_firmware_id = true;
- f34->v7.img.firmware_id =
- get_unaligned_le32(content + 4);
- break;
- default:
- break;
- }
- }
- }
- static int rmi_f34v7_parse_image_info(struct f34_data *f34)
- {
- const struct image_header_10 *header = f34->v7.image;
- memset(&f34->v7.img, 0x00, sizeof(f34->v7.img));
- rmi_dbg(RMI_DEBUG_FN, &f34->fn->dev,
- "%s: header->major_header_version = %d\n",
- __func__, header->major_header_version);
- switch (header->major_header_version) {
- case IMAGE_HEADER_VERSION_10:
- rmi_f34v7_parse_image_header_10(f34);
- break;
- default:
- dev_err(&f34->fn->dev, "Unsupported image file format %02X\n",
- header->major_header_version);
- return -EINVAL;
- }
- if (!f34->v7.img.contains_flash_config) {
- dev_err(&f34->fn->dev, "%s: No flash config in fw image\n",
- __func__);
- return -EINVAL;
- }
- rmi_f34v7_parse_partition_table(f34, f34->v7.img.fl_config.data,
- &f34->v7.img.blkcount, &f34->v7.img.phyaddr);
- return 0;
- }
- int rmi_f34v7_do_reflash(struct f34_data *f34, const struct firmware *fw)
- {
- int ret;
- f34->fn->rmi_dev->driver->set_irq_bits(f34->fn->rmi_dev,
- f34->fn->irq_mask);
- rmi_f34v7_read_queries_bl_version(f34);
- f34->v7.image = fw->data;
- f34->update_progress = 0;
- f34->update_size = 0;
- ret = rmi_f34v7_parse_image_info(f34);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_check_bl_config_size(f34);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_erase_all(f34);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_write_partition_table(f34);
- if (ret < 0)
- return ret;
- dev_info(&f34->fn->dev, "%s: Partition table programmed\n", __func__);
- /*
- * Reset to reload partition table - as the previous firmware has been
- * erased, we remain in bootloader mode.
- */
- ret = rmi_scan_pdt(f34->fn->rmi_dev, NULL, rmi_initial_reset);
- if (ret < 0)
- dev_warn(&f34->fn->dev, "RMI reset failed!\n");
- dev_info(&f34->fn->dev, "Writing firmware (%d bytes)...\n",
- f34->v7.img.ui_firmware.size);
- ret = rmi_f34v7_write_firmware(f34);
- if (ret < 0)
- return ret;
- dev_info(&f34->fn->dev, "Writing config (%d bytes)...\n",
- f34->v7.img.ui_config.size);
- f34->v7.config_area = v7_UI_CONFIG_AREA;
- ret = rmi_f34v7_write_ui_config(f34);
- if (ret < 0)
- return ret;
- if (f34->v7.has_display_cfg && f34->v7.img.contains_display_cfg) {
- dev_info(&f34->fn->dev, "Writing display config...\n");
- ret = rmi_f34v7_write_dp_config(f34);
- if (ret < 0)
- return ret;
- }
- if (f34->v7.has_guest_code && f34->v7.img.contains_guest_code) {
- dev_info(&f34->fn->dev, "Writing guest code...\n");
- ret = rmi_f34v7_write_guest_code(f34);
- if (ret < 0)
- return ret;
- }
- return 0;
- }
- static int rmi_f34v7_enter_flash_prog(struct f34_data *f34)
- {
- int ret;
- f34->fn->rmi_dev->driver->set_irq_bits(f34->fn->rmi_dev, f34->fn->irq_mask);
- ret = rmi_f34v7_read_flash_status(f34);
- if (ret < 0)
- return ret;
- if (f34->v7.in_bl_mode) {
- dev_info(&f34->fn->dev, "%s: Device in bootloader mode\n",
- __func__);
- return 0;
- }
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_write_command(f34, v7_CMD_ENABLE_FLASH_PROG);
- if (ret < 0)
- return ret;
- ret = rmi_f34v7_check_command_status(f34, F34_ENABLE_WAIT_MS);
- if (ret < 0)
- return ret;
- return 0;
- }
- int rmi_f34v7_start_reflash(struct f34_data *f34, const struct firmware *fw)
- {
- int ret = 0;
- f34->v7.config_area = v7_UI_CONFIG_AREA;
- f34->v7.image = fw->data;
- ret = rmi_f34v7_parse_image_info(f34);
- if (ret < 0)
- return ret;
- dev_info(&f34->fn->dev, "Firmware image OK\n");
- return rmi_f34v7_enter_flash_prog(f34);
- }
- int rmi_f34v7_probe(struct f34_data *f34)
- {
- int ret;
- /* Read bootloader version */
- ret = rmi_read_block(f34->fn->rmi_dev,
- f34->fn->fd.query_base_addr + V7_BOOTLOADER_ID_OFFSET,
- f34->bootloader_id,
- sizeof(f34->bootloader_id));
- if (ret < 0) {
- dev_err(&f34->fn->dev, "%s: Failed to read bootloader ID\n",
- __func__);
- return ret;
- }
- if (f34->bootloader_id[1] == '5') {
- f34->bl_version = 5;
- } else if (f34->bootloader_id[1] == '6') {
- f34->bl_version = 6;
- } else if (f34->bootloader_id[1] == 7) {
- f34->bl_version = 7;
- } else if (f34->bootloader_id[1] == 8) {
- f34->bl_version = 8;
- } else {
- dev_err(&f34->fn->dev,
- "%s: Unrecognized bootloader version: %d (%c) %d (%c)\n",
- __func__,
- f34->bootloader_id[0], f34->bootloader_id[0],
- f34->bootloader_id[1], f34->bootloader_id[1]);
- return -EINVAL;
- }
- memset(&f34->v7.blkcount, 0x00, sizeof(f34->v7.blkcount));
- memset(&f34->v7.phyaddr, 0x00, sizeof(f34->v7.phyaddr));
- init_completion(&f34->v7.cmd_done);
- ret = rmi_f34v7_read_queries(f34);
- if (ret < 0)
- return ret;
- return 0;
- }
|