This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

Starterware/EK-TM4C1294XL: Firmware Upgrade via serial port shows anomalous behavior.

Part Number: EK-TM4C1294XL

Tool/software: Starterware

Hello.I have tried and studied the bootloader examples that are provided in the examples folder for the TM4C1294Xl launchpad. In my example, i'm trying to flash the firmware bytes (bytes that are generated in the bin file after successful builds) using the API " ROM_FlashProgram((uint32_t *)&FirmwareData, ui32ProgAddr, 4); ". which was explained in the usb_stick_update example.


Serially fetching the bytes on the UART3 on the TIVAC and then successfully writing them on the desired memory blocks as intended. 

The code which performs all this operation is stored from memory location 0x000 and the serial data received is stored from location 0x00011F40.

In my example i'm trying to flash the bin file which is generated after changing the APP START ADDR of the default blinky code to 0x00011F40. 

Now, after flashing the bytes i also verified the contents of the memory via the "Verify flash contents " option in the LM flash programmer.

Everything is perfect up till now.

Here is the code

CODE1

uint_fast32_t ui32ProgAddr = 0x00011F40;
void CallApplication(uint_fast32_t ui32StartAddr)
{
    HWREG(NVIC_VTABLE) = ui32StartAddr;
    __asm("    ldr     r1, [r0]\n"
          "    mov     sp, r1\n");
    __asm("    ldr     r0, [r0, #4]\n"
          "    bx      r0\n");
}
int
main(void)
{

     g_ui32SysClock = MAP_SysCtlClockFreqSet((SYSCTL_XTAL_25MHZ |SYSCTL_OSC_MAIN |SYSCTL_USE_PLL |SYSCTL_CFG_VCO_480), 120000000);

     SysCtlPeripheralEnable(SYSCTL_PERIPH_GPION);
     GPIOPinTypeGPIOOutput(GPIO_PORTN_BASE, GPIO_PIN_0);
// ************************************************************************************** //
     SysCtlPeripheralEnable(SYSCTL_PERIPH_UART3);
     SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
     GPIOPinTypeGPIOOutput(GPIO_PORTA_BASE, GPIO_PIN_6);
     GPIOPinWrite (GPIO_PORTA_BASE, GPIO_PIN_6, GPIO_PIN_6);
     GPIOPinConfigure(GPIO_PA4_U3RX);
     GPIOPinConfigure(GPIO_PA5_U3TX);
     GPIOPinTypeUART(GPIO_PORTA_BASE, GPIO_PIN_4 | GPIO_PIN_5);
     UARTConfigSetExpClk(UART3_BASE, g_ui32SysClock, 115200, (UART_CONFIG_WLEN_8 | UART_CONFIG_STOP_ONE | UART_CONFIG_PAR_NONE));

// ************************************************************************************** //

// ************************************************************************************** //
    while(1)
    {
        if (APPflag == SET) CallApplication(APP_START_ADDRESS);
        while(UARTCharsAvail(UART3_BASE))
        {
            i++;
            rdata[i] = UARTCharGetNonBlocking(UART3_BASE);
            Updatefirmware(i); // This function will write to the memory and set APPflag when all the bytes will be written 
        }
    }
}

The problem comes here in the CallApplication function that is getting called after successful flashing.

    HWREG(NVIC_VTABLE) = 0x00011F40;
    __asm("    ldr     r1, [r0]\n"
          "    mov     sp, r1\n");
    __asm("    ldr     r0, [r0, #4]\n"
          "    bx      r0\n");

Now the observed result is that the LED is blinking very fast. Now this is the abnormal behavior which i was talking about.

I selected the Flash necessary pages option in the CCS debug and flashed this following code into TIVAC.

CODE2

