PWM Sequence from HAL to NRFX

Hello

I am trying to change the code from the HAL code up to NRFX in order to make it event-based instead of using infinite while loops. 

I am using 52840 on a custom PCA with goal in mind to create a half-duplex-half-UART communication.

Working with HAL drivers, the code works as intended, but it is not optimized and is very power hungry for our needs. 

The HAL Setup and sending of the message (Byte by byte, each byte 10 bits at a time (Start - 8xData - Stop)): 

TxSetup()
{
    NRF_P0->DIRSET = ((1UL) << PIN_NEG); // PIN_NEG as output
    NRF_P0->OUTCLR = ((1UL) << PIN_NEG); // initial LOW on PIN_NEG

    NRF_P0->DIRSET = ((1UL) << PIN_POS); // PIN_POS as output
    NRF_P0->OUTSET = ((1UL) << PIN_POS); // initial HIGH on PIN_POS

    CounterTopRegister = F_CLK / F_PWM; // F_CLK = 16000000
    cyclesPerBit = F_PWM / LL_BAUDRATE; // F_PWM = 32400 ; LL_BAUDRATE = 1200
    P_PWM_HIGH = CounterTopRegister;
    P_PWM_LOW = (int)(0.5 * CounterTopRegister);
    N_PWM_HIGH = P_PWM_HIGH + 0x8000;
    N_PWM_LOW = P_PWM_LOW + 0x8000;

    NRF_PWM2->PSEL.OUT[0] = (PIN_POS << PWM_PSEL_OUT_PIN_Pos) | (PIN_POS << PWM_PSEL_OUT_PORT_Pos) | (PWM_PSEL_OUT_CONNECT_Connected << PWM_PSEL_OUT_CONNECT_Pos);
    NRF_PWM2->PSEL.OUT[1] = (PIN_NEG << PWM_PSEL_OUT_PIN_Pos) | (PIN_NEG << PWM_PSEL_OUT_PORT_Pos) | (PWM_PSEL_OUT_CONNECT_Connected << PWM_PSEL_OUT_CONNECT_Pos);
    NRF_PWM2->MODE = (PWM_MODE_UPDOWN_Up << PWM_MODE_UPDOWN_Pos);
    NRF_PWM2->PRESCALER = (PWM_PRESCALER_PRESCALER_DIV_1 << PWM_PRESCALER_PRESCALER_Pos);
    NRF_PWM2->COUNTERTOP = (CounterTopRegister << PWM_COUNTERTOP_COUNTERTOP_Pos);
    NRF_PWM2->LOOP = (0 << PWM_LOOP_CNT_Pos);
    NRF_PWM2->DECODER = (PWM_DECODER_LOAD_Individual << PWM_DECODER_LOAD_Pos) | (PWM_DECODER_MODE_RefreshCount << PWM_DECODER_MODE_Pos);
    NRF_PWM2->SEQ[0].PTR = ((uint32_t)(initialSequence) << PWM_SEQ_PTR_PTR_Pos);

    NRF_PWM2->SEQ[0].CNT = 44;
    NRF_PWM2->SEQ[0].REFRESH = cyclesPerBit - 1;
    NRF_PWM2->SEQ[0].ENDDELAY = 0;
}

void TxWrite(uint8_t txVal)
{
    int i;
    // Generate the Startbit (LOW)
    initialSequence[0] = P_PWM_LOW;
    initialSequence[1] = N_PWM_LOW;
    initialSequence[2] = 0;
    initialSequence[3] = 0;
    // Generate the 8 databits
    for (i = 1; i < 9; i++)
    {
        if (txVal & 1)
        {
            // Bit is HIGH
            initialSequence[4 * i + 0] = P_PWM_HIGH;
            initialSequence[4 * i + 1] = N_PWM_HIGH;
        }
        else
        {
            // Bit is LOW
            initialSequence[4 * i + 0] = P_PWM_LOW;
            initialSequence[4 * i + 1] = N_PWM_LOW;
        }
        initialSequence[4 * i + 2] = 0;
        initialSequence[4 * i + 3] = 0;
        txVal = txVal >> 1;
    }

    // Generate the Stopbit (HIGH)
    initialSequence[36] = P_PWM_HIGH;
    initialSequence[37] = N_PWM_HIGH;
    initialSequence[38] = 0;
    initialSequence[39] = 0;
    initialSequence[40] = P_PWM_HIGH;
    initialSequence[41] = N_PWM_HIGH;
    initialSequence[42] = 0;
    initialSequence[43] = 0;

    NRF_PWM2->ENABLE = (PWM_ENABLE_ENABLE_Enabled << PWM_ENABLE_ENABLE_Pos);

    NRF_PWM2->TASKS_SEQSTART[0] = 1;
}

