summaryrefslogblamecommitdiffstats
path: root/cpukit/libcsupport/include/rtems/termiostypes.h
blob: df3ec0c2d7c7cb9292a9a20f6c25facc2f08c1a5 (plain) (tree)
1
2
3
4
5
6
7
8

                             

                                                        

   
  
                            



                                                           
                                         

   

                         


                        
                        
                        
                   
                    




                  







                                                                
  


                                 
                                                      



                    


                                                 
               



                              
  
























                                                                             
                                                                             




































































































                                                                             



                                                   
                                  














                                               
                                   













































                                                                





                                       




















                                                                 

                                      






























































































                                                                               
 












                                                                               
                             










                                                                              


                                                                
 



                                        

                                
                                


                               


                                    
                                                          






                                                        
 
                               
                                      
 


                                   
                                                       
 
   

                                                                               
  

                                                       



                                                                 
                                                                               
  
                                                          
  

                                                          

                                                                    
 


                                          
                                                                    
 




                                                                 
   


                                   
  
 





                                                           



                  
                            
/**
 * @file rtems/termiostypes.h
 *
 * RTEMS termios device support internal data structures
 */

/*
 *  COPYRIGHT (c) 1989-2011.
 *  On-Line Applications Research Corporation (OAR).
 *
 *  The license and distribution terms for this file may be
 *  found in the file LICENSE in this distribution or at
 *  http://www.rtems.org/license/LICENSE.
 */

#ifndef  __TERMIOSTYPES_H
#define  __TERMIOSTYPES_H

#include <rtems.h>
#include <rtems/libio.h>
#include <rtems/assoc.h>
#include <rtems/chain.h>
#include <stdint.h>
#include <termios.h>