#define APP_START_ADDRESS 0x00011F40
int
main(void)
{

    ROM_SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOJ);
    ROM_GPIODirModeSet(GPIO_PORTJ_BASE, GPIO_PIN_0, GPIO_DIR_MODE_IN);
    MAP_GPIOPadConfigSet(GPIO_PORTJ_BASE, GPIO_PIN_0, GPIO_STRENGTH_2MA, GPIO_PIN_TYPE_STD_WPU);

    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPION);
    while(!SysCtlPeripheralReady(SYSCTL_PERIPH_GPION))
    {
    }

    GPIOPinTypeGPIOOutput(GPIO_PORTN_BASE, GPIO_PIN_0);
    GPIOPinTypeGPIOOutput(GPIO_PORTN_BASE, GPIO_PIN_1);
    while(1)
    {
        if(ROM_GPIOPinRead(GPIO_PORTJ_BASE, GPIO_PIN_0) == 0)
        {
            CallApplication(APP_START_ADDRESS);
        }
    }
}
void CallApplication(uint_fast32_t ui32StartAddr)
{
    HWREG(NVIC_VTABLE) = ui32StartAddr;
    __asm("    ldr     r1, [r0]\n"
          "    mov     sp, r1\n");
    __asm("    ldr     r0, [r0, #4]\n"
          "    bx      r0\n");
}

Note: At this point of time i have the above code(CODE2) starting from 0x000 and the flashed bin file contents of the previous code(CODE1) from 0x00011F40 (Since i selected erase necessary pages only in CCS Debug)

After running CODE2 the execution goes perfectly to the APP_START_ADDR which is 0x00011F40 in my case and the blinky is executed perfectly.

This experiment verifies that the program which i'm using to write on the flash memory(CODE1) is working perfectly. But when i try to call CallApplication() in CODE1 it shows abnormal execution of the program which works perfectly when i try to call CallApplication() from CODE2. I'm not sure how to debug this problem sort of problem.

Regards, 

