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.

TMS320F28388D: Difference in GPIO pin read delays observed for 'Boot from RAM' and 'Boot from Flash mode'

Part Number: TMS320F28388D

Hi,

I am trying out a simple workflow for F2838x ControlCard checking the GPIO pins. In this case I have used external jumper and shorted GPIO10 and GPIO12. In the code I am configuring GPIO10 as input and GPIO 12 as output. I will set the value of GPIO12 and read the data from GPIO10 immediately to a variable. Here is the code I am using.

5415.gpio_ex1_setup.c
Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
//###########################################################################
//
// FILE: gpio_ex1_setup.c
//
// TITLE: Device GPIO Setup
//! <h1> Device GPIO Setup </h1>
//! \addtogroup bitfield_example_list
//!
//! Configures the F2838X GPIO into two different configurations
//! This code is verbose to illustrate how the GPIO could be setup.
//! In a real application, lines of code can be combined for improved
//! code size and efficiency.
//! This example only sets-up the GPIO.
//! Nothing is actually done with the pins after setup.
//
//#############################################################################
// $TI Release: F2838x Support Library v2.00.00.03 $
// $Release Date: Sun Sep 29 07:45:41 CDT 2019 $
// $Copyright:
// Copyright (C) 2019 Texas Instruments Incorporated - http://www.ti.com/
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions
// are met:
//
// Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
//
// Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the
// distribution.
//
// Neither the name of Texas Instruments Incorporated nor the names of
// its contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// $
//#############################################################################
//
// Included Files
//
#include "f28x_project.h"
//
// Defines
// Select the example to compile in. Only one example should be set as 1
// the rest should be set as 0.
//
#define EXAMPLE1 0 // Basic pinout configuration example
#define EXAMPLE2 0 // Communication pinout example
//
// Function Prototypes
//
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

The input sequence for the variable InputValue (GPIO12) will be [1 1 0 0 ] but the output sequence I observe for Boot from RAM mode is OutputValue = [0 1 1 0] but the output sequence observed for Boot from Flash mode is OutputValue = [1 1 1 0].

  1. Boot from RAM, here we are observing the delay in updating the change for 0 to 1 as well as 1 to 0. This looks like the GPIO data will be updated after the execution of the instruction OutputValue = GpioDataRegs.GPADAT.bit.GPIO10; Hence the delay is observed for both the transitions.
  2. Boot from Flash, here we are observing the delay only in the transition from 1 to 0. The 0 to 1 transition looks proper and is immediately reflected. I am not sure what is the order of execution of statement OutputValue = GpioDataRegs.GPADAT.bit.GPIO10; before or after data is updated for GPIO10.

Is this expected behavior? What is the recommended workflow in these cases?

Thanks,