void TxWaitDone()
{

    while (NRF_PWM2->EVENTS_SEQEND[0] == 0)
        ;
    NRF_PWM2->EVENTS_SEQEND[0] = 0;
}

void TxStop()
{

    NRF_PWM2->TASKS_STOP = 1;

    while (NRF_PWM2->EVENTS_STOPPED == 0)
        ;
    NRF_PWM2->EVENTS_STOPPED = 0;

    NRF_PWM2->ENABLE = (PWM_ENABLE_ENABLE_Disabled << PWM_ENABLE_ENABLE_Pos);
}


WriteMessage(uint8_t *msg, uint8_t len)
{
    nrfx_lpcomp_disable();
    NRFX_IRQ_DISABLE(SAADC_IRQn);
    for (int i = 0; i < len; i++)
    {
        TxWrite(msg[i]);
        TxWaitDone();
        TxStop();
    }
    lpcomp_event_count = 0;
    is_carrier_found = 0;
    NRFX_IRQ_ENABLE(COMP_LPCOMP_IRQn);
    nrfx_lpcomp_enable();
    return 0;
}

The issue with this approach, again, is the while loops that, among other issues, can sometimes get stuck and hang the rest of the process. 

The actual issue rewriting this with NRFX drivers is the fact that I cannot, for the life of me, figure out how to write out my "initialSequence." I can get the PWM to toggle on or off, but not actually make it do that in sequence as I need it to. I tried following a few examples I found on Devzone, but nothing concrete enough to help me with my issue. 

