2011-03-20 19:18:19 +01:00
|
|
|
/****************************************************************************
|
2021-06-16 09:22:16 +02:00
|
|
|
* apps/examples/pipe/redirect_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>
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
#include <stdint.h>
|
2011-03-20 19:18:19 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sched.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "pipe.h"
|
|
|
|
|
|
|
|
/****************************************************************************
|
2015-10-02 22:06:11 +02:00
|
|
|
* Pre-processor Definitions
|
2011-03-20 19:18:19 +01:00
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#define READ_SIZE 37
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: redirect_reader
|
|
|
|
****************************************************************************/
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
static void *redirect_reader(pthread_addr_t pvarg)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
|
|
|
char buffer[READ_SIZE];
|
2023-03-27 00:19:48 +02:00
|
|
|
int *fd = (int *)pvarg;
|
2011-03-20 19:18:19 +01:00
|
|
|
int fdin;
|
|
|
|
int ret;
|
|
|
|
int nbytes = 0;
|
2013-09-29 00:50:07 +02:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_reader: started with fdin=%d\n", fd[0]);
|
2011-03-20 19:18:19 +01:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fdin = fd[0];
|
2011-03-20 19:18:19 +01:00
|
|
|
|
|
|
|
/* Re-direct the fdin to stdin */
|
|
|
|
|
|
|
|
ret = dup2(fdin, 0);
|
2023-03-27 00:19:48 +02:00
|
|
|
if (ret < 0)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirect_reader: dup2 failed: %d\n", errno);
|
|
|
|
close(fdin);
|
2023-03-27 00:19:48 +02:00
|
|
|
return (void *)(uintptr_t)1;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Then read from stdin until we hit the end of file */
|
|
|
|
|
2022-10-08 10:31:21 +02:00
|
|
|
for (; ; )
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
|
|
|
/* Read from stdin */
|
|
|
|
|
|
|
|
ret = read(0, buffer, READ_SIZE);
|
2022-10-08 10:31:21 +02:00
|
|
|
if (ret < 0)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_reader: read failed, errno=%d\n", errno);
|
|
|
|
return (void *)(uintptr_t)2;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
else if (ret == 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2017-11-29 22:27:20 +01:00
|
|
|
|
2011-03-20 19:18:19 +01:00
|
|
|
nbytes += ret;
|
|
|
|
|
|
|
|
/* Echo to stdout */
|
2013-09-29 00:50:07 +02:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
ret = write(2, buffer, ret);
|
2011-03-20 19:18:19 +01:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_reader: write failed, errno=%d\n",
|
2022-10-08 10:31:21 +02:00
|
|
|
errno);
|
2023-03-27 00:19:48 +02:00
|
|
|
return (void *)(uintptr_t)3;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_reader: %d bytes read\n", nbytes);
|
2011-03-20 19:18:19 +01:00
|
|
|
ret = close(0);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirect_reader: failed to close fd=0\n");
|
2023-03-27 00:19:48 +02:00
|
|
|
return (void *)(uintptr_t)4;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_reader: Returning success\n");
|
|
|
|
return NULL;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: redirect_writer
|
|
|
|
****************************************************************************/
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
static void *redirect_writer(pthread_addr_t pvarg)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
int *fd = (int *)pvarg;
|
2011-03-20 19:18:19 +01:00
|
|
|
int fdout;
|
|
|
|
int nbytes = 0;
|
|
|
|
int ret;
|
2013-09-29 00:50:07 +02:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_writer: started with fdout=%d\n", fd[1]);
|
2013-09-29 00:50:07 +02:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fdout = fd[1];
|
2011-03-20 19:18:19 +01:00
|
|
|
|
|
|
|
/* Re-direct the fdout to stdout */
|
|
|
|
|
|
|
|
ret = dup2(fdout, 1);
|
2023-03-27 00:19:48 +02:00
|
|
|
if (ret < 0)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirect_writer: dup2 failed: %d\n", ret);
|
|
|
|
return (void *)(uintptr_t)1;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Then write a bunch of stuff to stdout */
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
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");
|
2011-03-20 19:18:19 +01:00
|
|
|
nbytes += printf("should do this.\n");
|
2023-03-27 00:19:48 +02:00
|
|
|
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");
|
2011-03-20 19:18:19 +01:00
|
|
|
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)
|
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirect_writer: failed to close fd=1\n");
|
2023-03-27 00:19:48 +02:00
|
|
|
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;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "redirect_writer: Returning success\n");
|
2023-03-27 00:19:48 +02:00
|
|
|
return NULL;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: redirection_test
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int redirection_test(void)
|
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
pthread_t readerid;
|
|
|
|
pthread_t writerid;
|
|
|
|
void *value;
|
2013-09-29 00:50:07 +02:00
|
|
|
int fd[2];
|
2011-03-20 19:18:19 +01:00
|
|
|
int ret;
|
|
|
|
|
2022-10-08 10:31:21 +02:00
|
|
|
/* Create the pipe */
|
2011-03-20 19:18:19 +01:00
|
|
|
|
2013-09-29 00:50:07 +02:00
|
|
|
ret = pipe(fd);
|
2011-03-20 19:18:19 +01:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirection_test: pipe failed with errno=%d\n",
|
|
|
|
errno);
|
2011-03-20 19:18:19 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start redirect_writer task */
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
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)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirection_test: "
|
|
|
|
"Failed to create redirect_writer task: %d\n", errno);
|
2023-03-27 00:19:48 +02:00
|
|
|
ret = pthread_cancel(writerid);
|
2011-03-20 19:18:19 +01:00
|
|
|
if (ret != 0)
|
|
|
|
{
|
2022-10-08 10:31:21 +02:00
|
|
|
fprintf(stderr, "redirection_test: "
|
|
|
|
"Failed to delete redirect_reader task %d\n", errno);
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
2017-11-29 22:27:20 +01:00
|
|
|
|
2011-03-20 19:18:19 +01:00
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
/* Start redirect_reader thread */
|
2011-03-20 19:18:19 +01:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
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)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirection_test: "
|
|
|
|
"Failed to create redirect_writer task: %d\n", ret);
|
|
|
|
return 3;
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
2017-11-29 22:27:20 +01:00
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
/* 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)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
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;
|
|
|
|
}
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
/* We should be able to close the pipe file descriptors now. */
|
|
|
|
|
|
|
|
if (close(fd[0]) != 0)
|
2011-03-20 19:18:19 +01:00
|
|
|
{
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "pipe_main: close failed: %d\n", errno);
|
2011-03-20 19:18:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for redirect_writer thread to complete */
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirection_test: Waiting...\n");
|
2011-03-20 19:18:19 +01:00
|
|
|
fflush(stdout);
|
|
|
|
|
2023-03-27 00:19:48 +02:00
|
|
|
fprintf(stderr, "redirection_test: returning %d\n", ret);
|
2011-03-20 19:18:19 +01:00
|
|
|
return ret;
|
|
|
|
}
|