-
Notifications
You must be signed in to change notification settings - Fork 4
/
clem_device.h
335 lines (295 loc) · 8.92 KB
/
clem_device.h
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
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
#ifndef CLEM_DEVICE_H
#define CLEM_DEVICE_H
#include "clem_disk.h"
#include "clem_mmio_types.h"
#include "clem_types.h"
/**
* Device Interface
* These functions emulate the various peripherals (internal and external)
*/
#ifdef __cplusplus
extern "C" {
#endif
void clem_rtc_reset(struct ClemensDeviceRTC *rtc, clem_clocks_duration_t latency);
void clem_rtc_set_clock_time(struct ClemensDeviceRTC *rtc, uint32_t seconds_since_1904);
bool clem_rtc_clear_bram_dirty(struct ClemensDeviceRTC *rtc);
void clem_rtc_set_bram_dirty(struct ClemensDeviceRTC *rtc);
void clem_rtc_command(struct ClemensDeviceRTC *rtc, clem_clocks_time_t ts, unsigned op);
/**
* @brief Resets the timer counters and interrupt flags
*
* @param timer
*/
void clem_timer_reset(struct ClemensDeviceTimer *timer);
/**
* @brief Issues interrupts for the 1 sec and 0.266667 timer
*
* @param timer
* @param delta_us Microsecond increment since last sync
* @param irq_line
* @return uint32_t modified irq_line
*/
void clem_timer_sync(struct ClemensDeviceTimer *timer, uint32_t delta_us);
/**
* @brief Resets the ADB state
*
* @param adb ADB data
*/
void clem_adb_reset(struct ClemensDeviceADB *adb);
/**
* @brief This method is provided for testing and custom implementations.
*
* Normally this is executed from clemens_input to capture input events from
* the host application. This is the 'Device' side of the ADB architecture.
*
* @param adb ADB device data
* @param input The input event from the host OS/application
*/
void clem_adb_device_input(struct ClemensDeviceADB *adb, const struct ClemensInputEvent *input);
/**
* @brief Supplies the emulator with the current state of a toggle key
*
* @param adb
* @param enabled see CLEM_ADB_KEYB_TOGGLE_xxx
*/
void clem_adb_device_key_toggle(struct ClemensDeviceADB *adb, unsigned enabled);
/**
* @brief
*
* @param input
* @return uint8_t*
*/
uint8_t *clem_adb_ascii_from_a2code(unsigned input);
/**
* @brief
*
* @param gameport
* @param clocks
*/
void clem_gameport_sync(struct ClemensDeviceGameport *gameport, struct ClemensClock *clocks);
/**
* @brief Executed frequently enough to emulate the GLU Microcontroller
*
* @param adb ADB device data (1 ms worth of cycles?)
* @param clocks Microsecond increment since last sync
*/
void clem_adb_glu_sync(struct ClemensDeviceADB *adb, struct ClemensDeviceMega2Memory* m2mem,
uint32_t delta_us);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param adb ADB data
* @param ioreg The MMIO register accessed
* @param value The value written to the MMIO register
*/
void clem_adb_write_switch(struct ClemensDeviceADB *adb, uint8_t ioreg, uint8_t value);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param adb ADB data
* @param ioreg The MMIO register accessed
* @param flags Access flags used to determine if the read is a no-op
* @return uint8_t The value read from the MMIO register
*/
uint8_t clem_adb_read_switch(struct ClemensDeviceADB *adb, uint8_t ioreg, uint8_t flags);
/**
* @brief Mouse and Apple II keyboard inputs
*
* @param adb
* @param ioreg
* @param flags
* @return uint8_t
*/
uint8_t clem_adb_read_mega2_switch(struct ClemensDeviceADB *adb, uint8_t ioreg, uint8_t flags);
/**
* @brief
*
* @param adb
* @param key
*/
void clem_adb_clipboard_push_ascii_char(struct ClemensDeviceADB *adb, unsigned char key);
/**
* @brief Resets the Sound state
*
* @param glu sound data
*/
void clem_sound_reset(struct ClemensDeviceAudio *glu);
/**
* @brief Invoked by the host to update the internal audio mixer read pointer
*
* @param glu
* @param consumed
*/
void clem_sound_consume_frames(struct ClemensDeviceAudio *glu, unsigned consumed);
/**
* @brief Executed frequently enough to emulate the GLU Microcontroller
*
* @param glu device data (1 ms worth of cycles?)
* @param clocks Reference clock
*/
void clem_sound_glu_sync(struct ClemensDeviceAudio *glu, struct ClemensClock *clocks);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param glu GLU data
* @param ioreg The MMIO register accessed
* @param value The value written to the MMIO register
*/
void clem_sound_write_switch(struct ClemensDeviceAudio *glu, uint8_t ioreg, uint8_t value);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param glu GLU data
* @param ioreg The MMIO register accessed
* @param flags Access flags used to determine if the read is a no-op
* @return uint8_t The value read from the MMIO register
*/
uint8_t clem_sound_read_switch(struct ClemensDeviceAudio *glu, uint8_t ioreg, uint8_t flags);
/**
* @brief
*
* @param drives
*/
void clem_disk_reset_drives(struct ClemensDriveBay *drives);
/**
* @brief Resets the IWM controller state
*
* @param iwm Device data
*/
void clem_iwm_reset(struct ClemensDeviceIWM *iwm, struct ClemensTimeSpec *tspec);
/**
* @brief
*
* @param iwm
* @param drive
* @return struct ClemensNibbleDisk*
*/
struct ClemensNibbleDisk *clem_iwm_insert_disk(struct ClemensDeviceIWM *iwm,
struct ClemensDrive *drive);
/**
* @brief For 3.5" disks, returns true if ejecting or ejected but not yet finished
*
* @param iwm
* @param drive
* @return unsigned See CLEM_EJECT_DISK_STATUS_XXX
*/
unsigned clem_iwm_eject_disk_in_progress(struct ClemensDeviceIWM *iwm, struct ClemensDrive *drive);
/**
* @brief For 3.5" disks, returns true if ejecting or ejected but not yet finished
*
* @param iwm
* @param drive
* @return unsigned See CLEM_EJECT_DISK_STATUS_XXX
*/
struct ClemensNibbleDisk *clem_iwm_eject_disk(struct ClemensDeviceIWM *iwm,
struct ClemensDrive *drive);
/**
* @brief Executed frequently enough to emulate a IWM controller
*
* Note, due to the emulation rate (down to the microsecond of machine time),
* most emulation will occur on demand via swtiches. This function is here
* to be consistent with our device/controller pattern.
*
* @param iwm device data
* @param drives the IWM managed drives
* @param clock Current clock referemce time
*/
void clem_iwm_glu_sync(struct ClemensDeviceIWM *iwm, struct ClemensDriveBay *drives,
struct ClemensTimeSpec *tspec);
/**
* @brief Returns if the IWM is actively in read or write mode to a specific drive
*
* @param iwm
* @param drives
* @return true
* @return false
*/
bool clem_iwm_is_active(struct ClemensDeviceIWM *iwm, struct ClemensDriveBay *drives);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param iwm IWM data
* @param drives Drives to be managed by the IWM
* @param clock The current reference clock*
* @param ioreg The MMIO register accessed
* @param value The value written to the MMIO register
*/
void clem_iwm_write_switch(struct ClemensDeviceIWM *iwm, struct ClemensDriveBay *drives,
struct ClemensTimeSpec *tspec, uint8_t ioreg, uint8_t value);
/**
* @brief Executed from the memory subsystem for MMIO
*
* @param iwm IWM data
* @param drives Drives to be managed by the IWM
* @param clock The current reference clock
* @param ioreg The MMIO register accessed
* @param flags Access flags used to determine if the read is a no-op
* @return uint8_t The value read from the MMIO register
*/
uint8_t clem_iwm_read_switch(struct ClemensDeviceIWM *iwm, struct ClemensDriveBay *drives,
struct ClemensTimeSpec *tspec, uint8_t ioreg, uint8_t flags);
/**
* @brief
*
* @param mmio
* @param tspec
*/
void clem_iwm_speed_disk_gate(ClemensMMIO *mmio, struct ClemensTimeSpec *tspec);
/**
* @brief
*
* @param iwm
*/
void clem_iwm_debug_start(struct ClemensDeviceIWM *iwm);
/**
* @brief
*
* @param iwm
*/
void clem_iwm_debug_stop(struct ClemensDeviceIWM *iwm);
/**
* @brief
*
* @param unit
* @param unit_count
* @param io_flags
* @param out_phase
* @param delta_ns pass 0 for bus control, non 0 for bus + packet (IWM step)
* @return true
* @return false
*/
bool clem_smartport_bus(struct ClemensSmartPortUnit *unit, unsigned unit_count, unsigned *io_flags,
unsigned *out_phase, clem_clocks_time_t ts, unsigned delta_ns);
/**
* @brief
*
* @param iwm
* @param drive_type
* @param disk
*/
void clem_iwm_insert_disk_old(struct ClemensDeviceIWM *iwm, struct ClemensDrive *drive,
struct ClemensNibbleDisk *disk);
/**
* @brief
*
* @param iwm
* @param drive_type
* @param disk
*/
void clem_iwm_eject_disk_old(struct ClemensDeviceIWM *iwm, struct ClemensDrive *drive,
struct ClemensNibbleDisk *disk);
/**
* @brief For 3.5" drives, issues the eject command and polls until complete
*
* @param iwm
* @param drive_type
* @param disk The output disk once eject is complete
* @return true Disk has ejected
* @return false Disk is still ejecting
*/
bool clem_iwm_eject_disk_async_old(struct ClemensDeviceIWM *iwm, struct ClemensDrive *drive,
struct ClemensNibbleDisk *disk);
#ifdef __cplusplus
}
#endif
#endif