nrf PDM driver : broken on migration NCS2.6 to NCS2.8

My project uses a PDM microphone connected to a nrf5340. I previousy had this working using the nrfx_pdm driver, with NCS 2.6 but then had to migrate the project to NCS 2.8...

It appears the PDM driver library has updated the version (to 3.7.0 it appears from the ifdefs in nrfx_pdm.c), which broke my code...

The api now provides for multiple PDM instances, so requires an 'instance' pointer to all the calls. This update was relatively simple, but I couldn't find how to do this line:

    IRQ_CONNECT(DT_IRQN(PDM_NL),  IRQ_PRIO_LOWEST, nrfx_pdm_irq_handler, 0, 0);
The nrfx_pdm_irq_handler() function is no longer defined...
If I just comment out this line, then as soon as the pdm sampling starts it crashes due to an 'unhandled interrupt' (which is logical...)
I cannot find a sample code for using the PDM driver on 2.8 (or any version for that matter) that shows the use of the new API.
The documentation page for the driver is (in common with most of the doc) a simple reformating of the header file, with no explanation of the correct way to use the various functions and multiple macros....
Is there a well coded example for the PDM driver available somewhere?
Parents
  • I have found the correct macro for getting the interrupt routine:

        IRQ_CONNECT(DT_IRQN(PDM_NL),  IRQ_PRIO_LOWEST, NRFX_PDM_INST_HANDLER_GET(0), 0, 0);
    This now works ok.
    However, when I attempt to deinit the pdm driver and set the pins to 'sleep' like this:
        if ((errcode = pinctrl_apply_state(PINCTRL_DT_DEV_CONFIG_GET(PDM_NL), PINCTRL_STATE_SLEEP))!=0) {
            log_warn("pdmmgr : PDM set to sleep fails, errcode %d",errcode);
        }
    I get error -2 ie ENOENT. This means that it cannot find the config for the "sleep" case.
    The same call used to setup the pins before use is fine:
        int errcode = pinctrl_apply_state(PINCTRL_DT_DEV_CONFIG_GET(PDM_NL), PINCTRL_STATE_DEFAULT);
        if (errcode!=0) {
            log_warn("pdmmgr : PDM pin setup from DTS fails, errcode %d",errcode);
            return false;
        }
    In the DTS I have defined both sets of pinctls:

    &pdm0 {          
        status = "okay";
        compatible = "nordic,nrf-pdm";
        // label="PDM micro in";
        pinctrl-0 = <&pdm0_default>;
        pinctrl-1 = <&pdm0_sleep>;
        pinctrl-names = "default", "sleep";
        clock-source = "ACLK";
    };
     
    &pinctrl {
        pdm0_default: pdm0_default {
            group1 {
                psels = <NRF_PSEL(PDM_CLK, 0, 5)>,
                    <NRF_PSEL(PDM_DIN, 0, 6)>;
            };
        };

        pdm0_sleep: pdm0_sleep {
            group1 {
                psels = <NRF_PSEL(PDM_CLK, 0, 5)>,
                <NRF_PSEL(PDM_DIN, 0, 6)>;
                low-power-enable;
            };
        };
    };
    Why can't it find the 'sleep' pinctrl set?
    (freely admitting the whole DTS thing is still largely opaque and overly complex for me)
Reply
  • I have found the correct macro for getting the interrupt routine:

        IRQ_CONNECT(DT_IRQN(PDM_NL),  IRQ_PRIO_LOWEST, NRFX_PDM_INST_HANDLER_GET(0), 0, 0);
    This now works ok.
    However, when I attempt to deinit the pdm driver and set the pins to 'sleep' like this:
        if ((errcode = pinctrl_apply_state(PINCTRL_DT_DEV_CONFIG_GET(PDM_NL), PINCTRL_STATE_SLEEP))!=0) {
            log_warn("pdmmgr : PDM set to sleep fails, errcode %d",errcode);
        }
    I get error -2 ie ENOENT. This means that it cannot find the config for the "sleep" case.
    The same call used to setup the pins before use is fine:
        int errcode = pinctrl_apply_state(PINCTRL_DT_DEV_CONFIG_GET(PDM_NL), PINCTRL_STATE_DEFAULT);
        if (errcode!=0) {
            log_warn("pdmmgr : PDM pin setup from DTS fails, errcode %d",errcode);
            return false;
        }
    In the DTS I have defined both sets of pinctls:

    &pdm0 {          
        status = "okay";
        compatible = "nordic,nrf-pdm";
        // label="PDM micro in";
        pinctrl-0 = <&pdm0_default>;
        pinctrl-1 = <&pdm0_sleep>;
        pinctrl-names = "default", "sleep";
        clock-source = "ACLK";
    };
     
    &pinctrl {
        pdm0_default: pdm0_default {
            group1 {
                psels = <NRF_PSEL(PDM_CLK, 0, 5)>,
                    <NRF_PSEL(PDM_DIN, 0, 6)>;
            };
        };

        pdm0_sleep: pdm0_sleep {
            group1 {
                psels = <NRF_PSEL(PDM_CLK, 0, 5)>,
                <NRF_PSEL(PDM_DIN, 0, 6)>;
                low-power-enable;
            };
        };
    };
    Why can't it find the 'sleep' pinctrl set?
    (freely admitting the whole DTS thing is still largely opaque and overly complex for me)
Children
No Data
Related