/****************************************************************************
 * apps/examples/pipe/interlock_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 <errno.h>
#include <fcntl.h>
#include <pthread.h>
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>

#include "pipe.h"

/****************************************************************************
 * Pre-processor Definitions
 ****************************************************************************/

/****************************************************************************
 * Private Types
 ****************************************************************************/

/****************************************************************************
 * Private Data
 ****************************************************************************/

/****************************************************************************
 * Private Functions
 ****************************************************************************/

/****************************************************************************
 * Name: null_reader
 ****************************************************************************/

static void *null_reader(pthread_addr_t pvarg)
{
  int fd;

  /* Wait a bit */

  printf("null_reader: started -- sleeping\n");
  sleep(5);

  /* Then open the FIFO for read access */

  printf("null_reader: Opening FIFO for read access\n");
  fd = open(FIFO_PATH2, O_RDONLY);
  if (fd < 0)
    {
      fprintf(stderr, \
              "null_reader: Failed to open FIFO %s for reading, errno=%d\n",
              FIFO_PATH2, errno);
      return (void *)(uintptr_t)1;
    }

  /* Wait a bit more */

  printf("null_reader: Opened %s for reading -- sleeping\n", FIFO_PATH2);
  sleep(5);

  /* Then close the FIFO */

  printf("null_reader: Closing %s\n", FIFO_PATH2);
  if (close(fd) != 0)
    {
      fprintf(stderr, "null_reader: close failed: %d\n", errno);
    }

  sleep(5);

  printf("null_reader: Returning success\n");
  return NULL;
}

/****************************************************************************
 * Name: null_writer
 ****************************************************************************/

static void *null_writer(pthread_addr_t pvarg)
{
  int fd;

  /* Wait a bit */

  printf("null_writer: started -- sleeping\n");
  sleep(5);

  /* Then open the FIFO for write access */

  printf("null_writer: Opening FIFO for write access\n");
  fd = open(FIFO_PATH2, O_WRONLY);
  if (fd < 0)
    {
      fprintf(stderr, \
              "null_writer: Failed to open FIFO %s for writing, errno=%d\n",
              FIFO_PATH2, errno);
      return (void *)(uintptr_t)1;
    }

  /* Wait a bit more */

  printf("null_writer: Opened %s for writing -- sleeping\n", FIFO_PATH2);
  sleep(5);

  /* Then close the FIFO */

  printf("null_writer: Closing %s\n", FIFO_PATH2);
  if (close(fd) != 0)
    {
      fprintf(stderr, "null_writer: close failed: %d\n", errno);
    }

  sleep(5);

  printf("null_writer: Returning success\n");
  return NULL;
}

/****************************************************************************
 * Public Functions
 ****************************************************************************/

/****************************************************************************
 * Name: interlock_test
 ****************************************************************************/

