/* serial.h - serial device interface */
/*
* VasEBoot -- GRand Unified Bootloader
* Copyright (C) 2010 Free Software Foundation, Inc.
*
* VasEBoot is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* VasEBoot is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with VasEBoot. If not, see .
*/
#ifndef VasEBoot_SERIAL_HEADER
#define VasEBoot_SERIAL_HEADER 1
#include
#if defined(__mips__) || defined (__i386__) || defined (__x86_64__)
#include
#endif
#include
#include
#include
#ifdef VasEBoot_MACHINE_IEEE1275
#include
#endif
#ifdef VasEBoot_MACHINE_ARC
#include
#endif
struct VasEBoot_serial_port;
struct VasEBoot_serial_config;
struct VasEBoot_serial_driver
{
VasEBoot_err_t (*configure) (struct VasEBoot_serial_port *port,
struct VasEBoot_serial_config *config);
int (*fetch) (struct VasEBoot_serial_port *port);
void (*put) (struct VasEBoot_serial_port *port, const int c);
void (*fini) (struct VasEBoot_serial_port *port);
};
/* The type of parity. */
typedef enum
{
VasEBoot_SERIAL_PARITY_NONE,
VasEBoot_SERIAL_PARITY_ODD,
VasEBoot_SERIAL_PARITY_EVEN,
} VasEBoot_serial_parity_t;
typedef enum
{
VasEBoot_SERIAL_STOP_BITS_1,
VasEBoot_SERIAL_STOP_BITS_1_5,
VasEBoot_SERIAL_STOP_BITS_2,
} VasEBoot_serial_stop_bits_t;
struct VasEBoot_serial_config
{
unsigned speed;
int word_len;
VasEBoot_serial_parity_t parity;
VasEBoot_serial_stop_bits_t stop_bits;
VasEBoot_uint64_t base_clock;
int rtscts;
};
struct VasEBoot_serial_port
{
struct VasEBoot_serial_port *next;
struct VasEBoot_serial_port **prev;
char *name;
struct VasEBoot_serial_driver *driver;
struct VasEBoot_serial_config config;
int configured;
int broken;
/* This should be void *data but since serial is useful as an early console
when malloc isn't available it's a union.
*/
union
{
#if defined(__mips__) || defined (__i386__) || defined (__x86_64__)
VasEBoot_port_t port;
#endif
struct
{
VasEBoot_usb_device_t usbdev;
int configno;
int interfno;
char buf[64];
int bufstart, bufend;
struct VasEBoot_usb_desc_endp *in_endp;
struct VasEBoot_usb_desc_endp *out_endp;
};
struct VasEBoot_escc_descriptor *escc_desc;
#ifdef VasEBoot_MACHINE_IEEE1275
struct
{
VasEBoot_ieee1275_ihandle_t handle;
struct ofserial_hash_ent *elem;
};
#endif
#ifdef VasEBoot_MACHINE_EFI
struct VasEBoot_efi_serial_io_interface *interface;
#endif
#ifdef VasEBoot_MACHINE_ARC
struct
{
VasEBoot_arc_fileno_t handle;
int handle_valid;
};
#endif
};
VasEBoot_term_output_t term_out;
VasEBoot_term_input_t term_in;
};
VasEBoot_err_t EXPORT_FUNC(VasEBoot_serial_register) (struct VasEBoot_serial_port *port);
void EXPORT_FUNC(VasEBoot_serial_unregister) (struct VasEBoot_serial_port *port);
/* Convenience functions to perform primitive operations on a port. */
static inline VasEBoot_err_t
VasEBoot_serial_port_configure (struct VasEBoot_serial_port *port,
struct VasEBoot_serial_config *config)
{
return port->driver->configure (port, config);
}
static inline int
VasEBoot_serial_port_fetch (struct VasEBoot_serial_port *port)
{
return port->driver->fetch (port);
}
static inline void
VasEBoot_serial_port_put (struct VasEBoot_serial_port *port, const int c)
{
port->driver->put (port, c);
}
static inline void
VasEBoot_serial_port_fini (struct VasEBoot_serial_port *port)
{
port->driver->fini (port);
}
/* Set default settings. */
static inline VasEBoot_err_t
VasEBoot_serial_config_defaults (struct VasEBoot_serial_port *port)
{
struct VasEBoot_serial_config config =
{
#ifdef VasEBoot_MACHINE_MIPS_LOONGSON
.speed = 115200,
/* On Loongson machines serial port has only 3 wires. */
.rtscts = 0,
#else
.speed = 9600,
.rtscts = 1,
#endif
.word_len = 8,
.parity = VasEBoot_SERIAL_PARITY_NONE,
.stop_bits = VasEBoot_SERIAL_STOP_BITS_1,
.base_clock = 0
};
return port->driver->configure (port, &config);
}
#if defined(__mips__) || defined (__i386__) || defined (__x86_64__)
void VasEBoot_ns8250_init (void);
char *VasEBoot_serial_ns8250_add_port (VasEBoot_port_t port);
#endif
#ifdef VasEBoot_MACHINE_IEEE1275
void VasEBoot_ofserial_init (void);
#endif
#ifdef VasEBoot_MACHINE_EFI
void
VasEBoot_efiserial_init (void);
#endif
#ifdef VasEBoot_MACHINE_ARC
void
VasEBoot_arcserial_init (void);
const char *
VasEBoot_arcserial_add_port (const char *path);
#endif
struct VasEBoot_serial_port *VasEBoot_serial_find (const char *name);
extern struct VasEBoot_serial_driver VasEBoot_ns8250_driver;
void EXPORT_FUNC(VasEBoot_serial_unregister_driver) (struct VasEBoot_serial_driver *driver);
#ifndef VasEBoot_MACHINE_EMU
extern void VasEBoot_serial_init (void);
extern void VasEBoot_serial_fini (void);
#endif
#endif