/****************************************************************************
 * libs/libc/termios/lib_tcsendbreak.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 <sys/ioctl.h>

#include <termios.h>
#include <errno.h>

#include <nuttx/serial/tioctl.h>

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

/****************************************************************************
 * Name: tcsendbreak
 *
 * Description:
 *   Transmit a serial line Break, which is a continuous stream of 0 bits.
 *
 *   A proper Break needs to last longer than a NUL ('\0') character.
 *   Effectively this causes a Framing Error at the receiver's UART. Most
 *   UARTs have the ability to distinguish between a Break and other types
 *   of Framing Errors and may treat it specially. Some devices utilize
 *   Breaks for things like synchronization.
 *
 *   Regarding the duration parameter and its portability between Unix-like
 *   systems: It seems that a duration parameter of 0 specifies a Break of
 *   0.25 to 0.5 seconds with consistency between various Unix flavors, but
 *   the interpretation of a nonzero duration parameter varies wildly from
 *   one operating system to another:
 *
 *     - Linux, AIX, Tru64: duration in milliseconds
 *     - BSDs, macOS, HP-UX: ignore duration
 *     - Solaris, UnixWare: behaves like tcdrain()
 *
 *   On FreeBSD and OpenBSD, a duration of 0 gives a Break of 0.4 seconds.
 *
 *   Sources of information about the duration parameter include: the man
 *   page for termios(3) as found in release 5.13 of the Linux man-pages
 *   project, the man page for tcsendbreak(3) as found in FreeBSD 13.1 and
 *   OpenBSD 7.2, and some StackOverflow search results.
 *
 *   In NuttX, we combine the above treatments of duration: Zero or negative
 *   values will give a Break of 0.4 seconds; positive values will be
 *   treated as a duration in milliseconds. We hope this combination will
 *   work well for programs that are ported from any of the above families.
 *
 *   NuttX-specific implementation details: tcsendbreak() calls IOCTL
 *   TCSBRK. This is handled in the upper half serial driver (see
 *   drivers/serial/serial.c), which treats its argument like 'duration'
 *   described here, except unsigned, supporting only 0 or positive values;
 *   0 maps to 400 and all other values are used as-is, in milliseconds.
 *   This, in turn, calls IOCTLs TIOCSBRK and TIOCCBRK to start and end a
 *   BSD-compatible Break, respectively, with a sleep of the given duration
 *   between them. For tcsendbreak() to work, the arch-specific lower half
 *   serial driver must implement TIOCSBRK and TIOCCBRK. Some architectures
 *   do not implement these at all. Others implement them but rely upon one
 *   or more Kconfig options being set, such as CONFIG_*_U[S]ART_BREAKS and,
 *   on some architectures, a separate CONFIG_*_SERIALBRK_BSDCOMPAT, to
 *   provide the required Break behavior. Furthermore, the sleep time will
 *   be at least the given duration, but may be substantially longer
 *   depending on the system's tick resolution (see CONFIG_USEC_PER_TICK)
 *   and other factors.
 *
 *   This function blocks for the duration of the Break.
 *
 * Input Parameters:
 *   fd       - The 'fd' argument is an open file descriptor associated
 *              with a terminal.
 *   duration - If 0 or negative, request a serial line Break lasting 0.4
 *              seconds. If non-zero, request a serial line Break lasting
 *              that duration in milliseconds.
 *
 * Returned Value:
 *   Upon successful completion, 0 is returned. Otherwise, -1 is returned
 *   and errno is set to indicate the error.
 *
 ****************************************************************************/

int tcsendbreak(int fd, int duration)
{
  if (duration <= 0)
    {
      duration = 400;
    }

  return ioctl(fd, TCSBRK, (unsigned long)duration);
}