The HAL Implementation repeats every 100 milliseconds, so the NRFX one should be able to match that. If anyone has an idea of how to write this using the NRFX drivers, or to point me in the direction of a set-up example, I'd be very grateful. 
To add to this: The idea is for this code to run along with a Zephyr based system that runs the rest of the functionality of the board (Thus why we're trying to make it event based as opposed to current implementation).

Thank you in advanced for response. If you need any more information, please ask. 

EDIT: Added the missing code that was calling the TX sequence in order. (Write, Wait, Stop)

Parents
  • I should add the current (latest) attempt at implementing, so maybe that could be corrected in the right way:

    nrfx_pwm_t pwm = NRFX_PWM_INSTANCE(2);
    static nrf_pwm_values_individual_t pwm_values[2];
    nrf_pwm_sequence_t pwm_sequence;
    
    void pwm_tx_init()
    {
        nrfx_pwm_config_t pwm_config;
    
        pwm_config.output_pins[0] = PIN_POS;
        pwm_config.output_pins[1] = PIN_NEG;
        pwm_config.output_pins[2] = NRFX_PWM_PIN_NOT_USED;
        pwm_config.output_pins[3] = NRFX_PWM_PIN_NOT_USED;
    
        pwm_config.base_clock = NRF_PWM_CLK_16MHz;
        pwm_config.count_mode = NRF_PWM_MODE_UP;
        pwm_config.top_value = (F_CLK / F_PWM);
        pwm_config.load_mode = NRF_PWM_LOAD_WAVE_FORM;
        pwm_config.step_mode = NRF_PWM_STEP_AUTO;
        pwm_config.skip_gpio_cfg = false;
        pwm_config.skip_psel_cfg = false;
        pwm_config.irq_priority = 5;
    
        nrfx_pwm_init(&pwm, &pwm_config, pwm_transmit_evt_handler, 0);
        IRQ_DIRECT_CONNECT(COMP_LPCOMP_IRQn, 5, nrfx_pwm_2_irq_handler, 0);
    }
    
    
    void pwm_tx_transmit(uint8_t *msg)
    {
        set_sequence(0x80);
        nrf_pwm_values_t sequence_values;
        sequence_values.p_raw = initialSequence;
        nrf_pwm_sequence_t sequence1;
        sequence1.repeats = (F_PWM / LL_BAUDRATE) - 1;
        sequence1.length = 44;
        sequence1.end_delay = 0;
        sequence1.values = sequence_values;
       
    }
    void make_sequence(uint16_t duty_p_p, uint16_t duty_p_n, uint16_t duty_n_p, uint16_t duty_n_n)
    {
    
        pwm_sequence.values.p_individual = pwm_values;
        pwm_sequence.length = NRF_PWM_VALUES_LENGTH(pwm_values);
        uint32_t repeats = (F_PWM / LL_BAUDRATE) - 1;
        pwm_sequence.repeats = repeats;
        pwm_sequence.end_delay = 0;
    
        pwm_values[0].channel_0 = duty_p_p;
        pwm_values[0].channel_1 = duty_p_n;
    
        pwm_values[1].channel_0 = duty_n_p;
        pwm_values[1].channel_1 = duty_n_n;
        nrfx_pwm_simple_playback(&pwm, &pwm_sequence, 1, NRFX_PWM_FLAG_STOP);
    }

    This is just a few itterations after, at this point I'm unsure if it has any relation to my initial sequence anyway. 
    The "initialSequence" array is filled in the same way as in the HAL implementation.

  • I can't get this reply to line up, but anyway I see your issue. The PWM continues to play the last (maybe only)  cycle indefinitely until you issue a STOP TASK and wait for the STOPPED Event. This can be automated by setting SHORTS:

       NRF_PWM_Type * const pNRF_PWM = NRF_PWM2;
       // PWM_SHORTS_SEQEND0_STOP_Msk         Shortcut between SEQEND[0] event and STOP task.
       // PWM_SHORTS_SEQEND1_STOP_Msk         Shortcut between SEQEND[1] event and STOP task.
       // PWM_SHORTS_LOOPSDONE_SEQSTART0_Msk  Shortcut between LOOPSDONE event and SEQSTART[0] task.
       // PWM_SHORTS_LOOPSDONE_SEQSTART1_Msk  Shortcut between LOOPSDONE event and SEQSTART[1] task.
       // PWM_SHORTS_LOOPSDONE_STOP_Msk        Shortcut between LOOPSDONE event and STOP task.
       pNRF_PWM->SHORTS = NRF_PWM_SHORT_LOOPSDONE_STOP_MASK;

    "After the last value in the sequence has been loaded and started executing, a SEQEND[n] event is generated. The PWM generation will then continue with the last loaded value."

Reply
  • I can't get this reply to line up, but anyway I see your issue. The PWM continues to play the last (maybe only)  cycle indefinitely until you issue a STOP TASK and wait for the STOPPED Event. This can be automated by setting SHORTS:

       NRF_PWM_Type * const pNRF_PWM = NRF_PWM2;
       // PWM_SHORTS_SEQEND0_STOP_Msk         Shortcut between SEQEND[0] event and STOP task.
       // PWM_SHORTS_SEQEND1_STOP_Msk         Shortcut between SEQEND[1] event and STOP task.
       // PWM_SHORTS_LOOPSDONE_SEQSTART0_Msk  Shortcut between LOOPSDONE event and SEQSTART[0] task.
       // PWM_SHORTS_LOOPSDONE_SEQSTART1_Msk  Shortcut between LOOPSDONE event and SEQSTART[1] task.
       // PWM_SHORTS_LOOPSDONE_STOP_Msk        Shortcut between LOOPSDONE event and STOP task.
       pNRF_PWM->SHORTS = NRF_PWM_SHORT_LOOPSDONE_STOP_MASK;

    "After the last value in the sequence has been loaded and started executing, a SEQEND[n] event is generated. The PWM generation will then continue with the last loaded value."

Children
  • Replies seem to have issues for some reason. 

    But regarding the issue, I think you misunderstood me, I have issue actually setting up the PWM ( I Think).

    I call "nrfx_pwm_simple_playback(&pwm, &pwm_sequence, 1, NRFX_PWM_FLAG_STOP);"
    And this is what drives my sequence. I no longer want to have direct registry access, unless absolutely needed. I wonder what, from the provided code, is wrong with the way I'm using NRFX API. And what do I need to modify in the code for it to execute the same function that my HAL code does. 
    The best I can achieve with NRFX is making PWM output a loop of repeating peak values at a defined duty cycle. 
    And it keeps being stuck on 1khz, where the HAL code works on 32Khz. 

  • Oh, maybe try changing the LOOPS (disguised as repeats):

        uint32_t repeats = (F_PWM / LL_BAUDRATE) - 1;
        uint32_t repeats = 0;

  • Still no Cigar. 

    I thought that LOOPS are actually the "playback Count" in the arguments of simple_playback. 
    And the Repeats was the same as SEQ[n].REFRESH? 

    My assumption is that something in the API is not properly configured, or I am not doign something in the right order. 

  • 1, not 0, for the repeats - sorry my typo. Maybe post the current code you are using, the earlier code is a little hard to follow with some unused code. Also include the main sequence if possible. I sometimes use the same api without any issues, though often use the bare metal drivers. A screen shot of the PWM registers might help in debug to confirm actual values loaded are not being obfuscated by the library code.

    REFRESH is a way of using more clock cycles per PWM value step, so typically set to 0.

  • Here's the current code in NRFX: 

    void pwm_transmit_evt_handler(nrfx_pwm_evt_type_t event_type, void *p_context)
    {
        switch (event_type)
        {
        case NRFX_PWM_EVT_FINISHED:
            LOG_INF("PWM EVENT! NRFX_PWM_EVT_FINISHED");
            break;
        case NRFX_PWM_EVT_END_SEQ0:
            LOG_INF("PWM EVENT! NRFX_PWM_EVT_END_SEQ0");
            break;
        case NRFX_PWM_EVT_END_SEQ1:
            LOG_INF("PWM EVENT! NRFX_PWM_EVT_END_SEQ1");
            break;
        case NRFX_PWM_EVT_STOPPED:
            LOG_INF("PWM EVENT! NRFX_PWM_EVT_STOPPED");
            break;
        }
    }
    
    void pwm_tx_init()
    {
        uint16_t CounterTopRegister;
        CounterTopRegister = F_CLK / F_PWM;
        nrfx_pwm_config_t pwm_config;
    
        pwm_config.output_pins[0] = PIN_POS;
        pwm_config.output_pins[1] = PIN_NEG;
        pwm_config.output_pins[2] = NRFX_PWM_PIN_NOT_USED;
        pwm_config.output_pins[3] = NRFX_PWM_PIN_NOT_USED;
    
        pwm_config.base_clock = NRF_PWM_CLK_16MHz ;
        pwm_config.count_mode = NRF_PWM_MODE_UP;
        pwm_config.top_value = CounterTopRegister;
        pwm_config.load_mode = NRF_PWM_LOAD_INDIVIDUAL;
        pwm_config.step_mode = NRF_PWM_STEP_AUTO;
        pwm_config.skip_gpio_cfg = false;
        pwm_config.skip_psel_cfg = false;
        pwm_config.irq_priority = 5;
    
        nrfx_pwm_init(&pwm, &pwm_config, pwm_transmit_evt_handler, 0);
        IRQ_DIRECT_CONNECT(COMP_LPCOMP_IRQn, 5, nrfx_pwm_2_irq_handler, 0);
    }
    
    void set_sequence(uint8_t txVal)
    {
        uint16_t P_PWM_HIGH;
        uint16_t P_PWM_LOW;
        uint16_t N_PWM_HIGH;
        uint16_t N_PWM_LOW;
        P_PWM_HIGH = (F_CLK / F_PWM);
        P_PWM_LOW = (int)(0.5 * (F_CLK / F_PWM));
        N_PWM_HIGH = P_PWM_HIGH + 0x8000;
        N_PWM_LOW = P_PWM_LOW + 0x8000;
    
        int i;
    
        // Generate the Startbit (LOW)
        initialSequence[0] = P_PWM_LOW;
        initialSequence[1] = N_PWM_LOW;
        initialSequence[2] = 0;
        initialSequence[3] = 0;
        //  LOG_INF("S");
    
        // Generate the 8 databits
        for (i = 1; i < 9; i++)
        {
            if (txVal & 1)
            {
                // Bit is HIGH
                //  LOG_INF("1");
                initialSequence[4 * i + 0] = P_PWM_HIGH;
                initialSequence[4 * i + 1] = N_PWM_HIGH;
            }
            else
            {
                // Bit is LOW
                //  LOG_INF("0");
                initialSequence[4 * i + 0] = P_PWM_LOW;
                initialSequence[4 * i + 1] = N_PWM_LOW;
            }
            initialSequence[4 * i + 2] = 0;
            initialSequence[4 * i + 3] = 0;
            txVal = txVal >> 1;
        }
    
        // Generate the Stopbit (HIGH)
        initialSequence[36] = P_PWM_HIGH;
        initialSequence[37] = N_PWM_HIGH;
        initialSequence[38] = 0;
        initialSequence[39] = 0;
        initialSequence[40] = P_PWM_HIGH;
        initialSequence[41] = N_PWM_HIGH;
        initialSequence[42] = 0;
        initialSequence[43] = 0;
    }
    
    
    nrf_pwm_sequence_t pwm_sequence;
    
    void make_sequence()
    {
        pwm_sequence.values.p_raw = initialSequence;
        pwm_sequence.length = 44;
        pwm_sequence.repeats = 18;
        pwm_sequence.end_delay = 0;
    
        nrfx_pwm_simple_playback(&pwm, &pwm_sequence, 0, 0);
    }
    
    void pwm_tx_transmit(uint8_t msg)
    {
        set_sequence(msg);
        make_sequence();
    }
    
    int TxMessage(uint8_t *msg, uint8_t len)
    {
        nrfx_lpcomp_disable();
    
        NRFX_IRQ_DISABLE(SAADC_IRQn);
        for (int i = 0; i < len; i++)
        {
            pwm_tx_transmit(msg[i]);
        }
        lpcomp_event_count = 0;
        is_carrier_found = 0;
        NRFX_IRQ_ENABLE(COMP_LPCOMP_IRQn);
        nrfx_lpcomp_enable();
        return 0;
    }
    
    
    // Elsewhere in Code:
    void Init()
    {
        pwm_tx_init();
        initLpComp();
    }
    
    request_data()
    {
        uint8_t STX = 0x82;
        uint8_t signal_id = 0x64 | 0x80;
        uint8_t data_byte_0 = 0x03 | 0x80;
    
        uint8_t checksum = (signal_id) + (data_byte_0);
        checksum &= 0x7F;
    
        uint8_t message[4] = {STX, signal_id, data_byte_0, checksum};
        TxMessage(message, 4);
    }

    After trying and implementing your method of constructing the message, the result was no different. Following your advice on pins being set/cleared. I've specifically set the pin, but no cigar. 

    Here is the Screengrab of the registers. On the left side is the current code (NOT WORKING) 
    On the right side is the Bare-metal/HAL Code (WORKING). 

    One thing I've noticed, that keeps being obvious, is that no matter what I do  to the top_value, the countertop doesn't change in any way. Likewise, the DECODER register keeps being missmatched, no matter which one I try. Also for some reason with the NRFX code there is value on PTR_1, CNT_1, REFRESH_1. Unlike the Bare Metal approach. 

    Here is the Scope screen grab of the Current NRFX approach. The bare metal approach still looks like in the image provided earlier. 

    If you have any clue as to what is causing this issue, that'd be amazing. 

    I have edited the code according to the bare-metal code trying to match as many registers as possible. But it does not seem to work for me in any approach.


    Thank you so much for taking your time to try and help me out! 

Related