-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathdisk.cpp
237 lines (191 loc) · 5.32 KB
/
disk.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
#include <Arduino.h>
#include <memory.h>
#include <hardware.h>
#include <serialio.h>
#include <filer.h>
#include <flash_filer.h>
#include <acia.h>
#include <pia.h>
#include <timed.h>
#include "config.h"
#include "disk.h"
// See https://github.com/jefftranter/6502/blob/master/asm/OSI/diskboot.s for commented disassembly
// of the CEGMON diskboot routine (option 'D')
// See https://github.com/jefftranter/6502/blob/master/asm/OSI/os65dv33.s for commented disassembly of OS65D
//
// Disk geometry is discussed here: https://osiweb.org/computes_gazette/Compute_n020_p136_OS65D_Disk_Routines_Part1.pdf
//
// A disk interface consisted of a 6820 PIA mapped at 0xC000-0xC003 and a 6850 ACIA mapped at 0xC010-0xC011.
// PIA port A reads disk control lines and port B writes them. Data is read a byte at a time through
// the ACIA's data register.
//
// There is a notch at track 0 on the disk, and detection of this is the only way the disk head
// "knows" where it is. This is flagged by bit #7 low on read of DRA.
// A high->low transition on bit #3 of DRB indicates that the head should step in the direction
// given by bit #2: 0xb -> 0x3 means "step out" (increase track number) and 0xb -> 0x7 "step in".
//
// Current status:
// - writing to disk only works on esp8266 (LittleFS with "r+" file mode)
// - 8" disks not tested
// PB commands
#define WRITE_ENABLE 0x01
#define STEP_DIR 0x04
#define STEP_HEAD 0x08
#define FAULT_RESET 0x10
#define SIDE_SELECT 0x20 // 1 = A/B, 0 = C/D
#define LOW_CURRENT 0x40
#define LOAD_HEAD 0x80
// PA indicators (except DRIVE_SELECT)
#define DRIVE1_READY 0x01
#define TRACK0 0x02
#define FAULT 0x04
#define DRIVE2_READY 0x10
#define WRITE_PROT 0x20
#define DRIVE_SELECT 0x40 // 1 = A/C, 0 = B/D
#define INDEX_HOLE 0x80
#define TICK_FREQ 1000 // 1ms
// 5.25" floppy disk
#if USE_DISK == DISK525
#define DISK_TRACKS 40
#define TRACK_SECTORS 9
#define SECTOR_BYTES 256
#define RPM 300
#define T_REV_MS 60000 / RPM // 200 ms
#elif USE_DISK == DISK8
#define DISK_TRACKS 77
#define TRACK_SECTORS 12
#define SECTOR_BYTES 256
#define RPM 360
#define T_REV_MS 60000 / RPM // 166 ms
#endif
#if defined(USE_DISK)
static inline bool is_step_head(uint8_t b) { return !(b & STEP_HEAD); }
static inline bool is_step_in(uint8_t b) { return !(b & STEP_DIR); }
static inline bool is_load_head(uint8_t b) { return !(b & LOAD_HEAD); }
static inline bool is_index_hole(uint8_t b) { return !(b & INDEX_HOLE); }
static inline bool is_write_enabled(uint8_t b) { return !(b & WRITE_ENABLE); }
static inline bool is_side_ab(uint8_t b) { return b & SIDE_SELECT; }
static inline bool is_drive_ac(uint8_t b) { return b & DRIVE_SELECT; }
static inline uint32_t start_offset(int track) { return track * TRACK_SECTORS * SECTOR_BYTES; }
static disk *d;
static void IRAM_ATTR timer_handler() { d->tick(); }
void disk::reset() {
DBG(printf("reset\r\n"));
if (!d) {
d = this;
timer_create(TICK_FREQ, timer_handler);
}
PIA::write_porta_in_bit(DRIVE_SELECT, 1); // ensure drive-1 is selected
drive = &driveA;
track = 0xff;
pos = 0;
}
void IRAM_ATTR disk::tick() {
ticks++;
if (ticks == T_REV_MS)
ticks = 0;
PIA::write_porta_in_bit(INDEX_HOLE, *drive && ticks > 0);
}
void disk::operator=(uint8_t b) {
if (_acc < 0x10)
PIA::write(_acc, b);
else
ACIA::write(_acc, b);
}
void disk::write_portb(uint8_t b) {
DBG(printf("DRB! %02x\r\n", b));
uint8_t dra = PIA::read_porta(), drb = PIA::read_portb();
bool last = (drive == &driveA || drive == &driveC);
if (is_drive_ac(dra))
drive = is_side_ab(b)? &driveA: &driveC;
else
drive = is_side_ab(b)? &driveB: &driveD;
bool curr = (drive == &driveA || drive == &driveC);
if (last != curr) {
track = 0;
seek_start();
}
if (!is_step_head(drb) && is_step_head(b)) {
// track numbers increase inwards
if (is_step_in(b))
track++;
else
track--;
DBG(printf("track: %d\r\n", track));
seek_start();
PIA::write_porta_in_bit(TRACK0, track > 0);
}
if (is_load_head(b))
seek_start();
PIA::write_portb(b);
}
disk::operator uint8_t() {
if (_acc < 0x10)
return PIA::read(_acc);
return ACIA::read(_acc - 0x10);
}
uint8_t disk::read_porta() {
uint8_t dra = PIA::read_porta();
dra |= WRITE_PROT;
if (driveA || driveC)
dra &= ~DRIVE1_READY;
else
dra |= DRIVE1_READY;
if (driveB || driveD)
dra &= ~DRIVE2_READY;
else
dra |= DRIVE2_READY;
DBG(printf("DRA? %02x\r\n", dra));
return dra;
}
uint8_t disk::read_status() {
uint8_t dra = PIA::read_porta();
uint8_t r = dcd | cts;
if (is_index_hole(dra))
return r;
uint8_t b = ACIA::read_status();
b |= r;
DBG(printf("ASR? %02x\r\n", b));
return b;
}
uint8_t disk::read_data() {
uint8_t b = drive->read();
DBG(printf("ADR? %04x %02x\r\n", pos, b));
pos++;
return b;
}
void disk::write(uint8_t b) {
drive->write(b);
pos++;
}
void disk::write_data(uint8_t b) {
uint8_t dra = PIA::read_porta();
if (is_write_enabled(dra)) {
// write header
if (pos == start_offset(track)) {
write(0x43);
write(0x57);
write(16*(track / 10) + (track % 10));
write(0x58);
}
write(b);
}
DBG(printf("ADR! %04x %02x\r\n", pos, b));
}
void disk::write_control(uint8_t b) {
DBG(printf("ACR! %02x\r\n", b));
ACIA::write_control(b);
}
void disk::seek_start() {
pos = start_offset(track);
drive->seek(pos);
}
uint8_t disk_timer::_get(Memory::address a) {
if (a == 0) {
uint8_t s = _state;
_state ^= 0x80;
return s;
}
return 0;
}
#endif