2011-03-20 19:18:19 +01:00
|
|
|
/****************************************************************************
|
2021-06-16 09:22:16 +02:00
|
|
|
* apps/examples/pipe/transfer_test.c
|
2011-03-20 19:18:19 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* 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
|
2011-03-20 19:18:19 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2011-03-20 19:18:19 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* 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.
|
2011-03-20 19:18:19 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "pipe.h"
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Pre-proecessor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define MAX_BYTE 13
|
|
|
|
|
|
|
|
#define WRITE_SIZE MAX_BYTE
|
|
|
|
#define NWRITES 1400
|
|
|
|
#define NWRITE_BYTES (NWRITES * WRITE_SIZE)
|
|
|
|
|
|
|
|
#define READ_SIZE (2*MAX_BYTE)
|
|
|
|
#define NREADS (NWRITES / 2)
|
|
|
|
#define NREAD_BYTES NWRITE_BYTES
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: transfer_reader
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void *transfer_reader(pthread_addr_t pvarg)
|
|
|
|
{
|
|
|
|
char buffer[READ_SIZE];
|
|
|
|
int fd = (int)pvarg;
|
|
|
|
int ret;
|
|
|
|
int nbytes;
|
|
|
|
int value;
|
|
|
|
int ndx;
|
|
|
|
|
|
|
|
printf("transfer_reader: started\n");
|
|
|
|
for (nbytes = 0, value = 0; nbytes < NREAD_BYTES;)
|
|
|
|
{
|
|
|
|
ret = read(fd, buffer, READ_SIZE);
|
|
|
|
if (ret < 0 )
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_reader: read failed, errno=%d\n", errno);
|
|
|
|
return (void*)1;
|
|
|
|
}
|
|
|
|
else if (ret == 0)
|
|
|
|
{
|
|
|
|
if (nbytes < NREAD_BYTES)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_reader: Too few bytes read -- aborting: %d\n", nbytes);
|
|
|
|
return (void*)2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
for (ndx = 0; ndx < ret; ndx++)
|
|
|
|
{
|
|
|
|
if (value >= WRITE_SIZE)
|
|
|
|
{
|
|
|
|
value = 0;
|
|
|
|
}
|
|
|
|
if (buffer[ndx] != value)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_reader: Byte %d, expected %d, found %d\n",
|
|
|
|
nbytes + ndx, value, buffer[ndx]);
|
|
|
|
return (void*)3;
|
|
|
|
}
|
|
|
|
value++;
|
|
|
|
}
|
|
|
|
nbytes += ret;
|
|
|
|
if (nbytes > NREAD_BYTES)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_reader: Too many bytes read -- aborting: %d\n", nbytes);
|
|
|
|
return (void*)4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("transfer_reader: %d bytes read\n", nbytes);
|
|
|
|
return (void*)0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: transfer_writer
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static void *transfer_writer(pthread_addr_t pvarg)
|
|
|
|
{
|
|
|
|
char buffer[WRITE_SIZE];
|
|
|
|
int fd = (int)pvarg;
|
|
|
|
int ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("transfer_writer: started\n");
|
|
|
|
for (i = 0; i < WRITE_SIZE; i++)
|
|
|
|
{
|
|
|
|
buffer[i] = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < NWRITES; i++)
|
|
|
|
{
|
|
|
|
ret = write(fd, buffer, WRITE_SIZE);
|
|
|
|
if (ret < 0 )
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_writer: write failed, errno=%d\n", errno);
|
|
|
|
return (void*)1;
|
|
|
|
}
|
|
|
|
else if (ret != WRITE_SIZE)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_writer: Unexpected write size=%d\n", ret);
|
|
|
|
return (void*)2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("transfer_writer: %d bytes written\n", NWRITE_BYTES);
|
|
|
|
return (void*)0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: transfer_test
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int transfer_test(int fdin, int fdout)
|
|
|
|
{
|
|
|
|
pthread_t readerid;
|
|
|
|
pthread_t writerid;
|
|
|
|
void *value;
|
|
|
|
int tmp;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Start transfer_reader thread */
|
|
|
|
|
|
|
|
printf("transfer_test: Starting transfer_reader thread\n");
|
|
|
|
ret = pthread_create(&readerid, NULL, transfer_reader, (pthread_addr_t)fdin);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_test: Failed to create transfer_reader thread, error=%d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start transfer_writer thread */
|
|
|
|
|
|
|
|
printf("transfer_test: Starting transfer_writer thread\n");
|
|
|
|
ret = pthread_create(&writerid, NULL, transfer_writer, (pthread_addr_t)fdout);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_test: Failed to create transfer_writer thread, error=%d\n", ret);
|
|
|
|
pthread_detach(readerid);
|
|
|
|
ret = pthread_cancel(readerid);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_test: Failed to cancel transfer_reader thread, error=%d\n", ret);
|
|
|
|
}
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for transfer_writer thread to complete */
|
|
|
|
|
|
|
|
printf("transfer_test: Waiting for transfer_writer thread\n");
|
|
|
|
ret = pthread_join(writerid, &value);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_test: pthread_join failed, error=%d\n", ret);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ret = (int)value;
|
|
|
|
printf("transfer_test: transfer_writer returned %d\n", ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for transfer_reader thread to complete */
|
|
|
|
|
|
|
|
printf("transfer_test: Waiting for transfer_reader thread\n");
|
|
|
|
tmp = pthread_join(readerid, &value);
|
|
|
|
if (tmp != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "transfer_test: pthread_join failed, error=%d\n", ret);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tmp = (int)value;
|
|
|
|
printf("transfer_test: transfer_reader returned %d\n", tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == 0)
|
|
|
|
{
|
|
|
|
ret = tmp;
|
|
|
|
}
|
|
|
|
printf("transfer_test: returning %d\n", ret);
|
|
|
|
return ret;
|
|
|
|
}
|