1 |
/* |
2 |
* macos_util.h - MacOS definitions/utility functions |
3 |
* |
4 |
* SheepShaver (C) 1997-2002 Christian Bauer and Marc Hellwig |
5 |
* |
6 |
* This program is free software; you can redistribute it and/or modify |
7 |
* it under the terms of the GNU General Public License as published by |
8 |
* the Free Software Foundation; either version 2 of the License, or |
9 |
* (at your option) any later version. |
10 |
* |
11 |
* This program is distributed in the hope that it will be useful, |
12 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
13 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
14 |
* GNU General Public License for more details. |
15 |
* |
16 |
* You should have received a copy of the GNU General Public License |
17 |
* along with this program; if not, write to the Free Software |
18 |
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
19 |
*/ |
20 |
|
21 |
#ifndef MACOS_UTIL_H |
22 |
#define MACOS_UTIL_H |
23 |
|
24 |
#include "cpu_emulation.h" |
25 |
|
26 |
|
27 |
/* |
28 |
* General definitions |
29 |
*/ |
30 |
|
31 |
struct Point { |
32 |
int16 v; |
33 |
int16 h; |
34 |
}; |
35 |
|
36 |
struct Rect { |
37 |
int16 top; |
38 |
int16 left; |
39 |
int16 bottom; |
40 |
int16 right; |
41 |
}; |
42 |
|
43 |
|
44 |
/* |
45 |
* Queues |
46 |
*/ |
47 |
|
48 |
enum { // Queue types |
49 |
dummyType = 0, |
50 |
vType = 1, |
51 |
ioQType = 2, |
52 |
drvQType = 3, |
53 |
evType = 4, |
54 |
fsQType = 5, |
55 |
sIQType = 6, |
56 |
dtQType = 7, |
57 |
nmType = 8 |
58 |
}; |
59 |
|
60 |
enum { // QElem struct |
61 |
qLink = 0, |
62 |
qType = 4, |
63 |
qData = 6 |
64 |
}; |
65 |
|
66 |
enum { // QHdr struct |
67 |
qFlags = 0, |
68 |
qHead = 2, |
69 |
qTail = 6 |
70 |
}; |
71 |
|
72 |
|
73 |
/* |
74 |
* Definitions for Deferred Task Manager |
75 |
*/ |
76 |
|
77 |
enum { // DeferredTask struct |
78 |
dtFlags = 6, |
79 |
dtAddr = 8, |
80 |
dtParam = 12, |
81 |
dtReserved = 16 |
82 |
}; |
83 |
|
84 |
|
85 |
/* |
86 |
* Definitions for Device Manager |
87 |
*/ |
88 |
|
89 |
// Error codes |
90 |
enum { |
91 |
noErr = 0, |
92 |
controlErr = -17, /* I/O System Errors */ |
93 |
statusErr = -18, /* I/O System Errors */ |
94 |
readErr = -19, /* I/O System Errors */ |
95 |
writErr = -20, /* I/O System Errors */ |
96 |
badUnitErr = -21, /* I/O System Errors */ |
97 |
unitEmptyErr = -22, /* I/O System Errors */ |
98 |
openErr = -23, /* I/O System Errors */ |
99 |
closErr = -24, /* I/O System Errors */ |
100 |
abortErr = -27, /* IO call aborted by KillIO */ |
101 |
notOpenErr = -28, /* Driver not open */ |
102 |
dskFulErr = -34, /* disk full */ |
103 |
nsvErr = -35, /* no such volume */ |
104 |
ioErr = -36, /* I/O error (bummers) */ |
105 |
bdNamErr = -37, /* bad name */ |
106 |
fnOpnErr = -38, /* file not open */ |
107 |
eofErr = -39, /* End-of-file encountered */ |
108 |
posErr = -40, /* tried to position to before start of file (r/w) */ |
109 |
tmfoErr = -42, /* too many files open */ |
110 |
fnfErr = -43, /* file not found */ |
111 |
wPrErr = -44, /* diskette is write protected. */ |
112 |
fLckdErr = -45, /* file is locked */ |
113 |
fBsyErr = -47, /* file busy, dir not empty */ |
114 |
dupFNErr = -48, /* duplicate filename already exists */ |
115 |
paramErr = -50, /* error in user parameter list */ |
116 |
rfNumErr = -51, /* bad ioRefNum */ |
117 |
permErr = -54, /* permission error */ |
118 |
nsDrvErr = -56, /* no such driver number */ |
119 |
extFSErr = -58, /* external file system */ |
120 |
noDriveErr = -64, /* drive not installed */ |
121 |
offLinErr = -65, /* r/w requested for an off-line drive */ |
122 |
noNybErr = -66, /* couldn't find 5 nybbles in 200 tries */ |
123 |
noAdrMkErr = -67, /* couldn't find valid addr mark */ |
124 |
dataVerErr = -68, /* read verify compare failed */ |
125 |
badCksmErr = -69, /* addr mark checksum didn't check */ |
126 |
badBtSlpErr = -70, /* bad addr mark bit slip nibbles */ |
127 |
noDtaMkErr = -71, /* couldn't find a data mark header */ |
128 |
badDCksum = -72, /* bad data mark checksum */ |
129 |
badDBtSlp = -73, /* bad data mark bit slip nibbles */ |
130 |
wrUnderrun = -74, /* write underrun occurred */ |
131 |
cantStepErr = -75, /* step handshake failed */ |
132 |
tk0BadErr = -76, /* track 0 detect doesn't change */ |
133 |
initIWMErr = -77, /* unable to initialize IWM */ |
134 |
twoSideErr = -78, /* tried to read 2nd side on a 1-sided drive */ |
135 |
spdAdjErr = -79, /* unable to correctly adjust disk speed */ |
136 |
seekErr = -80, /* track number wrong on address mark */ |
137 |
sectNFErr = -81, /* sector number never found on a track */ |
138 |
fmt1Err = -82, /* can't find sector 0 after track format */ |
139 |
fmt2Err = -83, /* can't get enough sync */ |
140 |
verErr = -84, /* track failed to verify */ |
141 |
memFullErr = -108, |
142 |
dirNFErr = -120 /* directory not found */ |
143 |
}; |
144 |
|
145 |
// Misc constants |
146 |
enum { |
147 |
goodbye = -1, /* heap being reinitialized */ |
148 |
|
149 |
ioInProgress = 1, /* predefined value of ioResult while I/O is pending */ |
150 |
aRdCmd = 2, /* low byte of ioTrap for Read calls */ |
151 |
aWrCmd = 3, /* low byte of ioTrap for Write calls */ |
152 |
asyncTrpBit = 10, /* trap word modifier */ |
153 |
noQueueBit = 9, /* trap word modifier */ |
154 |
|
155 |
dReadEnable = 0, /* set if driver responds to read requests */ |
156 |
dWritEnable = 1, /* set if driver responds to write requests */ |
157 |
dCtlEnable = 2, /* set if driver responds to control requests */ |
158 |
dStatEnable = 3, /* set if driver responds to status requests */ |
159 |
dNeedGoodBye = 4, /* set if driver needs time for performing periodic tasks */ |
160 |
dNeedTime = 5, /* set if driver needs time for performing periodic tasks */ |
161 |
dNeedLock = 6, /* set if driver must be locked in memory as soon as it is opened */ |
162 |
|
163 |
dOpened = 5, /* driver is open */ |
164 |
dRAMBased = 6, /* dCtlDriver is a handle (1) or pointer (0) */ |
165 |
drvrActive = 7, /* driver is currently processing a request */ |
166 |
|
167 |
rdVerify = 64, |
168 |
|
169 |
fsCurPerm = 0, // Whatever is currently allowed |
170 |
fsRdPerm = 1, // Exclusive read |
171 |
fsWrPerm = 2, // Exclusive write |
172 |
fsRdWrPerm = 3, // Exclusive read/write |
173 |
fsRdWrShPerm = 4, // Shared read/write |
174 |
|
175 |
fsAtMark = 0, // At current mark |
176 |
fsFromStart = 1, // Set mark rel to BOF |
177 |
fsFromLEOF = 2, // Set mark rel to logical EOF |
178 |
fsFromMark = 3, // Set mark rel to current mark |
179 |
|
180 |
sony = 0, |
181 |
hard20 = 1 |
182 |
}; |
183 |
|
184 |
enum { /* Large Volume Constants */ |
185 |
kWidePosOffsetBit = 8, |
186 |
kMaximumBlocksIn4GB = 0x007FFFFF |
187 |
}; |
188 |
|
189 |
enum { // IOParam struct |
190 |
ioTrap = 6, |
191 |
ioCmdAddr = 8, |
192 |
ioCompletion = 12, |
193 |
ioResult = 16, |
194 |
ioNamePtr = 18, |
195 |
ioVRefNum = 22, |
196 |
ioRefNum = 24, |
197 |
ioVersNum = 26, |
198 |
ioPermssn = 27, |
199 |
ioMisc = 28, |
200 |
ioBuffer = 32, |
201 |
ioReqCount = 36, |
202 |
ioActCount = 40, |
203 |
ioPosMode = 44, |
204 |
ioPosOffset = 46, |
205 |
ioWPosOffset = 46, // Wide positioning offset when ioPosMode has kWidePosOffsetBit set |
206 |
SIZEOF_IOParam = 50 |
207 |
}; |
208 |
|
209 |
enum { // CntrlParam struct |
210 |
csCode = 26, |
211 |
csParam = 28 |
212 |
}; |
213 |
|
214 |
enum { // DrvSts struct |
215 |
dsTrack = 0, |
216 |
dsWriteProt = 2, |
217 |
dsDiskInPlace = 3, |
218 |
dsInstalled = 4, |
219 |
dsSides = 5, |
220 |
dsQLink = 6, |
221 |
dsQType = 10, |
222 |
dsQDrive = 12, |
223 |
dsQRefNum = 14, |
224 |
dsQFSID = 16, |
225 |
dsTwoSideFmt = 18, |
226 |
dsNewIntf = 19, |
227 |
dsDiskErrs = 20, |
228 |
dsMFMDrive = 22, |
229 |
dsMFMDisk = 23, |
230 |
dsTwoMegFmt = 24 |
231 |
}; |
232 |
|
233 |
enum { // DrvSts2 struct |
234 |
dsDriveSize = 18, |
235 |
dsDriveS1 = 20, |
236 |
dsDriveType = 22, |
237 |
dsDriveManf = 24, |
238 |
dsDriveChar = 26, |
239 |
dsDriveMisc = 28, |
240 |
SIZEOF_DrvSts = 30 |
241 |
}; |
242 |
|
243 |
enum { // DCtlEntry struct |
244 |
dCtlDriver = 0, |
245 |
dCtlFlags = 4, |
246 |
dCtlQHdr = 6, |
247 |
dCtlPosition = 16, |
248 |
dCtlStorage = 20, |
249 |
dCtlRefNum = 24, |
250 |
dCtlCurTicks = 26, |
251 |
dCtlWindow = 30, |
252 |
dCtlDelay = 34, |
253 |
dCtlEMask = 36, |
254 |
dCtlMenu = 38, |
255 |
dCtlSlot = 40, |
256 |
dCtlSlotId = 41, |
257 |
dCtlDevBase = 42, |
258 |
dCtlOwner = 46, |
259 |
dCtlExtDev = 50, |
260 |
dCtlFillByte = 51, |
261 |
dCtlNodeID = 52 |
262 |
}; |
263 |
|
264 |
|
265 |
/* |
266 |
* Definitions for native Device Manager |
267 |
*/ |
268 |
|
269 |
// Registry EntryID |
270 |
struct RegEntryID { |
271 |
uint32 contents[4]; |
272 |
}; |
273 |
|
274 |
// Command codes |
275 |
enum { |
276 |
kOpenCommand = 0, |
277 |
kCloseCommand = 1, |
278 |
kReadCommand = 2, |
279 |
kWriteCommand = 3, |
280 |
kControlCommand = 4, |
281 |
kStatusCommand = 5, |
282 |
kKillIOCommand = 6, |
283 |
kInitializeCommand = 7, /* init driver and device*/ |
284 |
kFinalizeCommand = 8, /* shutdown driver and device*/ |
285 |
kReplaceCommand = 9, /* replace an old driver*/ |
286 |
kSupersededCommand = 10, /* prepare to be replaced by a new driver*/ |
287 |
kSuspendCommand = 11, /* prepare driver to go to sleep*/ |
288 |
kResumeCommand = 12 /* wake up sleeping driver*/ |
289 |
}; |
290 |
|
291 |
// Command kinds |
292 |
enum { |
293 |
kSynchronousIOCommandKind = 0x00000001, |
294 |
kAsynchronousIOCommandKind = 0x00000002, |
295 |
kImmediateIOCommandKind = 0x00000004 |
296 |
}; |
297 |
|
298 |
|
299 |
/* |
300 |
* Definitions for Mixed Mode Manager |
301 |
*/ |
302 |
|
303 |
typedef uint32 ProcInfoType; |
304 |
typedef int8 ISAType; |
305 |
typedef uint16 RoutineFlagsType; |
306 |
typedef long (*ProcPtr)(); |
307 |
typedef uint8 RDFlagsType; |
308 |
|
309 |
struct RoutineRecord { |
310 |
ProcInfoType procInfo; /* calling conventions */ |
311 |
int8 reserved1; /* Must be 0 */ |
312 |
ISAType ISA; /* Instruction Set Architecture */ |
313 |
RoutineFlagsType routineFlags; /* Flags for each routine */ |
314 |
ProcPtr procDescriptor; /* Where is the thing weĠre calling? */ |
315 |
uint32 reserved2; /* Must be 0 */ |
316 |
uint32 selector; /* For dispatched routines, the selector */ |
317 |
}; |
318 |
|
319 |
struct RoutineDescriptor { |
320 |
uint16 goMixedModeTrap; /* Our A-Trap */ |
321 |
int8 version; /* Current Routine Descriptor version */ |
322 |
RDFlagsType routineDescriptorFlags; /* Routine Descriptor Flags */ |
323 |
uint32 reserved1; /* Unused, must be zero */ |
324 |
uint8 reserved2; /* Unused, must be zero */ |
325 |
uint8 selectorInfo; /* If a dispatched routine, calling convention, else 0 */ |
326 |
uint16 routineCount; /* Number of routines in this RD */ |
327 |
RoutineRecord routineRecords[1]; /* The individual routines */ |
328 |
}; |
329 |
|
330 |
#define BUILD_PPC_ROUTINE_DESCRIPTOR(procInfo, procedure) \ |
331 |
{ \ |
332 |
0xAAFE, /* Mixed Mode A-Trap */ \ |
333 |
7, /* version */ \ |
334 |
0, /* RD Flags - not dispatched */ \ |
335 |
0, /* reserved 1 */ \ |
336 |
0, /* reserved 2 */ \ |
337 |
0, /* selector info */ \ |
338 |
0, /* number of routines */ \ |
339 |
{ /* It's an array */ \ |
340 |
{ /* It's a struct */ \ |
341 |
(procInfo), /* the ProcInfo */ \ |
342 |
0, /* reserved */ \ |
343 |
1, /* ISA and RTA */ \ |
344 |
0 | /* Flags - it's absolute addr */\ |
345 |
0 | /* It's prepared */ \ |
346 |
4, /* Always use native ISA */ \ |
347 |
(ProcPtr)(procedure), /* the procedure */ \ |
348 |
0, /* reserved */ \ |
349 |
0 /* Not dispatched */ \ |
350 |
} \ |
351 |
} \ |
352 |
} |
353 |
|
354 |
|
355 |
// Functions |
356 |
extern void MacOSUtilReset(void); |
357 |
extern void Enqueue(uint32 elem, uint32 list); // Enqueue QElem to list |
358 |
extern int FindFreeDriveNumber(int num); // Find first free drive number, starting at "num" |
359 |
extern void MountVolume(void *fh); // Mount volume with given file handle (see sys.h) |
360 |
extern void FileDiskLayout(loff_t size, uint8 *data, loff_t &start_byte, loff_t &real_size); // Calculate disk image file layout given file size and first 256 data bytes |
361 |
extern void *FindLibSymbol(char *lib, char *sym); // Find symbol in shared library |
362 |
extern void InitCallUniversalProc(void); // Init CallUniversalProc() |
363 |
extern long CallUniversalProc(void *upp, uint32 info); // CallUniversalProc() |
364 |
extern uint32 TimeToMacTime(time_t t); // Convert time_t value to MacOS time |
365 |
|
366 |
// Construct four-character-code from string |
367 |
#define FOURCC(a,b,c,d) (((uint32)(a) << 24) | ((uint32)(b) << 16) | ((uint32)(c) << 8) | (uint32)(d)) |
368 |
|
369 |
// Emulator identification codes (4 and 2 characters) |
370 |
const uint32 EMULATOR_ID_4 = 0x62616168; // 'baah' |
371 |
const uint16 EMULATOR_ID_2 = 0x6261; // 'ba' |
372 |
|
373 |
// Test if basic MacOS initializations (of the ROM) are done |
374 |
static inline bool HasMacStarted(void) |
375 |
{ |
376 |
return ReadMacInt32(0xcfc) == FOURCC('W','L','S','C'); // Mac warm start flag |
377 |
} |
378 |
|
379 |
#endif |