Nishit.

  • Is the problem related to the Map file or is the problem related to the boot sequence of TIVAC ?
  • Hello Nishit,

    For one,

    if (APPflag == SET) CallApplication(APP_START_ADDRESS);

    That's bad coding and could result in odd bugs, try using:

    if (APPflag == SET)
    {
         CallApplication(APP_START_ADDRESS);
    }

    Secondly, regarding APPflag, where is that being set? I'm not clear on the flow for your CODE 1 example.

    Thirdly, whenever I hear an LED is blinking fast, I think there is a mismatch between expected and actual CPU frequency. Does your Blinky code include a SysCtlClockFreqSet configuration at all?

    Looking at Code 1 and Code 2, Code 2 does not have a SysCtlClockFreqSet configuration, but Code 1 does. Might it might that the SysCtlClockFreqSet in Code 1 is setting the clock faster than the blinky code expects?

  • *** LIKE ***     (somehow - STILL cannot find the "LIKE" button!)

  • Ralph Jacobi said:
    if (APPflag == SET) { CallApplication(APP_START_ADDRESS); }

    Noted.

    Ralph Jacobi said:
    regarding APPflag, where is that being set? I'm not clear on the flow for your CODE 1 example.

    The flag is been set in the Updatefirmware() routine.

    void Updatefirmware(int cnt)
    {
    /*
     * Some calculations on the received data from the UART
     * Then writing the correct bytes at the correct memory address untill all the bytes have been written
     * This is done by ROM_FlashProgram((uint32_t *)&FirmwareData, ui32ProgAddr, 4);
     * After all the bytes have been written set the APPflag as shown below
     */
      APPflag = SET;
    }

    Ralph Jacobi said:
    I think there is a mismatch between expected and actual CPU frequency

    Okay. I'll have a look at this.

    Also one more thing that I'm unaware of-- Does both these resets performs the same functionality?
    1. HWREG(NVIC_APINT) = NVIC_APINT_VECTKEY |NVIC_APINT_SYSRESETREQ;
    2. On board reset button press.

    What does the CPU exactly does (in terms of flow of execution ,register(SP and PC) level, vector table shift etc) when the above procedures are executed individually? There's not enough documentation regarding this 2 things. I'm curious to know about it.

    Regards,
    Nishit.

  • Hi Ralph,

    Thanks so much for spotting it. There was never any problem. The execution and the code was working fine. The LED had different frequency that's why it was having that behavior.

    As cb1_mobile always says, "There should be a LIKE button present."

    Thanks Ralph and cb1_mobile, for replying so quickly and providing to the point answers every time.

    Best regards,

    Nishit.
  • *** LIKE ***

    Good for you - glad that you focused - persisted - accepted direction ... and SUCCEEDED!

    Especially nice that you thanked Ralph. (who makes "above/beyond" efforts to assist!)

    Now to your writing, "LED had different frequency" - by that did you mean that the "blink rate" - fully under your control - was "too fast" - or was Ralph's inkling that "System Clock" (itself) had changed - (the cause of the higher than expected blink rate?)

    Not long before "you" assist others here - and (maybe) forum dictator will "Re-Instate the (so) valuable/sought LIKE!"
  • Hi,

    cb1_mobile said:
    System Clock" (itself) had changed - (the cause of the higher than expected blink rate?)

    Yes. This was exactly what happened. I made changes in the default blinky example by setting the clock which i used in CODE1 and flashed that code in to the microcontroller and observed the output. What i observed was that the LED was blinking at that same fast rate. So the abnormal behavior was actually correct because the controller was running at a different frequency.   

    Also please guide me to this one more question please,

    1. HWREG(NVIC_APINT) = NVIC_APINT_VECTKEY |NVIC_APINT_SYSRESETREQ;
    2. On board reset button press.

    What does the CPU exactly does (in terms of flow of execution ,register(SP and PC) level, vector table shift etc) when the above procedures are executed individually? There's not enough documentation regarding this 2 things. I'm curious to know about it.

     

    Thanks, 

    Nishit.

  • Nishit said:
    What does the CPU exactly do (in terms of flow of execution ,register(SP and PC) level, vector table shift etc)

    Firm & I work w/multiple ARM MCUs - from multiple vendors - so I don't have such "exact" detail, "At the ready."

    This may best be answered by vendor staff - who are "free" to "limit their focus."     ("one and only one" (vendor device behavior) is their charter!)

    Note that if (really) interested - could you not - via "KISS" - experiment - and "tease out" the "exactness" you seek?

  • Hi,

    Okay so as per my observation and the diminutive information provided in the data sheet it confirms that whenever the the reset button is pressed the CPU checks if there is a valid PC and SP value loaded into flash location 0x0 and 0x4. If yes then that program is executed as soon as the user presses the reset button. If the locations are 0xFF ie. not programmed then the PC and SP values are loaded from the memory locations from the ROM (which are predefined locations).  

    Now, my question is, can I via Software (API calls ) delineate this exact behavior ?

    Secondly, where does the execution shifts when this particular statement is executed?

    HWREG(NVIC_APINT) = NVIC_APINT_VECTKEY |NVIC_APINT_SYSRESETREQ;


  • Hello Nishit,

    Nishit said:

    1. HWREG(NVIC_APINT) = NVIC_APINT_VECTKEY |NVIC_APINT_SYSRESETREQ;

    2. On board reset button press.

    Case 1 is a Software Reset.

    Case 2 is a reset via the external reset input pin (~RST).

    The datasheet documentation is the extent of information available for these events though in Section 5.2.2 starting on page 221. I am not entirely clear what you are going to gain by understanding the exact CPU flow...?

    As far as some added information, here is a description from another expert on a past E2E post (note that CM4 = Cortex-M4): "After Power Up the CM4 executes code from the ROM. The peripheral/system registers are reset to the default values and not assigned the default value by the CM4. It then checks for the Flash Application code and if it does not see an application code it has the ROM Boot Loaders to search the serial interfaces for a boot image. If there is a valid application code in Flash (check is to see location 0x0 and 0x4 for a seemingly valid SP-PC), it will jump to the Flash Image."