nuttx-apps/examples/pipe/redirect_test.c
Tiago Medicci Serrano 0a9b983e90 apps/examples/pipe: check POSIX-compliant behavior of NuttX's FIFOs
Adjust the test considering the POSIX-compliant behavior of the
NuttX's FIFO (mkfifo), which should block `open` for read-only
and write-only. This test's result is expected to be the same with
any other POSIX-compliant system.

This commit also:

 * Fix redirect test;
 * Use pthread instead `task_create` to be able to run this test
on POSIX-compliant systems;
 * General fixes regarding formatting and error messages;
2023-04-22 01:28:25 +08:00

289 lines
8.8 KiB
C

/****************************************************************************
* apps/examples/pipe/redirect_test.c
*
* 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 <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sched.h>
#include <errno.h>
#include "pipe.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define READ_SIZE 37
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: redirect_reader
****************************************************************************/
static void *redirect_reader(pthread_addr_t pvarg)
{
char buffer[READ_SIZE];
int *fd = (int *)pvarg;
int fdin;
int ret;
int nbytes = 0;
fprintf(stderr, "redirect_reader: started with fdin=%d\n", fd[0]);
fdin = fd[0];
/* Re-direct the fdin to stdin */
ret = dup2(fdin, 0);
if (ret < 0)
{
fprintf(stderr, "redirect_reader: dup2 failed: %d\n", errno);
close(fdin);
return (void *)(uintptr_t)1;
}
/* Then read from stdin until we hit the end of file */
for (; ; )
{
/* Read from stdin */
ret = read(0, buffer, READ_SIZE);
if (ret < 0)
{
fprintf(stderr, "redirect_reader: read failed, errno=%d\n", errno);
return (void *)(uintptr_t)2;
}
else if (ret == 0)
{
break;
}
nbytes += ret;
/* Echo to stdout */
ret = write(2, buffer, ret);
if (ret < 0)
{
fprintf(stderr, "redirect_reader: write failed, errno=%d\n",
errno);
return (void *)(uintptr_t)3;
}
}
fprintf(stderr, "redirect_reader: %d bytes read\n", nbytes);
ret = close(0);
if (ret != 0)
{
fprintf(stderr, "redirect_reader: failed to close fd=0\n");
return (void *)(uintptr_t)4;
}
fprintf(stderr, "redirect_reader: Returning success\n");
return NULL;
}
/****************************************************************************
* Name: redirect_writer
****************************************************************************/
static void *redirect_writer(pthread_addr_t pvarg)
{
int *fd = (int *)pvarg;
int fdout;
int nbytes = 0;
int ret;
fprintf(stderr, "redirect_writer: started with fdout=%d\n", fd[1]);
fdout = fd[1];
/* Re-direct the fdout to stdout */
ret = dup2(fdout, 1);
if (ret < 0)
{
fprintf(stderr, "redirect_writer: dup2 failed: %d\n", ret);
return (void *)(uintptr_t)1;
}
/* Then write a bunch of stuff to stdout */
fflush(stdout);
nbytes += printf("\nFour score and seven years ago our fathers brought"
"forth on this continent a new nation,\n");
nbytes += printf("conceived in Liberty, and dedicated to the proposition"
"that all men are created equal.\n");
nbytes += printf("\nNow we are engaged in a great civil war, testing"
"whether that nation, or any nation, so\n");
nbytes += printf("conceived and so dedicated, can long endure. We are met"
"on a great battle-field of that war.\n");
nbytes += printf("We have come to dedicate a portion of that field, as a"
"final resting place for those who here\n");
nbytes += printf("gave their lives that that nation might live. It is"
"altogether fitting and proper that we\n");
nbytes += printf("should do this.\n");
nbytes += printf("\nBut, in a larger sense, we can not dedicate - we can"
"not consecrate - we can not hallow - this ground.\n");
nbytes += printf("The brave men, living and dead, who struggled here, have"
"consecrated it, far above our poor power\n");
nbytes += printf("to add or detract. The world will little note, nor long"
"remember what we say here, but it can\n");
nbytes += printf("never forget what they did here. It is for us the"
"living, rather, to be dedicated here to the\n");
nbytes += printf("unfinished work which they who fought here have thus far"
"so nobly advanced. It is rather for us to\n");
nbytes += printf("be here dedicated to the great task remaining before us"
"- that from these honored dead we take\n");
nbytes += printf("increased devotion to that cause for which they gave the"
"last full measure of devotion - that we\n");
nbytes += printf("here highly resolve that these dead shall not have died"
"in vain - that this nation, under God,\n");
nbytes += printf("shall have a new birth of freedom - and that government"
"of the people, by the people, for the\n");
nbytes += printf("people, shall not perish from the earth.\n\n");
fflush(stdout);
fprintf(stderr, "redirect_writer: %d bytes written\n", nbytes);
ret = close(1);
if (ret != 0)
{
fprintf(stderr, "redirect_writer: failed to close fd=1\n");
return (void *)(uintptr_t)2;
}
ret = close(fdout);
if (ret != 0)
{
fprintf(stderr, "redirect_writer: failed to close fdout\n");
return (void *)(uintptr_t)4;
}
fprintf(stderr, "redirect_writer: Returning success\n");
return NULL;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: redirection_test
****************************************************************************/
int redirection_test(void)
{
pthread_t readerid;
pthread_t writerid;
void *value;
int fd[2];
int ret;
/* Create the pipe */
ret = pipe(fd);
if (ret < 0)
{
fprintf(stderr, "redirection_test: pipe failed with errno=%d\n",
errno);
return 1;
}
/* Start redirect_writer task */
fprintf(stderr,
"redirection_test: Starting redirect_writer task with fd=%d\n",
fd[1]);
ret = pthread_create(&writerid, NULL, redirect_writer, fd);
if (ret < 0)
{
fprintf(stderr, "redirection_test: "
"Failed to create redirect_writer task: %d\n", errno);
ret = pthread_cancel(writerid);
if (ret != 0)
{
fprintf(stderr, "redirection_test: "
"Failed to delete redirect_reader task %d\n", errno);
}
return 2;
}
/* Start redirect_reader thread */
fprintf(stderr,
"redirection_test: Starting redirect_reader task with fd=%d\n",
fd[0]);
ret = pthread_create(&readerid, NULL, redirect_reader, fd);
if (ret < 0)
{
fprintf(stderr, "redirection_test: "
"Failed to create redirect_writer task: %d\n", ret);
return 3;
}
/* Wait for redirect_reader thread to complete */
fprintf(stderr, "redirection_test: Waiting for null_reader thread\n");
ret = pthread_join(readerid, &value);
if (ret != 0)
{
fprintf(stderr, \
"interlock_test: pthread_join failed, error=%d\n", ret);
ret = 4;
}
else
{
printf("interlock_test: reader returned %p\n", value);
if (value != NULL)
{
ret = 5;
}
}
/* We should be able to close the pipe file descriptors now. */
if (close(fd[0]) != 0)
{
fprintf(stderr, "pipe_main: close failed: %d\n", errno);
}
/* Wait for redirect_writer thread to complete */
fprintf(stderr, "redirection_test: Waiting...\n");
fflush(stdout);
fprintf(stderr, "redirection_test: returning %d\n", ret);
return ret;
}