mame/src/emu/diserial.h
2016-04-24 12:58:31 +02:00

190 lines
5.7 KiB
C++

// license:BSD-3-Clause
// copyright-holders:Carl, Miodrag Milanovic
#pragma once
#ifndef __EMU_H__
#error Dont include this file directly; include emu.h instead.
#endif
#ifndef __DISERIAL_H__
#define __DISERIAL_H__
// Windows headers are crap, let me count the ways
#undef PARITY_NONE
#undef PARITY_ODD
#undef PARITY_EVEN
#undef PARITY_MARK
#undef PARITY_SPACE
// ======================> device_serial_interface
class device_serial_interface : public device_interface
{
public:
enum
{
/* receive is waiting for start bit. The transition from high-low indicates
start of start bit. This is used to synchronise with the data being transfered */
RECEIVE_REGISTER_WAITING_FOR_START_BIT = 0x01,
/* receive is synchronised with data, data bits will be clocked in */
RECEIVE_REGISTER_SYNCHRONISED = 0x02,
/* set if receive register has been filled */
RECEIVE_REGISTER_FULL = 0x04
};
enum
{
/* register is empty and ready to be filled with data */
TRANSMIT_REGISTER_EMPTY = 0x0001
};
/* parity selections */
/* if all the bits are added in a byte, if the result is:
even -> parity is even
odd -> parity is odd
*/
enum parity_t
{
PARITY_NONE, /* no parity. a parity bit will not be in the transmitted/received data */
PARITY_ODD, /* odd parity */
PARITY_EVEN, /* even parity */
PARITY_MARK, /* one parity */
PARITY_SPACE /* zero parity */
};
enum stop_bits_t
{
STOP_BITS_0,
STOP_BITS_1 = 1,
STOP_BITS_1_5 = 2,
STOP_BITS_2 = 3
};
/* Communication lines. Beware, everything is active high */
enum
{
CTS = 0x0001, /* Clear to Send. (INPUT) Other end of connection is ready to accept data */
RTS = 0x0002, /* Request to Send. (OUTPUT) This end is ready to send data, and requests if the other */
/* end is ready to accept it */
DSR = 0x0004, /* Data Set ready. (INPUT) Other end of connection has data */
DTR = 0x0008, /* Data terminal Ready. (OUTPUT) TX contains new data. */
RX = 0x0010, /* Receive data. (INPUT) */
TX = 0x0020 /* TX = Transmit data. (OUTPUT) */
};
// construction/destruction
device_serial_interface(const machine_config &mconfig, device_t &device);
virtual ~device_serial_interface();
DECLARE_WRITE_LINE_MEMBER(rx_w);
DECLARE_WRITE_LINE_MEMBER(tx_clock_w);
DECLARE_WRITE_LINE_MEMBER(rx_clock_w);
DECLARE_WRITE_LINE_MEMBER(clock_w);
protected:
void set_data_frame(int start_bit_count, int data_bit_count, parity_t parity, stop_bits_t stop_bits);
void receive_register_reset();
void receive_register_update_bit(int bit);
void receive_register_extract();
void set_rcv_rate(const attotime &rate);
void set_tra_rate(const attotime &rate);
void set_rcv_rate(UINT32 clock, int div) { set_rcv_rate((clock && div) ? (attotime::from_hz(clock) * div) : attotime::never); }
void set_tra_rate(UINT32 clock, int div) { set_tra_rate((clock && div) ? (attotime::from_hz(clock) * div) : attotime::never); }
void set_rcv_rate(int baud) { set_rcv_rate(baud ? attotime::from_hz(baud) : attotime::never); }
void set_tra_rate(int baud) { set_tra_rate(baud ? attotime::from_hz(baud) : attotime::never); }
void set_rate(const attotime &rate) { set_rcv_rate(rate); set_tra_rate(rate); }
void set_rate(UINT32 clock, int div) { set_rcv_rate(clock, div); set_tra_rate(clock, div); }
void set_rate(int baud) { set_rcv_rate(baud); set_tra_rate(baud); }
void transmit_register_reset();
void transmit_register_add_bit(int bit);
void transmit_register_setup(UINT8 data_byte);
UINT8 transmit_register_get_data_bit();
UINT8 serial_helper_get_parity(UINT8 data) { return m_serial_parity_table[data]; }
bool is_receive_register_full();
bool is_transmit_register_empty();
bool is_receive_register_synchronized() const { return m_rcv_flags & RECEIVE_REGISTER_SYNCHRONISED; }
bool is_receive_register_shifting() const { return m_rcv_bit_count_received > 0; }
bool is_receive_framing_error() const { return m_rcv_framing_error; }
bool is_receive_parity_error() const { return m_rcv_parity_error; }
UINT8 get_received_char() const { return m_rcv_byte_received; }
virtual void tra_callback() { }
virtual void rcv_callback() { receive_register_update_bit(m_rcv_line); }
virtual void tra_complete() { }
virtual void rcv_complete() { }
// interface-level overrides
virtual void interface_pre_start() override;
// Must be called from device_timer in the underlying device
virtual void device_timer(emu_timer &timer, device_timer_id id, int param, void *ptr);
bool m_start_bit_hack_for_external_clocks;
const char *parity_tostring(parity_t stop_bits);
const char *stop_bits_tostring(stop_bits_t stop_bits);
void register_save_state(save_manager &save, device_t *device);
private:
enum { TRA_TIMER_ID = 10000, RCV_TIMER_ID };
UINT8 m_serial_parity_table[256];
// Data frame
// number of start bits
int m_df_start_bit_count;
// length of word in bits
UINT8 m_df_word_length;
// parity state
UINT8 m_df_parity;
// number of stop bits
UINT8 m_df_stop_bit_count;
// Receive register
/* data */
UINT16 m_rcv_register_data;
/* flags */
UINT8 m_rcv_flags;
/* bit count received */
UINT8 m_rcv_bit_count_received;
/* length of data to receive - includes data bits, parity bit and stop bit */
UINT8 m_rcv_bit_count;
/* the byte of data received */
UINT8 m_rcv_byte_received;
bool m_rcv_framing_error;
bool m_rcv_parity_error;
// Transmit register
/* data */
UINT16 m_tra_register_data;
/* flags */
UINT8 m_tra_flags;
/* number of bits transmitted */
UINT8 m_tra_bit_count_transmitted;
/* length of data to send */
UINT8 m_tra_bit_count;
emu_timer *m_rcv_clock;
emu_timer *m_tra_clock;
attotime m_rcv_rate;
attotime m_tra_rate;
UINT8 m_rcv_line;
int m_tra_clock_state, m_rcv_clock_state;
void tra_edge();
void rcv_edge();
};
#endif /* __DISERIAL_H__ */