#ifdef __cplusplus
extern "C" {
#endif

/**
 *  @defgroup TermiostypesSupport RTEMS Termios Device Support
 *
 *  @ingroup libcsupport
 *
 *  @brief RTEMS Termios Device Support Internal Data Structures
 */

/*
 * Wakeup callback data structure
 */
struct ttywakeup {
  void      (*sw_pfn)(struct termios *tty, void *arg);
  void      *sw_arg;
};

/*
 * Variables associated with the character buffer
 */
struct rtems_termios_rawbuf {
  char *theBuf;
  volatile unsigned int  Head;
  volatile unsigned int  Tail;
  volatile unsigned int  Size;
  rtems_id    Semaphore;
};

typedef enum {
  TERMIOS_POLLED,
  TERMIOS_IRQ_DRIVEN,
  TERMIOS_TASK_DRIVEN
} rtems_termios_device_mode;

struct rtems_termios_tty;

/**
 * @brief Termios device handler.
 *
 * @see rtems_termios_device_install().
 */
typedef struct {
  /**
   * @brief First open of this device.
   *
   * @param[in] tty The Termios control.
   * @param[in] args The open/close arguments.  This is parameter provided to
   *   support legacy drivers.  It must not be used by new drivers.
   *
   * @retval true Successful operation.
   * @retval false Cannot open device.
   *
   * @see rtems_termios_get_device_context() and rtems_termios_get_termios().
   */
  bool (*first_open)(
    struct rtems_termios_tty      *tty,
    rtems_libio_open_close_args_t *args
  );

  /**
   * @brief Last close of this device.
   *
   * @param[in] tty The Termios control.
   * @param[in] args The open/close arguments.  This is parameter provided to
   *   support legacy drivers.  It must not be used by new drivers.
   *
   * @see rtems_termios_get_device_context().
   */
  void (*last_close)(
    struct rtems_termios_tty      *tty,
    rtems_libio_open_close_args_t *args
  );

  /**
   * @brief Polled read.
   *
   * In case mode is TERMIOS_IRQ_DRIVEN or TERMIOS_TASK_DRIVEN, then data is
   * received via rtems_termios_enqueue_raw_characters().
   *
   * @param[in] tty The Termios control.
   *
   * @retval char The received data encoded as unsigned char.
   * @retval -1 No data currently available.
   *
   * @see rtems_termios_get_device_context().
   */
  int (*poll_read)(struct rtems_termios_tty *tty);

  /**
   * @brief Polled write in case mode is TERMIOS_POLLED or write support
   * otherwise.
   *
   * @param[in] tty The Termios control.
   * @param[in] buf The output buffer.
   * @param[in] len The output buffer length in characters.
   *
   * @see rtems_termios_get_device_context().
   */
  void (*write)(struct rtems_termios_tty *tty, const char *buf, size_t len);

  /**
   * @brief Set attributes after a Termios settings change.
   *
   * @param[in] tty The Termios control.
   * @param[in] term The new Termios attributes.
   *
   * @retval true Successful operation.
   * @retval false Invalid attributes.
   *
   * @see rtems_termios_get_device_context().
   */
  bool (*set_attributes)(
    struct rtems_termios_tty *tty,
    const struct termios     *term
  );

  /**
   * @brief Indicate to stop remote transmitter.
   *
   * @param[in] tty The Termios control.
   *
   * @see rtems_termios_get_device_context().
   */
  void (*stop_remote_tx)(struct rtems_termios_tty *tty);

  /**
   * @brief Indicate to start remote transmitter.
   *
   * @param[in] tty The Termios control.
   *
   * @see rtems_termios_get_device_context().
   */
  void (*start_remote_tx)(struct rtems_termios_tty *tty);

  /**
   * @brief Termios device mode.
   */
  rtems_termios_device_mode mode;
} rtems_termios_device_handler;

/**
 * @brief Termios device node for installed devices.
 *
 * @see rtems_termios_device_install().
 */
typedef struct rtems_termios_device_node {
  rtems_chain_node                    node;
  rtems_device_major_number           major;
  rtems_device_minor_number           minor;
  const rtems_termios_device_handler *handler;
  void                               *context;
  struct rtems_termios_tty           *tty;
} rtems_termios_device_node;

/*
 * Variables associated with each termios instance.
 * One structure for each hardware I/O device.
 */
typedef struct rtems_termios_tty {
  /*
   * Linked-list of active TERMIOS devices
   */
  struct rtems_termios_tty  *forw;
  struct rtems_termios_tty  *back;

  /*
   * How many times has this device been opened
   */
  int    refcount;

  /*
   * This device
   */
  rtems_device_major_number  major;
  rtems_device_minor_number  minor;

  /*
   * Mutual-exclusion semaphores
   */
  rtems_id  isem;
  rtems_id  osem;

  /*
   * The canonical (cooked) character buffer
   */
  char    *cbuf;
  int    ccount;
  int    cindex;

  /*
   * Keep track of cursor (printhead) position
   */
  int    column;
  int    read_start_column;

  /*
   * The ioctl settings
   */
  struct termios  termios;
  rtems_interval  vtimeTicks;

  /*
   * Raw input character buffer
   */
  struct rtems_termios_rawbuf rawInBuf;
  uint32_t                    rawInBufSemaphoreOptions;
  rtems_interval              rawInBufSemaphoreTimeout;
  rtems_interval              rawInBufSemaphoreFirstTimeout;
  unsigned int                rawInBufDropped;  /* Statistics */

  /*
   * Raw output character buffer
   */
  struct rtems_termios_rawbuf rawOutBuf;
  int  t_dqlen; /* count of characters dequeued from device */
  enum {rob_idle, rob_busy, rob_wait }  rawOutBufState;

  /*
   * Callbacks to device-specific routines
   */
  rtems_termios_callbacks  device;

  /**
   * @brief The device handler.
   */
  rtems_termios_device_handler handler;

  volatile unsigned int    flow_ctrl;
  unsigned int             lowwater,highwater;

  /*
   * I/O task IDs (for task-driven drivers)
   */
  rtems_id                rxTaskId;
  rtems_id                txTaskId;

  /*
   * line discipline related stuff
   */
  int t_line;   /* id of line discipline                       */
  void *t_sc;   /* hook for discipline-specific data structure */

  /*
   * Wakeup callback variables
   */
  struct ttywakeup tty_snd;
  struct ttywakeup tty_rcv;
  int              tty_rcvwakeup;

  rtems_interrupt_lock interrupt_lock;

  /**
   * @brief Corresponding device node.
   */
  rtems_termios_device_node *device_node;

  /**
   * @brief Context for device driver.
   *
   * @see rtems_termios_get_device_context().
   */
  void *device_context;
} rtems_termios_tty;

/**
 * @brief Installes a Termios device.
 *
 * @param[in] device_file If not @c NULL, then a device file for the specified
 * major and minor number will be created.
 * @param[in] major The device major number of the corresponding device driver.
 * @param[in] minor The device minor number of the corresponding device driver.
 * @param[in] handler The device handler.  It must be persistent throughout the
 *   installed time of the device.
 * @param[in] context The device context.  It must be persistent throughout the
 *   installed time of the device.
 *
 * @retval RTEMS_SUCCESSFUL Successful operation.
 * @retval RTEMS_NO_MEMORY Not enough memory to create a device node.
 * @retval RTEMS_UNSATISFIED Creation of the device file failed.
 * @retval RTEMS_RESOURCE_IN_USE There exists a device node for this major and
 * minor number pair.
 * @retval RTEMS_INCORRECT_STATE Termios is not initialized.
 *
 * @see rtems_termios_device_remove(), rtems_termios_device_open(),
 * rtems_termios_device_close() and rtems_termios_get_device_context().
 */
rtems_status_code rtems_termios_device_install(
  const char                         *device_file,
  rtems_device_major_number           major,
  rtems_device_minor_number           minor,
  const rtems_termios_device_handler *handler,
  void                               *context
);

/**
 * @brief Removes a Termios device.
 *
 * @param[in] device_file If not @c NULL, then the device file to remove.
 * @param[in] major The device major number of the corresponding device driver.
 * @param[in] minor The device minor number of the corresponding device driver.
 *
 * @retval RTEMS_SUCCESSFUL Successful operation.
 * @retval RTEMS_INVALID_ID There is no device installed with this major and
 * minor number pair.
 * @retval RTEMS_RESOURCE_IN_USE This device is currently in use.
 * @retval RTEMS_UNSATISFIED Removal of the device file failed.
 * @retval RTEMS_INCORRECT_STATE Termios is not initialized.
 *
 * @see rtems_termios_device_install().
 */
rtems_status_code rtems_termios_device_remove(
  const char                *device_file,
  rtems_device_major_number  major,
  rtems_device_minor_number  minor
);

/**
 * @brief Opens an installed Termios device.
 *
 * @see rtems_termios_device_install().
 */
rtems_status_code rtems_termios_device_open(
  rtems_device_major_number  major,
  rtems_device_minor_number  minor,
  void                      *arg
);

/**
 * @brief Closes an installed Termios device.
 *
 * @retval RTEMS_SUCCESSFUL Successful operation.
 *
 * @see rtems_termios_device_install().
 */
rtems_status_code rtems_termios_device_close(void *arg);

/**
 * @brief Returns the device context of an installed Termios device.
 */
RTEMS_INLINE_ROUTINE void *rtems_termios_get_device_context(
  const rtems_termios_tty *tty
)
{
  return tty->device_context;
}

/**
 * @brief Returns the Termios structure.
 *
 * It can be used for example in the first open handler to adjust or obtain the
 * initial attributes.
 */
RTEMS_INLINE_ROUTINE struct termios *rtems_termios_get_termios(
  const rtems_termios_tty *tty
)
{
  return &tty->termios;
}

struct rtems_termios_linesw {
  int (*l_open) (struct rtems_termios_tty *tp);
  int (*l_close)(struct rtems_termios_tty *tp);
  int (*l_read )(struct rtems_termios_tty *tp,rtems_libio_rw_args_t *args);
  int (*l_write)(struct rtems_termios_tty *tp,rtems_libio_rw_args_t *args);
  int (*l_rint )(int c,struct rtems_termios_tty *tp);
  int (*l_start)(struct rtems_termios_tty *tp);
  int (*l_ioctl)(struct rtems_termios_tty *tp,rtems_libio_ioctl_args_t *args);
  int (*l_modem)(struct rtems_termios_tty *tp,int flags);
};

/*
 * FIXME: this should move to termios.h!
 */
void rtems_termios_rxirq_occured(struct rtems_termios_tty *tty);

/*
 * FIXME: this should move to termios.h!
 * put a string to output ring buffer
 */
void rtems_termios_puts (
  const void               *buf,
  size_t                    len,
  struct rtems_termios_tty *tty
);

/*
 * global hooks for line disciplines
 */
extern struct rtems_termios_linesw rtems_termios_linesw[];
extern int   rtems_termios_nlinesw;

#define TTYDISC   0    /* termios tty line discipline */
#define TABLDISC  3    /* tablet discipline */
#define SLIPDISC  4    /* serial IP discipline */
#define PPPDISC   5    /* PPP discipline */
#define MAXLDISC  8

/* baudrate xxx integer type */
typedef uint32_t rtems_termios_baud_t;

/**
 *  @brief RTEMS Termios Baud Table
 */
extern const rtems_assoc_t rtems_termios_baud_table [];

/**
 *  @brief Converts the Integral Baud value @a baud to the Termios Control Flag
 *  Representation
 *
 *  @retval B0 Invalid baud value or a baud value of 0.
 *  @retval other Baud constant according to @a baud.
 */
tcflag_t rtems_termios_number_to_baud(rtems_termios_baud_t baud);

/**
 *  @brief Convert Baud Part of Termios control flags to an integral Baud Value
 *
 *  There is no need to mask the @a c_cflag with @c CBAUD.
 *
 *  @retval 0 Invalid baud value or a baud value of @c B0.
 *  @retval other Integral baud value.
 */
rtems_termios_baud_t rtems_termios_baud_to_number(tcflag_t c_cflag);

/** 
 *  @brief Convert Bxxx Constant to Index 
 */
int  rtems_termios_baud_to_index(rtems_termios_baud_t termios_baud);

/**
 * @brief Sets the initial @a baud in the Termios context @a tty.
 *
 * @retval 0 Successful operation.
 * @retval -1 Invalid baud value.
 */
int rtems_termios_set_initial_baud(
  struct rtems_termios_tty *tty,
  rtems_termios_baud_t baud
);

#define rtems_termios_interrupt_lock_acquire(tty, level) \
  rtems_interrupt_lock_acquire(&tty->interrupt_lock, level)

#define rtems_termios_interrupt_lock_release(tty, level) \
  rtems_interrupt_lock_release(&tty->interrupt_lock, level)

#ifdef __cplusplus
}
#endif

#endif  /* TERMIOSTYPES_H */