Aditya

  • Have you configured your flash wait states?

  • Hi Nima,

    In the attached code, the InitSysCtrl function is called first which in turn should call the function InitFlash, if _FLASH is configured. This function I think will configure the required wait state as per the logic below.

    //
    // Set waitstates according to frequency
    //
    // *CAUTION*
    // Minimum waitstates required for the flash operating at a given CPU rate
    // must be characterized by TI. Refer to the datasheet for the latest
    // information.
    //
    #if CPU_FRQ_200MHZ
    Flash0CtrlRegs.FRDCNTL.bit.RWAIT = 0x3;
    #endif

    #if CPU_FRQ_150MHZ
    Flash0CtrlRegs.FRDCNTL.bit.RWAIT = 0x2;
    #endif

    #if CPU_FRQ_120MHZ
    Flash0CtrlRegs.FRDCNTL.bit.RWAIT = 0x2;
    #endif

    Thanks,

    Aditya

  • Run through code, single stepping and make sure these lines get executed.

    Nima

  • Hi Nima,

    Yes I have stepped through these lines of code. The flash wait states configuration are executed properly.

    Thanks,

    Aditya

  •  Do you know what is causing this?

  • Aditya,

    You are executing this in while (1) loop. How are you observing the value for OutputValue at different time ? Can you have four different variables for each assignment so that it is more clear ?

    Regards,

    Vivek Singh

  • Hi Vivek,

    This is the while loop we have in the code.

    while(1)
    {
    InputValue = 1;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;

    DELAY_US(500000);

    InputValue = 1;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;

    DELAY_US(500000);

    InputValue = 0;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;

    DELAY_US(500000);

    InputValue = 0;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;
    }

    In order to observe the behavior I put break points at 4 places.

    1) At the first line of code inside the loop - InputValue = 1;

    2) 3) and 4) At the 3 delays I have in the loop - DELAY_US(500000);

    Every time I observe OutputValue and compare with the input value we set. This value varies when I do boot from Flash or RAM.

    Yes if you want to replicate, you can change the code and have four different values of  OutputValue as shown below  to observe the behavior more clearly.


    Uint16 OutputArray[4] = { 0, 0, 0, 0 };

    while(1)
    {
    InputValue = 1;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;
    OutputArray[0] = OutputValue;

    DELAY_US(500000);

    InputValue = 1;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;
    OutputArray[1] = OutputValue;

    DELAY_US(500000);

    InputValue = 0;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;
    OutputArray[2] = OutputValue;

    DELAY_US(500000);

    InputValue = 0;
    if (InputValue == 1)
    GpioDataRegs.GPASET.bit.GPIO12 = 1;
    else
    GpioDataRegs.GPACLEAR.bit.GPIO12 = 1;

    OutputValue = GpioDataRegs.GPADAT.bit.GPIO10;
    OutputArray[3] = OutputValue;
    }

    Thanks,

    Aditya

  • Have you been able to resolve this?

  • Hi Nima,

    The issue is still present.

    Thanks,

    Aditya

  • Hi Aditya,

    Please confirm that GPIO12 is correctly configured as output by looking at DIR register value in CCS register view after configuration is done.

    There is always some delay between output to input captured and it can take few cycles to update the value in input register after output value is set. To confirm this, can you insert some delay before reading the value (you can use DEALY function) and see if that helps in getting consistent value.

    Regards,

    Vivek Singh

  • Hi Vivek,

    Yes all the pin are configured correctly. I have confirmed that in CCS.

    Yes I do understand that there will be a delay of few cycles to update the value in input register after the output I set. And I have confirmed the same in my initial post. My question was specifically to understand the difference in the data read between Boot from Flash and Boot from RAM when delay is not used.

    1. Boot from RAM, here we are observing the delay in updating the change for 0 to 1 as well as 1 to 0. This looks like the GPIO data will be updated after the execution of the instruction OutputValue = GpioDataRegs.GPADAT.bit.GPIO10; Hence the delay is observed for both the transitions.
    2. Boot from Flash, here we are observing the delay only in the transition from 1 to 0. The 0 to 1 transition looks proper and is immediately reflected. I am not sure what is the order of execution of statement OutputValue = GpioDataRegs.GPADAT.bit.GPIO10; before or after data is updated for GPIO10.

    Thanks,

    Aditya

  • Hi Vivek,

    If we observe the sequence as explained above the different delays observed for transition from 1 to 0 and 0 to 1. But in data sheet I see that rise time and fall time for GPIO Output pins are same. (Chapter 5.8.6.1 GPIO - Output Timing, Table 5-21. General-Purpose Output Switching Characteristic)

    Thanks,

    Aditya

  • Do you have any external connections on the GPIOs?

    Nima

  • Hi Nima,

    Other than the jumper wires connecting GPIO10 and GPIO12 I don't have any thing else connected.

    Thanks,

    Aditya

  • Were you able to resolve this? I don't have any further recommendations.

      anything from your side on this topic?

  • Hi Nima,

    I wanted to understand if this behavior is reproducible from TI side as well. If yes, is the behavior due to some software settings or hardware behavior?

    If it is software then we can arrive at some solution or fix to solve the issue. (Looks like it is not as we verified different register combination in our discussions above.)

    If it is hardware then at least this behavior can be documented for others.

    I am ok to close this post. But it would be good to arrive at some conclusion for benefit of others in future.

    Thanks,

    Aditya

  • Aditya,

    As you noted there is some transport delay between setting the output and seeing the results back on an input.  For predictable testing here you should add a DELAY_US() statement before the OutputValue = GpioDataRegs.GPADAT.bit.GPIO10 .  If these are back to back without delay then there will be variable results that could be influenced by many things.

    Best regards,

    Jason

  • Hi Aditya,

    Besides the delay required, when setting and clearing GPIOs, it's recommended to use GPxSET and GPxCLEAR instead of GPxDAT. The read-modify-write that is performed in bitfield operations can have unintended consequences when GPxDAT is used to set and clear GPIOs.

  • Hi All,

    Thanks for your suggestions on this issue. We can close this discussion.

    Regards,

    Aditya