int interlock_test(void)
{
  pthread_t readerid;
  pthread_t writerid;
  void *value;
  char data[16];
  ssize_t nbytes;
  int fd;
  int ret;

  /* Create a FIFO */

  ret = mkfifo(FIFO_PATH2, 0666);
  if (ret < 0)
    {
      fprintf(stderr, \
              "interlock_test: mkfifo failed with errno=%d\n", errno);
      return 1;
    }

  /* Start the null_writer thread */

  printf("interlock_test: Starting null_writer thread\n");
  ret = pthread_create(&writerid, NULL, null_writer, NULL);
  if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: Failed to create null_writer thread,"
              "error=%d\n", ret);
      ret = 2;
      goto errout_with_fifo;
    }

  /* Open one end of the FIFO for reading. This open call should block until
   * the null_writer thread opens the other end of the FIFO for writing.
   */

  printf("interlock_test: Opening FIFO for read access\n");
  fd = open(FIFO_PATH2, O_RDONLY);
  if (fd < 0)
    {
      fprintf(stderr, \
              "interlock_test: Failed to open FIFO %s for reading"
              "errno=%d\n",
              FIFO_PATH2, errno);
      ret = 3;
      goto errout_with_null_writer_thread;
    }

  /* Attempt to read one byte from the FIFO. This should return end-of-file
   * because the null_writer closes the FIFO without writing anything.
   */

  printf("interlock_test: Reading from %s\n", FIFO_PATH2);
  nbytes = read(fd, data, 16);
  if (nbytes < 0)
    {
      fprintf(stderr, \
              "interlock_test: read failed, errno=%d\n", errno);
      ret = 4;
      goto errout_with_file;
    }
  else if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: Read %ld bytes of data -- aborting: %d\n",
              (long)nbytes, errno);
      ret = 5;
      goto errout_with_file;
    }

  printf("interlock_test: read returned\n");

  /* Close the file */

  printf("interlock_test: Closing %s\n", FIFO_PATH2);
  if (close(fd) != 0)
    {
      fprintf(stderr, "interlock_test: close failed: %d\n", errno);
    }

  /* Wait for null_writer thread to complete */

  printf("interlock_test: Waiting for null_writer thread\n");
  ret = pthread_join(writerid, &value);
  if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: pthread_join failed, error=%d\n", ret);
      ret = 6;
      goto errout_with_fifo;
    }
  else
    {
      printf("interlock_test: writer returned %p\n", value);
      if (value != NULL)
        {
          ret = 7;
          goto errout_with_fifo;
        }
    }

  /* Start the null_reader thread */

  printf("interlock_test: Starting null_reader thread\n");
  ret = pthread_create(&readerid, NULL, null_reader, NULL);
  if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: Failed to create null_reader thread,"
              "error=%d\n", ret);
      ret = 8;
      goto errout_with_fifo;
    }

  /* Open one end of the FIFO for writing. This open call should block until
   * the null_reader thread opens the other end of the FIFO for reading.
   */

  printf("interlock_test: Opening FIFO for write access\n");
  fd = open(FIFO_PATH2, O_WRONLY);
  if (fd < 0)
    {
      fprintf(stderr, \
              "interlock_test: Failed to open FIFO %s for writing,"
              "errno=%d\n",
              FIFO_PATH2, errno);
      ret = 9;
      goto errout_with_null_reader_thread;
    }

  /* Attempt to write one byte from the FIFO. This should return 0 bytes
   * written because the null_reader closes the FIFO.
   */

  printf("interlock_test: Writing to %s\n", FIFO_PATH2);
  nbytes = write(fd, data, 16);
  if (nbytes < 0)
    {
      fprintf(stderr, \
              "interlock_test: write failed, errno=%d\n", errno);
      ret = 10;
      goto errout_with_file;
    }
  else if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: Wrote %ld bytes of data -- aborting: %d\n",
              (long)nbytes, errno);
      ret = 11;
      goto errout_with_file;
    }

  printf("interlock_test: write returned\n");

  /* Close the file */

  printf("interlock_test: Closing %s\n", FIFO_PATH2);
  if (close(fd) != 0)
    {
      fprintf(stderr, "interlock_test: close failed: %d\n", errno);
    }

  /* Wait for null_reader thread to complete */

  printf("interlock_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 = 12;
      goto errout_with_fifo;
    }
  else
    {
      printf("interlock_test: reader returned %p\n", value);
      if (value != NULL)
        {
          ret = 13;
          goto errout_with_fifo;
        }
    }

  ret = 0;
  goto errout_with_fifo;

errout_with_file:
  if (close(fd) != 0)
    {
      fprintf(stderr, "interlock_test: close failed: %d\n", errno);
    }

errout_with_null_reader_thread:
  pthread_detach(readerid);
  pthread_cancel(readerid);

errout_with_null_writer_thread:
  pthread_detach(writerid);
  pthread_cancel(writerid);

errout_with_fifo:

  ret = remove(FIFO_PATH2);
  if (ret != 0)
    {
      fprintf(stderr, \
              "interlock_test: remove failed with errno=%d\n", errno);
    }

  printf("interlock_test: Returning %d\n", ret);
  return ret;
}