mirror of
https://github.com/holub/mame
synced 2025-05-29 09:03:08 +03:00

This method was necessary long ago when devices didn't have distinct load and create paths, which has since been addressed
192 lines
6.3 KiB
C++
192 lines
6.3 KiB
C++
// license:BSD-3-Clause
|
|
// copyright-holders:F. Ulivi
|
|
/*********************************************************************
|
|
|
|
hp_taco.h
|
|
|
|
HP TApe COntroller (5006-3012)
|
|
|
|
*********************************************************************/
|
|
|
|
#ifndef __HP_TACO_H__
|
|
#define __HP_TACO_H__
|
|
|
|
#include <map>
|
|
|
|
#define MCFG_TACO_IRQ_HANDLER(_devcb) \
|
|
devcb = &hp_taco_device::set_irq_handler(*device , DEVCB_##_devcb);
|
|
|
|
#define MCFG_TACO_FLG_HANDLER(_devcb) \
|
|
devcb = &hp_taco_device::set_flg_handler(*device , DEVCB_##_devcb);
|
|
|
|
#define MCFG_TACO_STS_HANDLER(_devcb) \
|
|
devcb = &hp_taco_device::set_sts_handler(*device , DEVCB_##_devcb);
|
|
|
|
class hp_taco_device : public device_t ,
|
|
public device_image_interface
|
|
{
|
|
public:
|
|
// construction/destruction
|
|
hp_taco_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, UINT32 clock, const char *shortname);
|
|
hp_taco_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock);
|
|
|
|
// static configuration helpers
|
|
template<class _Object> static devcb_base &set_irq_handler(device_t &device, _Object object) { return downcast<hp_taco_device &>(device).m_irq_handler.set_callback(object); }
|
|
template<class _Object> static devcb_base &set_flg_handler(device_t &device, _Object object) { return downcast<hp_taco_device &>(device).m_flg_handler.set_callback(object); }
|
|
template<class _Object> static devcb_base &set_sts_handler(device_t &device, _Object object) { return downcast<hp_taco_device &>(device).m_sts_handler.set_callback(object); }
|
|
|
|
// Register read/write
|
|
DECLARE_WRITE16_MEMBER(reg_w);
|
|
DECLARE_READ16_MEMBER(reg_r);
|
|
|
|
// Flag & status read
|
|
DECLARE_READ_LINE_MEMBER(flg_r);
|
|
DECLARE_READ_LINE_MEMBER(sts_r);
|
|
|
|
// device_image_interface overrides
|
|
virtual bool call_load() override;
|
|
virtual bool call_create(int format_type, util::option_resolution *format_options) override;
|
|
virtual void call_unload() override;
|
|
virtual std::string call_display() override;
|
|
virtual iodevice_t image_type() const override { return IO_MAGTAPE; }
|
|
virtual bool is_readable() const override { return true; }
|
|
virtual bool is_writeable() const override { return true; }
|
|
virtual bool is_creatable() const override { return true; }
|
|
virtual bool must_be_loaded() const override { return false; }
|
|
virtual bool is_reset_on_load() const override { return false; }
|
|
virtual const char *file_extensions() const override;
|
|
|
|
// Tape position, 1 unit = 1 inch / (968 * 1024)
|
|
typedef INT32 tape_pos_t;
|
|
|
|
// Words stored on tape
|
|
typedef UINT16 tape_word_t;
|
|
|
|
protected:
|
|
// device-level overrides
|
|
virtual void device_config_complete() override;
|
|
virtual void device_start() override;
|
|
virtual void device_stop() override;
|
|
virtual void device_reset() override;
|
|
virtual void device_timer(emu_timer &timer, device_timer_id id, int param, void *ptr) override;
|
|
|
|
private:
|
|
// Storage of tracks: mapping from a tape position to word stored there
|
|
typedef std::map<tape_pos_t , tape_word_t> tape_track_t;
|
|
|
|
devcb_write_line m_irq_handler;
|
|
devcb_write_line m_flg_handler;
|
|
devcb_write_line m_sts_handler;
|
|
|
|
// Registers
|
|
UINT16 m_data_reg;
|
|
bool m_data_reg_full;
|
|
UINT16 m_cmd_reg;
|
|
UINT16 m_status_reg;
|
|
UINT16 m_tach_reg;
|
|
tape_pos_t m_tach_reg_ref;
|
|
bool m_tach_reg_frozen;
|
|
UINT16 m_checksum_reg;
|
|
bool m_clear_checksum_reg;
|
|
UINT16 m_timing_reg;
|
|
|
|
// State
|
|
bool m_irq;
|
|
bool m_flg;
|
|
bool m_sts;
|
|
|
|
// Command FSM state
|
|
typedef enum {
|
|
CMD_IDLE,
|
|
CMD_INVERTING,
|
|
CMD_PH0,
|
|
CMD_PH1,
|
|
CMD_PH2,
|
|
CMD_END,
|
|
CMD_STOPPING
|
|
} cmd_state_t;
|
|
cmd_state_t m_cmd_state;
|
|
|
|
// Tape position & motion
|
|
tape_pos_t m_tape_pos;
|
|
attotime m_start_time; // Tape moving if != never
|
|
bool m_tape_fwd;
|
|
bool m_tape_fast;
|
|
|
|
// Timers
|
|
emu_timer *m_tape_timer;
|
|
emu_timer *m_hole_timer;
|
|
emu_timer *m_timeout_timer;
|
|
|
|
// Content of tape tracks
|
|
tape_track_t m_tracks[ 2 ];
|
|
bool m_image_dirty;
|
|
|
|
// Reading & writing
|
|
bool m_tape_wr;
|
|
tape_pos_t m_rw_pos;
|
|
UINT16 m_next_word;
|
|
tape_track_t::iterator m_rd_it;
|
|
bool m_rd_it_valid;
|
|
|
|
// Gap detection
|
|
tape_pos_t m_gap_detect_start;
|
|
|
|
typedef enum {
|
|
ADV_NO_MORE_DATA,
|
|
ADV_CONT_DATA,
|
|
ADV_DISCONT_DATA
|
|
} adv_res_t;
|
|
|
|
void clear_state(void);
|
|
void irq_w(bool state);
|
|
void set_error(bool state);
|
|
bool is_braking(void) const;
|
|
unsigned speed_to_tick_freq(void) const;
|
|
bool pos_offset(tape_pos_t& pos , tape_pos_t offset) const;
|
|
tape_pos_t current_tape_pos(void) const;
|
|
void update_tape_pos(void);
|
|
void update_tach_reg(void);
|
|
void freeze_tach_reg(bool freeze);
|
|
static void ensure_a_lt_b(tape_pos_t& a , tape_pos_t& b);
|
|
tape_pos_t next_hole(void) const;
|
|
attotime time_to_distance(tape_pos_t distance) const;
|
|
attotime time_to_target(tape_pos_t target) const;
|
|
attotime time_to_stopping_pos(void) const;
|
|
bool start_tape_cmd(UINT16 cmd_reg , UINT16 must_be_1 , UINT16 must_be_0);
|
|
void stop_tape(void);
|
|
tape_track_t& current_track(void);
|
|
static tape_pos_t word_length(tape_word_t w);
|
|
static tape_pos_t word_end_pos(const tape_track_t::iterator& it);
|
|
static void adjust_it(tape_track_t& track , tape_track_t::iterator& it , tape_pos_t pos);
|
|
void write_word(tape_pos_t start , tape_word_t word , tape_pos_t& length);
|
|
void write_gap(tape_pos_t a , tape_pos_t b);
|
|
bool just_gap(tape_pos_t a , tape_pos_t b);
|
|
tape_pos_t farthest_end(const tape_track_t::iterator& it) const;
|
|
bool next_data(tape_track_t::iterator& it , tape_pos_t pos , bool inclusive);
|
|
adv_res_t adv_it(tape_track_t::iterator& it);
|
|
attotime fetch_next_wr_word(void);
|
|
attotime time_to_rd_next_word(tape_pos_t& word_rd_pos);
|
|
tape_pos_t min_gap_size(void) const;
|
|
bool next_n_gap(tape_pos_t& pos , tape_track_t::iterator it , unsigned n_gaps , tape_pos_t min_gap);
|
|
bool next_n_gap(tape_pos_t& pos , unsigned n_gaps , tape_pos_t min_gap);
|
|
void clear_tape(void);
|
|
void dump_sequence(tape_track_t::const_iterator it_start , unsigned n_words);
|
|
void save_tape(void);
|
|
bool load_track(tape_track_t& track);
|
|
bool load_tape(void);
|
|
void set_tape_present(bool present);
|
|
attotime time_to_next_hole(void) const;
|
|
attotime time_to_tach_pulses(void) const;
|
|
void terminate_cmd_now(void);
|
|
void set_data_timeout(bool long_timeout);
|
|
void cmd_fsm(void);
|
|
void start_cmd_exec(UINT16 new_cmd_reg);
|
|
bool internal_load(bool is_create);
|
|
};
|
|
|
|
// device type definition
|
|
extern const device_type HP_TACO;
|
|
|
|
#endif /* __HP_TACO_H__ */
|