ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/cebix/BasiliskII/src/AmigaOS/video_amiga.cpp
Revision: 1.23
Committed: 2002-03-10T20:17:14Z (22 years, 8 months ago) by cebix
Branch: MAIN
Changes since 1.22: +26 -31 lines
Log Message:
minor cleanups

File Contents

# User Rev Content
1 cebix 1.1 /*
2     * video_amiga.cpp - Video/graphics emulation, AmigaOS specific stuff
3     *
4 cebix 1.22 * Basilisk II (C) 1997-2002 Christian Bauer
5 cebix 1.1 *
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     #include <exec/types.h>
22     #include <intuition/intuition.h>
23     #include <graphics/rastport.h>
24     #include <graphics/gfx.h>
25 cebix 1.7 #include <cybergraphics/cybergraphics.h>
26 cebix 1.1 #include <dos/dostags.h>
27     #include <devices/timer.h>
28     #include <proto/exec.h>
29     #include <proto/dos.h>
30     #include <proto/intuition.h>
31     #include <proto/graphics.h>
32     #include <proto/Picasso96.h>
33 cebix 1.4 #include <proto/cybergraphics.h>
34 cebix 1.1
35     #include "sysdeps.h"
36 jlachmann 1.9 #include "cpu_emulation.h"
37 cebix 1.1 #include "main.h"
38     #include "adb.h"
39     #include "prefs.h"
40     #include "user_strings.h"
41     #include "video.h"
42    
43 cebix 1.4 #define DEBUG 0
44 cebix 1.1 #include "debug.h"
45    
46    
47     // Display types
48     enum {
49     DISPLAY_WINDOW,
50     DISPLAY_PIP,
51 cebix 1.6 DISPLAY_SCREEN_P96,
52     DISPLAY_SCREEN_CGFX
53 cebix 1.1 };
54    
55     // Global variables
56     static int32 frame_skip;
57     static int display_type = DISPLAY_WINDOW; // See enum above
58     static struct Screen *the_screen = NULL;
59     static struct Window *the_win = NULL;
60     static struct BitMap *the_bitmap = NULL;
61 cebix 1.8 static UWORD *null_pointer = NULL; // Blank mouse pointer data
62     static UWORD *current_pointer = (UWORD *)-1; // Currently visible mouse pointer data
63 cebix 1.1 static LONG black_pen = -1, white_pen = -1;
64     static struct Process *periodic_proc = NULL; // Periodic process
65 jlachmann 1.20 static int window_width, window_height; // width and height for window display
66     static ULONG screen_mode_id; // mode ID for screen display
67 cebix 1.1
68     extern struct Task *MainTask; // Pointer to main task (from main_amiga.cpp)
69    
70    
71     // Amiga -> Mac raw keycode translation table
72     static const uint8 keycode2mac[0x80] = {
73     0x0a, 0x12, 0x13, 0x14, 0x15, 0x17, 0x16, 0x1a, // ` 1 2 3 4 5 6 7
74     0x1c, 0x19, 0x1d, 0x1b, 0x18, 0x2a, 0xff, 0x52, // 8 9 0 - = \ inv 0
75     0x0c, 0x0d, 0x0e, 0x0f, 0x11, 0x10, 0x20, 0x22, // Q W E R T Y U I
76     0x1f, 0x23, 0x21, 0x1e, 0xff, 0x53, 0x54, 0x55, // O P [ ] inv 1 2 3
77     0x00, 0x01, 0x02, 0x03, 0x05, 0x04, 0x26, 0x28, // A S D F G H J K
78     0x25, 0x29, 0x27, 0x2a, 0xff, 0x56, 0x57, 0x58, // L ; ' # inv 4 5 6
79     0x32, 0x06, 0x07, 0x08, 0x09, 0x0b, 0x2d, 0x2e, // < Z X C V B N M
80     0x2b, 0x2f, 0x2c, 0xff, 0x41, 0x59, 0x5b, 0x5c, // , . / inv . 7 8 9
81     0x31, 0x33, 0x30, 0x4c, 0x24, 0x35, 0x75, 0xff, // SPC BSP TAB ENT RET ESC DEL inv
82     0xff, 0xff, 0x4e, 0xff, 0x3e, 0x3d, 0x3c, 0x3b, // inv inv - inv CUP CDN CRT CLF
83     0x7a, 0x78, 0x63, 0x76, 0x60, 0x61, 0x62, 0x64, // F1 F2 F3 F4 F5 F6 F7 F8
84     0x65, 0x6d, 0x47, 0x51, 0x4b, 0x43, 0x45, 0x72, // F9 F10 ( ) / * + HLP
85     0x38, 0x38, 0x39, 0x36, 0x3a, 0x3a, 0x37, 0x37, // SHL SHR CAP CTL ALL ALR AML AMR
86     0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, // inv inv inv inv inv inv inv inv
87     0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, // inv inv inv inv inv inv inv inv
88     0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff // inv inv inv inv inv inv inv inv
89     };
90    
91    
92     // Prototypes
93 jlachmann 1.20 static bool video_open(const video_mode &mode);
94     static void video_close();
95 cebix 1.1 static void periodic_func(void);
96 jlachmann 1.20 static void add_mode(uint32 width, uint32 height, uint32 resolution_id, uint32 bytes_per_row, video_depth depth);
97     static void add_modes(uint32 width, uint32 height, video_depth depth);
98     static ULONG find_mode_for_depth(uint32 width, uint32 height, uint32 depth);
99     static ULONG bits_from_depth(video_depth depth);
100 jlachmann 1.21 static bool is_valid_modeid(int display_type, ULONG mode_id);
101     static bool check_modeid_p96(ULONG mode_id);
102     static bool check_modeid_cgfx(ULONG mode_id);
103 cebix 1.1
104    
105     /*
106     * Initialization
107     */
108    
109 cebix 1.13
110 cebix 1.1 // Open window
111     static bool init_window(int width, int height)
112     {
113     // Set absolute mouse mode
114     ADBSetRelMouseMode(false);
115    
116     // Open window
117     the_win = OpenWindowTags(NULL,
118     WA_Left, 0, WA_Top, 0,
119     WA_InnerWidth, width, WA_InnerHeight, height,
120     WA_SimpleRefresh, TRUE,
121     WA_NoCareRefresh, TRUE,
122     WA_Activate, TRUE,
123     WA_RMBTrap, TRUE,
124     WA_ReportMouse, TRUE,
125     WA_DragBar, TRUE,
126     WA_DepthGadget, TRUE,
127     WA_SizeGadget, FALSE,
128 cebix 1.2 WA_Title, (ULONG)GetString(STR_WINDOW_TITLE),
129 cebix 1.1 TAG_END
130     );
131     if (the_win == NULL) {
132 cebix 1.16 ErrorAlert(STR_OPEN_WINDOW_ERR);
133 cebix 1.1 return false;
134     }
135    
136     // Create bitmap ("height + 2" for safety)
137     the_bitmap = AllocBitMap(width, height + 2, 1, BMF_CLEAR, NULL);
138     if (the_bitmap == NULL) {
139 cebix 1.16 ErrorAlert(STR_NO_MEM_ERR);
140 cebix 1.1 return false;
141     }
142    
143 cebix 1.13 // Add resolution and set VideoMonitor
144 cebix 1.1 VideoMonitor.mac_frame_base = (uint32)the_bitmap->Planes[0];
145    
146     // Set FgPen and BgPen
147     black_pen = ObtainBestPenA(the_win->WScreen->ViewPort.ColorMap, 0, 0, 0, NULL);
148     white_pen = ObtainBestPenA(the_win->WScreen->ViewPort.ColorMap, 0xffffffff, 0xffffffff, 0xffffffff, NULL);
149     SetAPen(the_win->RPort, black_pen);
150     SetBPen(the_win->RPort, white_pen);
151     SetDrMd(the_win->RPort, JAM2);
152     return true;
153     }
154    
155     // Open PIP (requires Picasso96)
156     static bool init_pip(int width, int height)
157     {
158     // Set absolute mouse mode
159     ADBSetRelMouseMode(false);
160    
161     // Open window
162     ULONG error = 0;
163     the_win = p96PIP_OpenTags(
164     P96PIP_SourceFormat, RGBFB_R5G5B5,
165     P96PIP_SourceWidth, width,
166     P96PIP_SourceHeight, height,
167 cebix 1.2 P96PIP_ErrorCode, (ULONG)&error,
168 cebix 1.1 WA_Left, 0, WA_Top, 0,
169     WA_InnerWidth, width, WA_InnerHeight, height,
170     WA_SimpleRefresh, TRUE,
171     WA_NoCareRefresh, TRUE,
172     WA_Activate, TRUE,
173     WA_RMBTrap, TRUE,
174     WA_ReportMouse, TRUE,
175     WA_DragBar, TRUE,
176     WA_DepthGadget, TRUE,
177     WA_SizeGadget, FALSE,
178 cebix 1.2 WA_Title, (ULONG)GetString(STR_WINDOW_TITLE),
179     WA_PubScreenName, (ULONG)"Workbench",
180 cebix 1.1 TAG_END
181     );
182     if (the_win == NULL || error) {
183 cebix 1.16 ErrorAlert(STR_OPEN_WINDOW_ERR);
184 cebix 1.1 return false;
185     }
186    
187     // Find bitmap
188 cebix 1.2 p96PIP_GetTags(the_win, P96PIP_SourceBitMap, (ULONG)&the_bitmap, TAG_END);
189 cebix 1.1
190 cebix 1.13 // Add resolution and set VideoMonitor
191 cebix 1.15 VideoMonitor.mac_frame_base = p96GetBitMapAttr(the_bitmap, P96BMA_MEMORY);
192 jlachmann 1.20
193 cebix 1.1 return true;
194     }
195    
196 cebix 1.6 // Open Picasso96 screen
197     static bool init_screen_p96(ULONG mode_id)
198 cebix 1.1 {
199     // Set relative mouse mode
200     ADBSetRelMouseMode(true);
201    
202 cebix 1.6 // Check if the mode is one we can handle
203 jlachmann 1.21 if (!check_modeid_p96(mode_id))
204     {
205     ErrorAlert(STR_WRONG_SCREEN_FORMAT_ERR);
206     return false;
207     }
208 cebix 1.6
209     // Yes, get width and height
210 jlachmann 1.21 uint32 depth = p96GetModeIDAttr(mode_id, P96IDA_DEPTH);
211 cebix 1.6 uint32 width = p96GetModeIDAttr(mode_id, P96IDA_WIDTH);
212     uint32 height = p96GetModeIDAttr(mode_id, P96IDA_HEIGHT);
213    
214     // Open screen
215     the_screen = p96OpenScreenTags(
216     P96SA_DisplayID, mode_id,
217     P96SA_Title, (ULONG)GetString(STR_WINDOW_TITLE),
218     P96SA_Quiet, TRUE,
219     P96SA_NoMemory, TRUE,
220     P96SA_NoSprite, TRUE,
221     P96SA_Exclusive, TRUE,
222     TAG_END
223     );
224     if (the_screen == NULL) {
225 cebix 1.16 ErrorAlert(STR_OPEN_SCREEN_ERR);
226 cebix 1.1 return false;
227     }
228    
229 cebix 1.6 // Open window
230     the_win = OpenWindowTags(NULL,
231     WA_Left, 0, WA_Top, 0,
232     WA_Width, width, WA_Height, height,
233 jlachmann 1.20 WA_SimpleRefresh, TRUE,
234 cebix 1.6 WA_NoCareRefresh, TRUE,
235     WA_Borderless, TRUE,
236     WA_Activate, TRUE,
237     WA_RMBTrap, TRUE,
238     WA_ReportMouse, TRUE,
239     WA_CustomScreen, (ULONG)the_screen,
240     TAG_END
241     );
242     if (the_win == NULL) {
243 cebix 1.16 ErrorAlert(STR_OPEN_WINDOW_ERR);
244 cebix 1.6 return false;
245     }
246    
247     ScreenToFront(the_screen);
248 cebix 1.13
249     // Add resolution and set VideoMonitor
250 cebix 1.6 VideoMonitor.mac_frame_base = p96GetBitMapAttr(the_screen->RastPort.BitMap, P96BMA_MEMORY);
251 jlachmann 1.20
252 cebix 1.6 return true;
253     }
254    
255     // Open CyberGraphX screen
256     static bool init_screen_cgfx(ULONG mode_id)
257     {
258     // Set relative mouse mode
259     ADBSetRelMouseMode(true);
260 cebix 1.4
261 cebix 1.1 // Check if the mode is one we can handle
262 jlachmann 1.21 if (!check_modeid_cgfx(mode_id))
263     {
264     ErrorAlert(STR_WRONG_SCREEN_FORMAT_ERR);
265     return false;
266     }
267 cebix 1.1
268     // Yes, get width and height
269 jlachmann 1.21 uint32 depth = GetCyberIDAttr(CYBRIDATTR_DEPTH, mode_id);
270 cebix 1.6 uint32 width = GetCyberIDAttr(CYBRIDATTR_WIDTH, mode_id);
271     uint32 height = GetCyberIDAttr(CYBRIDATTR_HEIGHT, mode_id);
272 cebix 1.4
273 cebix 1.1 // Open screen
274 cebix 1.6 the_screen = OpenScreenTags(NULL,
275     SA_DisplayID, mode_id,
276     SA_Title, (ULONG)GetString(STR_WINDOW_TITLE),
277     SA_Quiet, TRUE,
278     SA_Exclusive, TRUE,
279     TAG_END
280     );
281 cebix 1.1 if (the_screen == NULL) {
282 cebix 1.16 ErrorAlert(STR_OPEN_SCREEN_ERR);
283 cebix 1.1 return false;
284     }
285    
286     // Open window
287     the_win = OpenWindowTags(NULL,
288     WA_Left, 0, WA_Top, 0,
289     WA_Width, width, WA_Height, height,
290 jlachmann 1.20 WA_SimpleRefresh, TRUE,
291 cebix 1.1 WA_NoCareRefresh, TRUE,
292     WA_Borderless, TRUE,
293     WA_Activate, TRUE,
294     WA_RMBTrap, TRUE,
295     WA_ReportMouse, TRUE,
296 cebix 1.2 WA_CustomScreen, (ULONG)the_screen,
297 cebix 1.1 TAG_END
298     );
299     if (the_win == NULL) {
300 cebix 1.16 ErrorAlert(STR_OPEN_WINDOW_ERR);
301 cebix 1.1 return false;
302     }
303    
304     ScreenToFront(the_screen);
305 cebix 1.6 static UWORD ptr[] = { 0, 0, 0, 0 };
306     SetPointer(the_win, ptr, 0, 0, 0, 0); // Hide mouse pointer
307 cebix 1.13
308     // Set VideoMonitor
309     ULONG frame_base;
310 cebix 1.6 APTR handle = LockBitMapTags(the_screen->RastPort.BitMap,
311 cebix 1.13 LBMI_BASEADDRESS, (ULONG)&frame_base,
312     TAG_END
313     );
314 cebix 1.6 UnLockBitMap(handle);
315 jlachmann 1.20
316 cebix 1.13 VideoMonitor.mac_frame_base = frame_base;
317 jlachmann 1.20
318 cebix 1.1 return true;
319     }
320    
321     bool VideoInit(bool classic)
322     {
323 jlachmann 1.20 int default_width, default_height, default_depth;
324    
325 cebix 1.8 // Allocate blank mouse pointer data
326     null_pointer = (UWORD *)AllocMem(12, MEMF_PUBLIC | MEMF_CHIP | MEMF_CLEAR);
327     if (null_pointer == NULL) {
328 cebix 1.16 ErrorAlert(STR_NO_MEM_ERR);
329 cebix 1.8 return false;
330     }
331    
332 cebix 1.1 // Read frame skip prefs
333     frame_skip = PrefsFindInt32("frameskip");
334     if (frame_skip == 0)
335     frame_skip = 1;
336    
337     // Get screen mode from preferences
338     const char *mode_str;
339     if (classic)
340     mode_str = "win/512/342";
341     else
342     mode_str = PrefsFindString("screen");
343    
344     // Determine type and mode
345     display_type = DISPLAY_WINDOW;
346 jlachmann 1.20
347     default_width = window_width = 512;
348     default_height = window_height = 384;
349    
350 cebix 1.1 if (mode_str) {
351 jlachmann 1.20 if (sscanf(mode_str, "win/%d/%d", &window_width, &window_height) == 2)
352 cebix 1.1 display_type = DISPLAY_WINDOW;
353 jlachmann 1.20 else if (sscanf(mode_str, "pip/%d/%d", &window_width, &window_height) == 2 && P96Base)
354 cebix 1.1 display_type = DISPLAY_PIP;
355 jlachmann 1.20 else if (sscanf(mode_str, "scr/%08lx", &screen_mode_id) == 1 && (CyberGfxBase || P96Base)) {
356     if (P96Base && p96GetModeIDAttr(screen_mode_id, P96IDA_ISP96))
357 cebix 1.6 display_type = DISPLAY_SCREEN_P96;
358 jlachmann 1.20 else if (CyberGfxBase && IsCyberModeID(screen_mode_id))
359 cebix 1.6 display_type = DISPLAY_SCREEN_CGFX;
360     else {
361 cebix 1.16 ErrorAlert(STR_NO_P96_MODE_ERR);
362 cebix 1.6 return false;
363     }
364     }
365 cebix 1.1 }
366    
367 cebix 1.23 D(bug("display_type %d, window_width %d, window_height %d\n", display_type, window_width, window_height));
368    
369 jlachmann 1.20 // Construct list of supported modes
370     switch (display_type) {
371     case DISPLAY_WINDOW:
372     default_width = window_width;
373     default_height = window_height;
374     default_depth = 1;
375     add_modes(window_width, window_height, VDEPTH_1BIT);
376     break;
377    
378     case DISPLAY_PIP:
379     default_depth = 16;
380     default_width = window_width;
381     default_height = window_height;
382     add_modes(window_width, window_height, VDEPTH_16BIT);
383     break;
384    
385     case DISPLAY_SCREEN_P96:
386     case DISPLAY_SCREEN_CGFX:
387     struct DimensionInfo dimInfo;
388     DisplayInfoHandle handle = FindDisplayInfo(screen_mode_id);
389    
390 cebix 1.23 if (handle == NULL)
391 jlachmann 1.20 return false;
392    
393     if (GetDisplayInfoData(handle, (UBYTE *) &dimInfo, sizeof(dimInfo), DTAG_DIMS, 0) <= 0)
394     return false;
395    
396     default_width = 1 + dimInfo.Nominal.MaxX - dimInfo.Nominal.MinX;
397     default_height = 1 + dimInfo.Nominal.MaxY - dimInfo.Nominal.MinY;
398     default_depth = dimInfo.MaxDepth;
399    
400 cebix 1.23 for (unsigned d=VDEPTH_8BIT; d<=VDEPTH_32BIT; d++) {
401 jlachmann 1.21 ULONG mode_id = find_mode_for_depth(default_width, default_height, bits_from_depth(video_depth(d)));
402    
403     if (is_valid_modeid(display_type, mode_id))
404 jlachmann 1.20 add_modes(default_width, default_height, video_depth(d));
405 cebix 1.23 }
406 jlachmann 1.20 break;
407     }
408    
409     video_init_depth_list();
410    
411     #if DEBUG
412     bug("Available video modes:\n");
413     vector<video_mode>::const_iterator i = VideoModes.begin(), end = VideoModes.end();
414     while (i != end) {
415     bug(" %dx%d (ID %02x), %d colors\n", i->x, i->y, i->resolution_id, 1 << bits_from_depth(i->depth));
416     ++i;
417     }
418     #endif
419    
420     D(bug("VideoInit: def_width=%ld def_height=%ld def_depth=%ld\n", default_width, default_height, default_depth));
421    
422     // Find requested default mode and open display
423     if (VideoModes.size() == 1)
424     return video_open(VideoModes[0]);
425     else {
426     // Find mode with specified dimensions
427     std::vector<video_mode>::const_iterator i, end = VideoModes.end();
428 cebix 1.23 for (i = VideoModes.begin(); i != end; ++i) {
429 jlachmann 1.20 D(bug("VideoInit: w=%ld h=%ld d=%ld\n", i->x, i->y, bits_from_depth(i->depth)));
430     if (i->x == default_width && i->y == default_height && bits_from_depth(i->depth) == default_depth)
431     return video_open(*i);
432 cebix 1.23 }
433 jlachmann 1.20 return video_open(VideoModes[0]);
434     }
435    
436     return true;
437     }
438    
439    
440     static bool video_open(const video_mode &mode)
441     {
442     ULONG depth_bits = bits_from_depth(mode.depth);
443     ULONG ID = find_mode_for_depth(mode.x, mode.y, depth_bits);
444    
445     D(bug("video_open: width=%ld height=%ld depth=%ld ID=%08lx\n", mode.x, mode.y, depth_bits, ID));
446    
447 cebix 1.23 if (ID == INVALID_ID) {
448 jlachmann 1.20 ErrorAlert(STR_NO_VIDEO_MODE_ERR);
449     return false;
450 cebix 1.23 }
451 jlachmann 1.20
452     VideoMonitor.mode = mode;
453    
454 cebix 1.1 // Open display
455     switch (display_type) {
456     case DISPLAY_WINDOW:
457 jlachmann 1.20 if (!init_window(mode.x, mode.y))
458 cebix 1.1 return false;
459     break;
460    
461     case DISPLAY_PIP:
462 jlachmann 1.20 if (!init_pip(mode.x, mode.y))
463 cebix 1.1 return false;
464     break;
465    
466 cebix 1.6 case DISPLAY_SCREEN_P96:
467 jlachmann 1.20 if (!init_screen_p96(ID))
468 cebix 1.6 return false;
469     break;
470    
471     case DISPLAY_SCREEN_CGFX:
472 jlachmann 1.20 if (!init_screen_cgfx(ID))
473 cebix 1.1 return false;
474     break;
475     }
476    
477     // Start periodic process
478     periodic_proc = CreateNewProcTags(
479 cebix 1.2 NP_Entry, (ULONG)periodic_func,
480     NP_Name, (ULONG)"Basilisk II IDCMP Handler",
481 cebix 1.1 NP_Priority, 0,
482     TAG_END
483     );
484     if (periodic_proc == NULL) {
485 cebix 1.16 ErrorAlert(STR_NO_MEM_ERR);
486 cebix 1.1 return false;
487     }
488 jlachmann 1.20
489 cebix 1.1 return true;
490     }
491    
492    
493 jlachmann 1.20 static void video_close()
494 cebix 1.1 {
495     // Stop periodic process
496     if (periodic_proc) {
497     SetSignal(0, SIGF_SINGLE);
498     Signal(&periodic_proc->pr_Task, SIGBREAKF_CTRL_C);
499     Wait(SIGF_SINGLE);
500     }
501    
502     switch (display_type) {
503    
504     case DISPLAY_WINDOW:
505    
506     // Window mode, free bitmap
507     if (the_bitmap) {
508     WaitBlit();
509     FreeBitMap(the_bitmap);
510     }
511    
512     // Free pens and close window
513     if (the_win) {
514     ReleasePen(the_win->WScreen->ViewPort.ColorMap, black_pen);
515     ReleasePen(the_win->WScreen->ViewPort.ColorMap, white_pen);
516    
517     CloseWindow(the_win);
518 jlachmann 1.20 the_win = NULL;
519 cebix 1.1 }
520     break;
521    
522     case DISPLAY_PIP:
523    
524     // Close PIP
525     if (the_win)
526     p96PIP_Close(the_win);
527     break;
528    
529 cebix 1.6 case DISPLAY_SCREEN_P96:
530 cebix 1.1
531     // Close window
532     if (the_win)
533 jlachmann 1.20 {
534 cebix 1.1 CloseWindow(the_win);
535 jlachmann 1.20 the_win = NULL;
536     }
537 cebix 1.1
538     // Close screen
539 cebix 1.4 if (the_screen) {
540 cebix 1.6 p96CloseScreen(the_screen);
541     the_screen = NULL;
542     }
543     break;
544    
545     case DISPLAY_SCREEN_CGFX:
546    
547     // Close window
548     if (the_win)
549 jlachmann 1.20 {
550 cebix 1.6 CloseWindow(the_win);
551 jlachmann 1.20 the_win = NULL;
552     }
553 cebix 1.4
554 cebix 1.6 // Close screen
555     if (the_screen) {
556     CloseScreen(the_screen);
557 cebix 1.4 the_screen = NULL;
558     }
559 cebix 1.1 break;
560     }
561 cebix 1.8
562     // Free mouse pointer
563     if (null_pointer) {
564     FreeMem(null_pointer, 12);
565     null_pointer = NULL;
566     }
567 cebix 1.1 }
568    
569    
570     /*
571 jlachmann 1.20 * Deinitialization
572     */
573    
574     void VideoExit(void)
575     {
576     video_close();
577     }
578    
579    
580     /*
581 cebix 1.1 * Set palette
582     */
583    
584 cebix 1.19 void video_set_palette(uint8 *pal, int num)
585 cebix 1.1 {
586 cebix 1.15 if ((display_type == DISPLAY_SCREEN_P96 || display_type == DISPLAY_SCREEN_CGFX)
587     && !IsDirectMode(VideoMonitor.mode)) {
588 cebix 1.1
589     // Convert palette to 32 bits
590     ULONG table[2 + 256 * 3];
591 cebix 1.17 table[0] = num << 16;
592     table[num * 3 + 1] = 0;
593     for (int i=0; i<num; i++) {
594 cebix 1.1 table[i*3+1] = pal[i*3] * 0x01010101;
595     table[i*3+2] = pal[i*3+1] * 0x01010101;
596     table[i*3+3] = pal[i*3+2] * 0x01010101;
597     }
598    
599     // And load it
600     LoadRGB32(&the_screen->ViewPort, table);
601     }
602     }
603    
604    
605     /*
606 cebix 1.14 * Switch video mode
607     */
608    
609     void video_switch_to_mode(const video_mode &mode)
610 cebix 1.19 {
611 jlachmann 1.20 // Close and reopen display
612     video_close();
613 cebix 1.23 if (!video_open(mode)) {
614 jlachmann 1.20 ErrorAlert(STR_OPEN_WINDOW_ERR);
615     QuitEmulator();
616 cebix 1.23 }
617 cebix 1.19 }
618    
619    
620     /*
621     * Close down full-screen mode (if bringing up error alerts is unsafe while in full-screen mode)
622     */
623    
624     void VideoQuitFullScreen(void)
625 cebix 1.14 {
626     }
627    
628    
629     /*
630 cebix 1.1 * Video message handling (not neccessary under AmigaOS, handled by periodic_func())
631     */
632    
633     void VideoInterrupt(void)
634     {
635     }
636    
637    
638     /*
639     * Process for window refresh and message handling
640     */
641    
642     static __saveds void periodic_func(void)
643     {
644     struct MsgPort *timer_port = NULL;
645     struct timerequest *timer_io = NULL;
646     struct IntuiMessage *msg;
647     ULONG win_mask = 0, timer_mask = 0;
648    
649     // Create message port for window and attach it
650     struct MsgPort *win_port = CreateMsgPort();
651     if (win_port) {
652     win_mask = 1 << win_port->mp_SigBit;
653     the_win->UserPort = win_port;
654 cebix 1.6 ModifyIDCMP(the_win, IDCMP_MOUSEBUTTONS | IDCMP_MOUSEMOVE | IDCMP_RAWKEY | ((display_type == DISPLAY_SCREEN_P96 || display_type == DISPLAY_SCREEN_CGFX) ? IDCMP_DELTAMOVE : 0));
655 cebix 1.1 }
656    
657     // Start 60Hz timer for window refresh
658     if (display_type == DISPLAY_WINDOW) {
659     timer_port = CreateMsgPort();
660     if (timer_port) {
661     timer_io = (struct timerequest *)CreateIORequest(timer_port, sizeof(struct timerequest));
662     if (timer_io) {
663     if (!OpenDevice((UBYTE *)TIMERNAME, UNIT_MICROHZ, (struct IORequest *)timer_io, 0)) {
664     timer_mask = 1 << timer_port->mp_SigBit;
665     timer_io->tr_node.io_Command = TR_ADDREQUEST;
666     timer_io->tr_time.tv_secs = 0;
667     timer_io->tr_time.tv_micro = 16667 * frame_skip;
668     SendIO((struct IORequest *)timer_io);
669     }
670     }
671     }
672     }
673    
674     // Main loop
675     for (;;) {
676    
677     // Wait for timer and/or window (CTRL_C is used for quitting the task)
678     ULONG sig = Wait(win_mask | timer_mask | SIGBREAKF_CTRL_C);
679    
680     if (sig & SIGBREAKF_CTRL_C)
681     break;
682    
683     if (sig & timer_mask) {
684    
685     // Timer tick, update display
686     BltTemplate(the_bitmap->Planes[0], 0, the_bitmap->BytesPerRow, the_win->RPort,
687 cebix 1.13 the_win->BorderLeft, the_win->BorderTop, VideoMonitor.mode.x, VideoMonitor.mode.y);
688 cebix 1.1
689     // Restart timer
690     timer_io->tr_node.io_Command = TR_ADDREQUEST;
691     timer_io->tr_time.tv_secs = 0;
692     timer_io->tr_time.tv_micro = 16667 * frame_skip;
693     SendIO((struct IORequest *)timer_io);
694     }
695    
696     if (sig & win_mask) {
697    
698     // Handle window messages
699     while (msg = (struct IntuiMessage *)GetMsg(win_port)) {
700    
701     // Get data from message and reply
702     ULONG cl = msg->Class;
703     UWORD code = msg->Code;
704     UWORD qualifier = msg->Qualifier;
705     WORD mx = msg->MouseX;
706     WORD my = msg->MouseY;
707     ReplyMsg((struct Message *)msg);
708    
709     // Handle message according to class
710     switch (cl) {
711     case IDCMP_MOUSEMOVE:
712 cebix 1.7 if (display_type == DISPLAY_SCREEN_P96 || display_type == DISPLAY_SCREEN_CGFX)
713 cebix 1.1 ADBMouseMoved(mx, my);
714 cebix 1.8 else {
715 cebix 1.1 ADBMouseMoved(mx - the_win->BorderLeft, my - the_win->BorderTop);
716 cebix 1.8 if (mx < the_win->BorderLeft
717     || my < the_win->BorderTop
718 cebix 1.13 || mx >= the_win->BorderLeft + VideoMonitor.mode.x
719     || my >= the_win->BorderTop + VideoMonitor.mode.y) {
720 cebix 1.8 if (current_pointer) {
721     ClearPointer(the_win);
722     current_pointer = NULL;
723     }
724     } else {
725     if (current_pointer != null_pointer) {
726     // Hide mouse pointer inside window
727     SetPointer(the_win, null_pointer, 1, 16, 0, 0);
728     current_pointer = null_pointer;
729     }
730     }
731     }
732 cebix 1.1 break;
733    
734     case IDCMP_MOUSEBUTTONS:
735     if (code == SELECTDOWN)
736     ADBMouseDown(0);
737     else if (code == SELECTUP)
738     ADBMouseUp(0);
739     else if (code == MENUDOWN)
740     ADBMouseDown(1);
741     else if (code == MENUUP)
742     ADBMouseUp(1);
743     else if (code == MIDDLEDOWN)
744     ADBMouseDown(2);
745     else if (code == MIDDLEUP)
746     ADBMouseUp(2);
747     break;
748    
749     case IDCMP_RAWKEY:
750     if (qualifier & IEQUALIFIER_REPEAT) // Keyboard repeat is done by MacOS
751     break;
752 cebix 1.11 if ((qualifier & (IEQUALIFIER_LALT | IEQUALIFIER_LSHIFT | IEQUALIFIER_CONTROL)) ==
753     (IEQUALIFIER_LALT | IEQUALIFIER_LSHIFT | IEQUALIFIER_CONTROL) && code == 0x5f) {
754 jlachmann 1.9 SetInterruptFlag(INTFLAG_NMI);
755     TriggerInterrupt();
756     break;
757 cebix 1.10 }
758 jlachmann 1.9
759 cebix 1.1 if (code & IECODE_UP_PREFIX)
760     ADBKeyUp(keycode2mac[code & 0x7f]);
761     else
762     ADBKeyDown(keycode2mac[code & 0x7f]);
763     break;
764     }
765     }
766     }
767     }
768    
769     // Stop timer
770     if (timer_io) {
771     if (!CheckIO((struct IORequest *)timer_io))
772     AbortIO((struct IORequest *)timer_io);
773     WaitIO((struct IORequest *)timer_io);
774     CloseDevice((struct IORequest *)timer_io);
775     DeleteIORequest(timer_io);
776     }
777     if (timer_port)
778     DeleteMsgPort(timer_port);
779    
780     // Remove port from window and delete it
781     Forbid();
782     msg = (struct IntuiMessage *)win_port->mp_MsgList.lh_Head;
783     struct Node *succ;
784     while (succ = msg->ExecMessage.mn_Node.ln_Succ) {
785     if (msg->IDCMPWindow == the_win) {
786     Remove((struct Node *)msg);
787     ReplyMsg((struct Message *)msg);
788     }
789     msg = (struct IntuiMessage *)succ;
790     }
791     the_win->UserPort = NULL;
792     ModifyIDCMP(the_win, 0);
793     Permit();
794     DeleteMsgPort(win_port);
795    
796     // Main task asked for termination, send signal
797     Forbid();
798     Signal(MainTask, SIGF_SINGLE);
799     }
800 jlachmann 1.20
801    
802     // Add mode to list of supported modes
803     static void add_mode(uint32 width, uint32 height, uint32 resolution_id, uint32 bytes_per_row, video_depth depth)
804     {
805     video_mode mode;
806     mode.x = width;
807     mode.y = height;
808     mode.resolution_id = resolution_id;
809     mode.bytes_per_row = bytes_per_row;
810     mode.depth = depth;
811    
812     D(bug("Added video mode: w=%ld h=%ld d=%ld\n", width, height, depth));
813    
814     VideoModes.push_back(mode);
815     }
816    
817 cebix 1.23 // Add standard list of modes for given color depth
818 jlachmann 1.20 static void add_modes(uint32 width, uint32 height, video_depth depth)
819     {
820     D(bug("add_modes: w=%ld h=%ld d=%ld\n", width, height, depth));
821    
822     if (width >= 512 && height >= 384)
823     add_mode(512, 384, 0x80, TrivialBytesPerRow(512, depth), depth);
824     if (width >= 640 && height >= 480)
825     add_mode(640, 480, 0x81, TrivialBytesPerRow(640, depth), depth);
826     if (width >= 800 && height >= 600)
827     add_mode(800, 600, 0x82, TrivialBytesPerRow(800, depth), depth);
828     if (width >= 1024 && height >= 768)
829     add_mode(1024, 768, 0x83, TrivialBytesPerRow(1024, depth), depth);
830     if (width >= 1152 && height >= 870)
831     add_mode(1152, 870, 0x84, TrivialBytesPerRow(1152, depth), depth);
832     if (width >= 1280 && height >= 1024)
833     add_mode(1280, 1024, 0x85, TrivialBytesPerRow(1280, depth), depth);
834     if (width >= 1600 && height >= 1200)
835     add_mode(1600, 1200, 0x86, TrivialBytesPerRow(1600, depth), depth);
836     }
837    
838    
839     static ULONG find_mode_for_depth(uint32 width, uint32 height, uint32 depth)
840     {
841     ULONG ID = BestModeID(BIDTAG_NominalWidth, width,
842     BIDTAG_NominalHeight, height,
843     BIDTAG_Depth, depth,
844 jlachmann 1.21 BIDTAG_DIPFMustNotHave, DIPF_IS_ECS | DIPF_IS_HAM | DIPF_IS_AA,
845 jlachmann 1.20 TAG_END);
846    
847     return ID;
848     }
849    
850    
851     static ULONG bits_from_depth(video_depth depth)
852     {
853     int bits = 1 << depth;
854     if (bits == 16)
855     bits = 15;
856     else if (bits == 32)
857     bits = 24;
858    
859     return bits;
860 jlachmann 1.21 }
861    
862    
863     static bool is_valid_modeid(int display_type, ULONG mode_id)
864     {
865     if (INVALID_ID == mode_id)
866     return false;
867    
868 cebix 1.23 switch (display_type) {
869     case DISPLAY_SCREEN_P96:
870     return check_modeid_p96(mode_id);
871     break;
872     case DISPLAY_SCREEN_CGFX:
873     return check_modeid_cgfx(mode_id);
874     break;
875     default:
876     return false;
877     break;
878     }
879 jlachmann 1.21 }
880    
881    
882     static bool check_modeid_p96(ULONG mode_id)
883     {
884     // Check if the mode is one we can handle
885     uint32 depth = p96GetModeIDAttr(mode_id, P96IDA_DEPTH);
886     uint32 format = p96GetModeIDAttr(mode_id, P96IDA_RGBFORMAT);
887    
888 cebix 1.23 D(bug("check_modeid_p96: mode_id=%08lx depth=%ld format=%ld\n", mode_id, depth, format));
889    
890 jlachmann 1.21 if (!p96GetModeIDAttr(screen_mode_id, P96IDA_ISP96))
891     return false;
892    
893     switch (depth) {
894     case 8:
895     break;
896     case 15:
897     case 16:
898     if (format != RGBFB_R5G5B5)
899     return false;
900     break;
901     case 24:
902     case 32:
903     if (format != RGBFB_A8R8G8B8)
904     return false;
905     break;
906     default:
907     return false;
908     }
909    
910     return true;
911     }
912    
913    
914     static bool check_modeid_cgfx(ULONG mode_id)
915     {
916     uint32 depth = GetCyberIDAttr(CYBRIDATTR_DEPTH, mode_id);
917     uint32 format = GetCyberIDAttr(CYBRIDATTR_PIXFMT, mode_id);
918    
919 cebix 1.23 D(bug("check_modeid_cgfx: mode_id=%08lx depth=%ld format=%ld\n", mode_id, depth, format));
920 jlachmann 1.21
921     if (!IsCyberModeID(mode_id))
922     return false;
923    
924     switch (depth) {
925     case 8:
926     break;
927     case 15:
928     case 16:
929     if (format != PIXFMT_RGB15)
930     return false;
931     break;
932     case 24:
933     case 32:
934     if (format != PIXFMT_ARGB32)
935     return false;
936     break;
937     default:
938     return false;
939     }
940    
941     return true;
942 jlachmann 1.20 }