How to play pcm raw data on nrf52840 DK through I2S ?

Hello,

I'm using nrf52840 DK to test 1khz tone. The nRF Connect SDK version is v2.6.0.

I put a pcm raw data array of 1khz tone in the code and used i2s write to send the data to i2s out.

Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
const struct device *const i2s_dev_tx = DEVICE_DT_GET(I2S_TX_NODE);
const struct device *const i2s_dev_rx = DEVICE_DT_GET(I2S_TX_NODE);
struct i2s_config config;
#define SAMPLE_FREQUENCY 48000
#define SAMPLE_BIT_WIDTH 16
#define BYTES_PER_SAMPLE sizeof(int16_t)
#define NUMBER_OF_CHANNELS 2
/* Such block length provides an echo with the delay of 100 ms. */
#define SAMPLES_PER_BLOCK ((SAMPLE_FREQUENCY / 10) * NUMBER_OF_CHANNELS)
#define INITIAL_BLOCKS 2
#define TIMEOUT 1000
#define BLOCK_SIZE (BYTES_PER_SAMPLE * SAMPLES_PER_BLOCK)
#define BLOCK_COUNT (INITIAL_BLOCKS + 2)
K_MEM_SLAB_DEFINE_STATIC(mem_slab, BLOCK_SIZE, BLOCK_COUNT, 4);
static int16_t data_buf_tmp[9600];
static bool configure_streams(const struct device *i2s_dev_tx,const struct i2s_config *config);
static void i2s_init(void){
int ret;
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

My pcm raw data array is an int16 stereo array, arranged as:
[left_data0,right_data0,left_data1,right_data1,....]

After testing, it is found that the output sound is distorted. Is there anything that needs to be adjusted?

  • Hi,

    There are several potential sources for distortion (this thread has a good discussion on that). Can you describe the distortion or share a sound recording? Also, how is the data fomated/encoded? Can you share the data buffer you are playing out with this code for reference?

  • Thank you for your reply.

    The state of distortion is that the sound is several whole tones lower than the real 1khz tone.

    I did not encode the data, I expected to use i2s to output pcm raw data.

    So I found a 1khz tone wav audio file and used the tool to convert it into an int16 data array.

    Here is my data buffer.

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    #define NUM_ELEMENTS 9740
    const short data[NUM_ELEMENTS] = {
    -26961, 0, -24693, 0, -22016, 0, -18947, 0, /* 0-7 */
    -15567, 0, -11911, 0, -8057, 0, -4062, 0, /* 8-15 */
    -5, 0, 4069, 0, 8052, 0, 11918, 0, /* 16-23 */
    15565, 0, 18949, 0, 22012, 0, 24695, 0, /* 24-31 */
    26957, 0, 28761, 0, 30066, 0, 30865, 0, /* 32-39 */
    31128, 0, 30860, 0, 30072, 0, 28754, 0, /* 40-47 */
    26969, 0, 24685, 0, 22020, 0, 18941, 0, /* 48-55 */
    15569, 0, 11911, 0, 8055, 0, 4064, 0, /* 56-63 */
    -4, 0, -4063, 0, -8057, 0, -11915, 0, /* 64-71 */
    -15564, 0, -18952, 0, -22010, 0, -24695, 0, /* 72-79 */
    -26961, 0, -28755, 0, -30073, 0, -30858, 0, /* 80-87 */
    -31132, 0, -30861, 0, -30068, 0, -28763, 0, /* 88-95 */
    -26949, 0, -24706, 0, -21998, 0, -18960, 0, /* 96-103 */
    -15554, 0, -11920, 0, -8050, 0, -4066, 0, /* 104-111 */
    4, 0, 4063, 0, 8061, 0, 11914, 0, /* 112-119 */
    15566, 0, 18951, 0, 22013, 0, 24695, 0, /* 120-127 */
    26963, 0, 28759, 0, 30070, 0, 30864, 0, /* 128-135 */
    31128, 0, 30865, 0, 30064, 0, 28763, 0, /* 136-143 */
    26957, 0, 24696, 0, 22011, 0, 18950, 0, /* 144-151 */
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

  • Here is my output sound recording.Play this audio clip

    Play this audio clip

  • I have solved this problem by just changing this line

    Fullscreen
    1
    ret = i2s_trigger(i2s_dev_tx, I2S_DIR_TX, I2S_TRIGGER_DROP);
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    modify to

    Fullscreen
    1
    ret = i2s_trigger(i2s_dev_tx, I2S_DIR_TX, I2S_TRIGGER_DRAIN);
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    The output sound is no longer distorted and now I can hear a normal 1khz tone.

    But I have now encountered a new problem. When I want to play 1khz tone continuously, reset or noise will occur.

    When my code is set as follows:

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    for(;;){
    prepare_transfer(i2s_dev_tx);
    // Put data into the tx buffer
    ...
    ret = i2s_trigger(i2s_dev_tx, I2S_DIR_TX, I2S_TRIGGER_START);
    if (ret < 0) {
    LOG_ERR("Failed to trigger command I2S_TRIGGER_START on TX: %d\n", ret);
    }
    ret = i2s_write(i2s_dev_tx, mem_block, BLOCK_SIZE);
    if (ret < 0) {
    LOG_ERR("Failed to write data: %d", ret);
    }else{
    LOG_INF("Success to write data.");
    }
    ret = i2s_trigger(i2s_dev_tx, I2S_DIR_TX, I2S_TRIGGER_DRAIN);
    if (ret < 0) {
    LOG_ERR("Failed to trigger command I2S_TRIGGER_DROP on TX: %d\n", ret);
    }
    }
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    After playing for a period of time, an error will occur causing a reset. The error log is as follows:

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    *** Booting nRF Connect SDK v3.5.99-ncs1 ***
    [00:00:00.330,932] <inf> i2s_nrfx: I2S MCK frequency: 1523809, actual PCM rate: 47619
    [00:00:00.333,801] <inf> central_test: loop times:1
    [00:00:00.340,179] <inf> central_test: Success to write data.
    [00:00:00.340,209] <inf> central_test: loop times:2
    [00:00:00.342,041] <err> i2s_nrfx: Cannot write in state: 3
    [00:00:00.342,071] <err> central_test: Failed to write block 0: -5
    [00:00:00.344,757] <err> central_test: Failed to trigger command I2S_TRIGGER_START on TX: -5
    [00:00:00.344,787] <err> i2s_nrfx: Cannot write in state: 3
    [00:00:00.344,787] <err> central_test: Failed to write data: -5
    [00:00:00.344,787] <err> central_test: Failed to trigger command I2S_TRIGGER_DROP on TX: -5
    [00:00:00.344,818] <inf> central_test: loop times:3
    [00:00:00.346,618] <err> i2s_nrfx: Cannot write in state: 3
    [00:00:00.346,649] <err> central_test: Failed to write block 0: -5
    [00:00:00.349,334] <err> central_test: Failed to trigger command I2S_TRIGGER_START on TX: -5
    [00:00:00.349,365] <err> i2s_nrfx: Cannot write in state: 3
    [00:00:00.349,365] <err> central_test: Failed to write data: -5
    [00:00:00.349,365] <err> central_test: Failed to trigger command I2S_TRIGGER_DROP on TX: -5
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    When I move prepare_transfer outside the for-loop, although it is no longer reset, there is noise.

    The error log is as follows:

    Fullscreen
    1
    2
    3
    4
    5
    [00:00:00.393,951] <err> central_test: Failed to trigger command I2S_TRIGGER_START on TX: -5
    [00:00:00.393,981] <err> i2s_nrfx: Cannot write in state: 3
    [00:00:00.393,981] <err> central_test: Failed to write data: -5
    [00:00:00.393,981] <err> central_test: Failed to trigger command I2S_TRIGGER_DROP on TX: -5
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    When I move i2s_trigger outside the for-loop

    An error will occur after i2s_write is played several times.

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    *** Booting nRF Connect SDK v3.5.99-ncs1 ***
    [00:00:00.386,260] <inf> i2s_nrfx: I2S MCK frequency: 1523809, actual PCM rate: 47619
    [00:00:00.395,660] <inf> central_test: Success to write data.
    [00:00:00.395,660] <inf> central_test: Success to write data.
    [00:00:00.396,026] <inf> central_test: Success to write data.
    [00:00:00.396,057] <inf> central_test: Success to write data.
    [00:00:00.496,856] <inf> central_test: Success to write data.
    [00:00:01.496,948] <err> central_test: Failed to write data: -11
    [00:00:02.497,100] <err> central_test: Failed to write data: -11
    [00:00:03.497,253] <err> central_test: Failed to write data: -11
    [00:00:04.497,375] <err> central_test: Failed to write data: -11
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    How to set up so that i2s write can continuously play 1khz tone?

  • Hi,

    I am glad to hear you resovled the first issue with distortion.

    Regarding the second issue, calling your prepare_transfer() in the main loop allocats memory with k_mem_slab_alloc(). As this is never freeed, calling this function in a loop will cause a memory leak, so that it fails is expected, so moving it out of the loop is needed.

    I assume the noise you hear is due to a pause in the playout? The sequence you play out should be so that the start of the sequence match the end (with no step in the signal) so that when played back-to-back it gives a perfect sine wave. However, in this approach you are doing separate transactions (stopping and re-starting and that takes some CPU time. That is fixable though, as the I2S peripheral's transmit and receive pointer registers are double buffered (see specification), so to play out continiously you can start, and then supply a new buffer regularily (if a simple sine way you could point to the same buffer again and again). Then you should not trigger I2S_TRIGGER_DRAIN, as this will stop transmission, which is not what you want if you want to send a continious signal. Essentially, it should be enough to call i2s_write() in a loop after starting.

1 2