If the bootloader fails to start the appmode image on the touch controller, it
stays in bootloader mode. It is possible to reflash a working firmware image
from this state.

Signed-off-by: Nick Dyer <nick.d...@itdev.co.uk>
Acked-by: Benson Leung <ble...@chromium.org>
---
 drivers/input/touchscreen/atmel_mxt_ts.c |   61 +++++++++++++++++++++++-------
 1 file changed, 48 insertions(+), 13 deletions(-)

diff --git a/drivers/input/touchscreen/atmel_mxt_ts.c 
b/drivers/input/touchscreen/atmel_mxt_ts.c
index 8d63780..c18803a 100644
--- a/drivers/input/touchscreen/atmel_mxt_ts.c
+++ b/drivers/input/touchscreen/atmel_mxt_ts.c
@@ -414,6 +414,30 @@ static int mxt_lookup_bootloader_address(struct mxt_data 
*data)
        return 0;
 }
 
+static int mxt_probe_bootloader(struct mxt_data *data)
+{
+       struct device *dev = &data->client->dev;
+       int ret;
+       u8 val;
+       bool crc_failure;
+
+       ret = mxt_lookup_bootloader_address(data);
+       if (ret)
+               return ret;
+
+       ret = mxt_bootloader_read(data, &val, 1);
+       if (ret)
+               return ret;
+
+       /* Check app crc fail mode */
+       crc_failure = (val & ~MXT_BOOT_STATUS_MASK) == MXT_APP_CRC_FAIL;
+
+       dev_err(dev, "Detected bootloader, status:%02X%s\n",
+                       val, crc_failure ? ", APP_CRC_FAIL" : "");
+
+       return 0;
+}
+
 static u8 mxt_get_bootloader_version(struct mxt_data *data, u8 val)
 {
        struct device *dev = &data->client->dev;
@@ -473,6 +497,7 @@ recheck:
        switch (state) {
        case MXT_WAITING_BOOTLOAD_CMD:
        case MXT_WAITING_FRAME_DATA:
+       case MXT_APP_CRC_FAIL:
                val &= ~MXT_BOOT_STATUS_MASK;
                break;
        case MXT_FRAME_CRC_PASS:
@@ -1387,8 +1412,14 @@ static int mxt_initialize(struct mxt_data *data)
        int error;
 
        error = mxt_get_info(data);
-       if (error)
-               return error;
+       if (error) {
+               error = mxt_probe_bootloader(data);
+               if (error)
+                       return error;
+
+               data->in_bootloader = true;
+               return 0;
+       }
 
        data->object_table = kcalloc(info->object_num,
                                     sizeof(struct mxt_object),
@@ -1570,15 +1601,19 @@ static int mxt_load_fw(struct device *dev, const char 
*fn)
        if (ret)
                goto release_firmware;
 
-       /* Change to the bootloader mode */
-       data->in_bootloader = true;
+       if (!data->in_bootloader) {
+               /* Change to the bootloader mode */
+               data->in_bootloader = true;
 
-       ret = mxt_t6_command(data, MXT_COMMAND_RESET, MXT_BOOT_VALUE, false);
-       if (ret)
-               goto release_firmware;
+               ret = mxt_t6_command(data, MXT_COMMAND_RESET,
+                                    MXT_BOOT_VALUE, false);
+               if (ret)
+                       goto release_firmware;
 
-       msleep(MXT_RESET_TIME);
+               msleep(MXT_RESET_TIME);
+       }
 
+       mxt_free_object_table(data);
        INIT_COMPLETION(data->bl_completion);
 
        ret = mxt_check_bootloader(data, MXT_WAITING_BOOTLOAD_CMD);
@@ -1661,8 +1696,6 @@ static ssize_t mxt_update_fw_store(struct device *dev,
        } else {
                dev_info(dev, "The firmware update succeeded\n");
 
-               mxt_free_object_table(data);
-
                error = mxt_initialize(data);
                if (error)
                        return error;
@@ -1946,9 +1979,11 @@ static int mxt_probe(struct i2c_client *client,
        if (error)
                goto err_free_irq;
 
-       error = mxt_initialize_t9_input_device(data);
-       if (error)
-               goto err_free_object;
+       if (!data->in_bootloader) {
+               error = mxt_initialize_t9_input_device(data);
+               if (error)
+                       goto err_free_object;
+       }
 
        error = sysfs_create_group(&client->dev.kobj, &mxt_attr_group);
        if (error) {
-- 
1.7.10.4

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to