blob: fdcc0c8ba7cc56d9e4e1fdef1c045cdd7fa6eb17 [file] [log] [blame]
/****************************************************************************
* apps/examples/i2schar/i2schar_receiver.c
*
* SPDX-License-Identifier: Apache-2.0
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/types.h>
#include <stdio.h>
#include <fcntl.h>
#include <pthread.h>
#include <errno.h>
#include <debug.h>
#include <unistd.h>
#include <string.h>
#include <nuttx/audio/audio.h>
#include "i2schar.h"
#ifdef CONFIG_EXAMPLES_I2SCHAR_RX
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/****************************************************************************
* Private Types
****************************************************************************/
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
/****************************************************************************
* Private Data
****************************************************************************/
/****************************************************************************
* Public Data
****************************************************************************/
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: i2schar_receiver()
*
* Description:
* This is the entry point for the receiver thread.
*
****************************************************************************/
pthread_addr_t i2schar_receiver(pthread_addr_t arg)
{
FAR struct ap_buffer_s *apb;
FAR struct ap_buffer_s *transmitter_apb = (FAR struct ap_buffer_s *)arg;
struct audio_buf_desc_s desc;
int bufsize;
int nread;
int ret;
int fd;
int i;
int j;
bool loopback = (transmitter_apb != NULL);
/* Open the I2C character device */
fd = open(g_i2schar.devpath, O_RDONLY);
if (fd < 0)
{
int errcode = errno;
printf("i2schar_receiver: ERROR: failed to open %s: %d\n",
g_i2schar.devpath, errcode);
pthread_exit(NULL);
}
/* Loop for the requested number of times */
for (i = 0; i < g_i2schar.rxcount; i++)
{
/* Allocate an audio buffer of the configured size for receiving data */
desc.numbytes = CONFIG_EXAMPLES_I2SCHAR_BUFSIZE;
desc.u.pbuffer = &apb;
ret = apb_alloc(&desc);
if (ret < 0)
{
printf(
"i2schar_receiver: ERROR: failed to allocate buffer %d: %d\n",
i + 1, ret);
close(fd);
pthread_exit(NULL);
}
bufsize = sizeof(struct ap_buffer_s) + CONFIG_EXAMPLES_I2SCHAR_BUFSIZE;
if (loopback)
{
printf("i2schar_receiver: Using allocated buffer for "
"loopback verification\n");
}
/* Then receive into the buffer */
do
{
/* Flush any output before reading */
fflush(stdout);
/* Read the buffer to the I2S character driver */
nread = read(fd, apb, bufsize);
if (nread < 0)
{
int errcode = errno;
if (errcode != EINTR)
{
printf("i2schar_receiver: ERROR: read failed: %d\n",
errcode);
close(fd);
apb_free(apb);
pthread_exit(NULL);
}
}
else if (nread != bufsize)
{
printf("i2schar_receiver: ERROR: partial read: %d\n",
nread);
close(fd);
apb_free(apb);
pthread_exit(NULL);
}
else
{
printf("i2schar_receiver: Received buffer %d\n", i + 1);
}
}
while (nread != bufsize);
/* Print received buffer data (first 16 bytes) regardless of mode */
printf("i2schar_receiver: Received data (first 16 bytes): ");
for (j = 0; j < 16 && j < CONFIG_EXAMPLES_I2SCHAR_BUFSIZE; j++)
{
printf("0x%02x ", ((uint8_t *)apb->samp)[j]);
}
printf("\n");
/* Verify the received data in loopback mode */
if (loopback)
{
/* Compare the received data with the expected transmitter buffer */
if (memcmp(apb->samp, transmitter_apb->samp,
CONFIG_EXAMPLES_I2SCHAR_BUFSIZE) == 0)
{
printf("i2schar_receiver: Loopback verification PASSED - "
"data matches expected\n");
}
else
{
printf("i2schar_receiver: Loopback verification FAILED - "
"data mismatch!\n");
printf("i2schar_receiver: Expected data (first 16 bytes): ");
for (j = 0;
j < 16 && j < CONFIG_EXAMPLES_I2SCHAR_BUFSIZE; j++)
{
printf("0x%02x ",
((uint8_t *)transmitter_apb->samp)[j]);
}
printf("\n");
close(fd);
apb_free(apb);
pthread_exit((pthread_addr_t)1); /* Return error */
}
}
/* Free the buffer after use */
apb_free(apb);
/* Make sure that the transmitter thread has a chance to run */
pthread_yield();
}
close(fd);
return NULL;
}
#endif /* CONFIG_EXAMPLES_I2SCHAR_RX */